1、Java.NET(C#)基本类型 基本类型C#中有无符号数,Java 没有。C#中有值类型,且可自己定义值类型的结构体(struct)。Java 中的基本类型(或叫基元类型) 即为值类型,但 Java 没有结构体,所以不能自定义值类型。C#中的值类型(包括所有基本类型)间接继承自 Object,有自己的方法可以调用;Java 中的值类型(即基本类型)不继承自Object,只是简单的数据,没有方法可以调用。C#中 int 等同于 System.Int32,是值类型; bool 等同于 System.Boolean;等。Java 中 int 是基本类型,是值类型,而 Integer 是引用类型,I
2、nteger 是 int 的包装器,int 自身没有方法,Integer 有一些方法;int 与Integer 之间可隐式转换(导致装箱和拆箱),但当 Integer 值为 null 的时候会在运行时抛出异常。boolean 等类似。Java 中的 int 与 Integer 的对应在 C#中类似 int 和 Nullable的对应,它们的后者都是前者的包装,且后者可以等于 null。但Nullable实际上仍然是值类型的(所以仍然很轻量级) ,所以从内存上讲 C#中 int 和 Object 的对应更接近 Java 的对应一些。C#中 Nullable到 int 的转换必须显式进行,因为 N
3、ullable中的值为 null 时会引发运行时异常。其他基本类型与之类似。委托,事件 无C#中的委托可以认为是方法的类型化,于是可以将方法放在变量里传递。事件是对委托做了一层包装。Java 通过接口来实现 C#中委托和事件的功能,可通过匿名类来达到 C#中匿名委托的作用(同样也能实现闭包的功能)。另,C# 中也有匿名类,但 C#和 Java 的匿名类刚好各做各的事情:Java 中的匿名类只有方法没有数据;C# 中的匿名类只有数据没有方法。非托管 无C#可以有非托管代码,可以有指针等。Java 没有。索引器 无C#有索引器,可方便容器类实现类似数组的效果。Java 没有,Java 的容器基本上
4、用 put,get,set 等方法达到同样效果。属性 无C#的属性通过在内部定义 get/set 方法,使外部使用时像是在使用变量字段,但其实是在调用 get/set 方法,以达到透明的封装数据的目的。Java 没有属性的概念。Java 通过约定为字段 XX 添加 getXX,setXX 方法达到同样的目的。预编译指令 无C#有预编译指令可方便调试,且有 ConditionalAttribute 来描述方法。 Java 没有。操作符重载 无C#可重载操作符。Java 没有。Java 自己重载了 String 的+和+=,但没有重载= ,这是我这段时间犯的最多的错误。C#中 String 的=是
5、比较值相等,Java 中=是Object 的默认行为:比较引用相等,要比较值相等得用 equals 方法。(这么多年编程以来,我似乎从来没有遇到过要比较两个字符串变量的引用相等。对于比较值相等来讲,=符号比 equals 方法调用看上去优雅得多,况且方法调用还得注意空指针的情况)内部类 内部类Java 的内部类可以直接访问外部类的实例成员。C#的不行。C# 的内部类等同于 Java 的静态内部类。goto、switch goto、switchC#允许用 goto。Java 的 goto 是保留关键字,不能使用。但 Java 允许有标签,在有嵌套循环时可以在 continue、break 后面跟
6、标签名。C#的 switch 可以使用 long、String;Java 不可以。Java 的 switch 中的 case 子句在后面没有跟 break 的情况下直接跳到下一个 case 子句;C#中只有在前一个 case 没有任何代码的情况下才允许不写 break 直接跳到下一个 case,C#中可以通过 goto 跳转到另一 case。enum enumC#中的枚举是值类型,且其基于数值类型(默认基于 int),可设置枚举项对应的数字,不能在其中添加方法等任何其他成员。Java 中的枚举是引用类型(Java 除了基本类型外,任何类型都是引用类型),不是基于数值类型。除了不能继承外,它跟普
7、通类差别不大,可以添加成员方法和成员变量等(当然也就可以重写 toString 方法)。C#和 Java 的枚举都可以用于 switch。可以将 C#的枚举作为数值看待而直接进行位运算,因此可以在一个变量中存储多个位标记。Java 的枚举跟数值没有直接关系,因此不能直接这么用。Java 用 EnumSet 来存储枚举标志,不需要直接使用位运算,更远离底层。override OverrideC#能被重写的方法必须添加 virtual 关键字声明为虚方法,派生类重写子类方法时添加 override 关键字。Java 默认方法都可被重写,派生类和子类方法签名一样时被认为是重写。要声明不能被重写的方法
8、需在方法前加 final 关键字。重写时可以在方法前添加标注(即 C#中的定制特性)Override,这样一旦此方法找不到被重写的方法时编译器会报错,以防止拼写错误。定制特性 标注C#用中括号将定制特性括起来。Java 用打头,后面跟定制特性的名字。泛型 泛型Java 中泛型实现使用的擦除机制,为类型参数传入类型并不导致新类型出现,即传入了类型参数后在运行时仍然完全不知道类型参数的具体类型,它的目的是为了兼容非泛型(所以可以在泛型和非泛型之间隐式转换,会有编译警告但不会有编译错误,这当然其实并不安全);这同时衍生了一系列问题:不能定义泛型类型参数的数组如 T,不能通过 new T()的方式实例
9、化泛型,等。Java 的泛型不支持值类型(使用的话会被自动包装成引用类型)。C#的泛型在类型参数传入类型后会产生一个新类型(虽然 CLR 的优化机制会使引用类型共享同样的代码 ),可以在运行时得到类型参数的类型信息。可以定义泛型数组,可以添加约束使其可以 new。C#的泛型可以使用值类型( 不会被装箱)。对于 Java 的泛型,简单的讲,它的好处只在编译时,运行时没有任何泛型的意义。当你在使用已有的泛型类时,这通常能满足要求;但如果你要自己定义泛型类,那你得知道它有多少你觉得它应该可以但事实上不可以的事情。参数引用传递 无C#允许使用关键字 out,ref 显式指定参数传递方式为引用传递。Ja
10、va 只有值传递。字符串 无C#在写字符串时可以在引号前加个符号来取消/的转义作用。Java 没有。? 无C#的 ?二元操作符当前面的表达式不为 null 时返回前面表达式的值,前面表达式为 null 时返回后面表达式的值。Java 没有。using importC#可以用 using 为命名空间或类指定别名。(using 还有 Dispose 的使用方式,与命名空间无关)Java 的 import 可以引入类或包(即 C#的命名空间),static import 可以引入类的成员。初始化 初始化C#调用基类构造函数的语法为:SubClass() : base() Java 调用基类构造函数的
11、语法为:SubClass()super();C#和 Java 都可以用类似的语法调用同一个类的其他构造函数。(分别将 base 和 super 换成 this)Java 有代码块概念,会在构造函数之前执行(基类的构造函数之后)。在成员变量声明时赋值,Java 允许其赋值表达式中引用前面声明的另一个变量,如:private int x = 1;private int y = x + 10;这里变量 y 的赋值语句有变量 x。C#不允许这样做。interface interfaceJava 的接口内允许有内部类、静态字段等。C#不允许。readonly,const finalC#的 const 是
12、绝对的常量,必须在声明语句中同时赋值,只有数值、枚举和 String 可以声明为 const。const 的值会内联到各个使用的地方。C#的 readonly 表示变量在构造函数执行完之后是不能再变化的。它只约束变量本身,而无法约束变量引用( 如果它是引用类型或者有成员是引用类型)的对象。Java 中的 final(在约束变量的时候)看上去更像 readonly。但 C#的 readonly 和 const 有个区别,readonly 的 int 是不能作为 switch 的 case 语句的,const 的可以。而 Java 的 final 则是:有时候可以有时候不可以-编译时可以得到明确值
13、的可以,反之不可以。如:final int x = 1; / 这个可以final int y = new Random().nextInt(); / 这个不可以那么可以理解为:编译时能得到明确值的时候,final 等同于 C#的 const(不清楚 Java 在这个情况下是否会内联,估计不会);编译时无法得到明确值的时候,final 等同于 C#的 readonly。无 throwsJava 在可能抛出异常时,除了 RuntimeException(包括派生类) ,都要么捕获,要么在方法声明中用 throws 关键字声明出来表示继续抛出。C#没有采用这种强制处理机制。功能相同但语法有差异的na
14、mespace = package (Java 的 package 对文件结构也有要求;C#没有)internal = 默认 (Java 中不写访问修饰符即表示访问权限是 package;C#默认是 private。C#的 internal protected 在 Java 中没有。)lock = synchronized (Java 中 synchronized 可以修饰方法,C#可以用定制特性MethodImplAttribute(MethodImplOptions.Synchronized)达到同样效果): = extends,implementsbase = superis = ins
15、tanceof (C#有 as,Java 没有)typeof = .classSerializableAttribute定制特性 = Serializable 接口NonSerializedAttribute定制特性 = transientparams = . (可变数目参数 )这个列表里,Java 比 C#更漂亮的地方基本上只有一处:枚举。Java 的枚举更高层一些,更灵活。但内存代价比 C#的枚举要高,这可能就是 Android 里仍然使用常量而不是枚举的原因吧。所以就从这次比较来讲,C#几乎完胜 Java,而 C#的新特性像完美的类型推断、动态编程特性、Lambda 表达式、LINQ 等等这里都没有列入比较。当然,.NET 和 Java 两大体系的比较,语言只是一个方面,还有平台、IDE、开源等其他很多方面,这里就不说了。