C#与Java的比较(转)
出处:http://www.cnblogs.com/zhucai/archive/2011/02/16/csharp-compare-java.html
这篇文章对C#与Java做一个语言级的对比,方便C#转Java或Java转C#的人有个大致了解。 这里大致用C#3.0与Java6.0做比较。
写完后得知维基百科里有更加全面得多的比较:
http://en.wikipedia.org/wiki/Comparison_of_C_Sharp_and_Java
.NET(C#) | Java | |
---|---|---|
基本类型 | 基本类型 | |
C#中有无符号数,Java没有。 |
委托,事件 | [无] | |
C#中的委托可以认为是方法的类型化,于是可以将方法放在变量里传递。事件是对委托做了一层包装。 Java通过接口来实现C#中委托和事件的功能,可通过匿名类来达到C#中匿名委托的作用(同样也能实现闭包的功能)。 另,C#中也有匿名类,但C#和Java的匿名类刚好各做各的事情:Java中的匿名类只有方法没有数据;C#中的匿名类只有数据没有方法。 |
非托管 | [无] | |
C#可以有非托管代码,可以有指针等。Java没有。 |
索引器 | [无] | |
C#有索引器,可方便容器类实现类似数组的效果。Java没有,Java的容器基本上用put,get,set等方法达到同样效果。 |
属性 | [无] | |
C#的属性通过在内部定义get/set方法,使外部使用时像是在使用变量字段,但其实是在调用get/set方法,以达到透明的封装数据的目的。 Java没有属性的概念。Java通过约定为字段XX添加getXX,setXX方法达到同样的目的。 |
预编译指令 | [无] | |
C#有预编译指令可方便调试,且有ConditionalAttribute来描述方法。Java没有。 |
操作符重载 | [无] | |
C#可重载操作符。Java没有。 |
内部类 | 内部类 | |
Java的内部类可以直接访问外部类的实例成员。 C#的不行。C#的内部类等同于Java的静态内部类。 |
goto、switch | [goto]、switch | |
C#允许用goto。Java的goto是保留关键字,不能使用。但Java允许有标签,在有嵌套循环时可以在continue、break后面跟标签名。 |
enum | enum | |
C#中的枚举是值类型,且其基于数值类型(默认基于int),可设置枚举项对应的数字,不能在其中添加方法等任何其他成员。 Java中的枚举是引用类型(Java除了基本类型外,任何类型都是引用类型),不是基于数值类型。除了不能继承外,它跟普通类差别不大,可以添加成员方法和成员变量等(当然也就可以重写toString方法)。 |
override | @Override | |
C#能被重写的方法必须添加virtual关键字声明为虚方法,派生类重写子类方法时添加override关键字。 Java默认方法都可被重写,派生类和子类方法签名一样时被认为是重写。要声明不能被重写的方法需在方法前加final关键字。重写时可以在方法前添加标注(即C#中的定制特性)@Override,这样一旦此方法找不到被重写的方法时编译器会报错,以防止拼写错误。 |
定制特性 | 标注 | |
C#用中括号[]将定制特性括起来。Java用@打头,后面跟定制特性的名字。 |
泛型 | 泛型 | |
Java中泛型实现使用的擦除机制,为类型参数传入类型并不导致新类型出现,即传入了类型参数后在运行时仍然完全不知道 类型参数的具体类型,它的目的是为了兼容非泛型(所以可以在泛型和非泛型之间隐式转换,会有编译警告但不会有编译错误,这当然其实并不安全);这同时衍生 了一系列问题:不能定义泛型类型参数的数组如T[],不能通过new T()的方式实例化泛型,等。 Java的泛型不支持值类型(使用的话会被自动包装成引用类型)。 |
C#的泛型在类型参数传入类型后会产生一个新类型(虽然CLR的优化机制会使引用类型共享同样的代码),可以在运行时得到类型参数的类型信息。可以定义泛型数组,可以添加约束使其可以new。C#的泛型可以使用值类型(不会被装箱)。 |
参数引用传递 | [无] | |
C#允许使用关键字out,ref显式指定参数传递方式为引用传递。 Java只有值传递。 |
@字符串 | [无] | |
C#在写字符串时可以在引号前加个@符号来取消/的转义作用。 Java没有。 |
?? | [无] | |
C#的??二元操作符当前面的表达式不为null时返回前面表达式的值,前面表达式为null时返回后面表达式的值。 Java没有。 |
using | import | |
C#可以用using为命名空间或类指定别名。(using还有Dispose的使用方式,与命名空间无关) Java的import可以引入类或包(即C#的命名空间),static import可以引入类的成员。 |
初始化 | 初始化 | |
C#调用基类构造函数的语法为: SubClass() : base() { } Java调用基类构造函数的语法为: SubClass(){ super(); } C#和Java都可以用类似的语法调用同一个类的其他构造函数。(分别将base和super换成this) |
interface | interface | |
Java的接口内允许有内部类、静态字段等。 C#不允许。 |
readonly,const | final | |
C#的const是绝对的常量,必须在声明语句中同时赋值,只有数值、枚举和String可以声明为const。const的值会内联到各个使用的地方。 C#的readonly表示变量在构造函数执行完之后是不能再变化的。它只约束变量本身,而无法约束变量引用(如果它是引用类型或者有成员是引用类型)的对象。 |
[无] | throws | |
Java在可能抛出异常时,除了RuntimeException(包括派生类),都要么捕获,要么在方法声明中用throws关键字声明出来表示继续抛出。 C#没有采用这种强制处理机制。 |
功能相同但语法有差异的 | ||
namespace == package (Java的package对文件结构也有要求;C#没有) |
这个列表里,Java比C#更漂亮的地方基本上只有一处:枚举。Java的枚举更高层一些,更灵活。但内存代价比C#的枚举要高,这可能就是Android里仍然使用常量而不是枚举的原因吧。 所以就从这次比较来讲,C#几乎完胜Java,而C#的新特性像完美的类型推断、动态编程特性、Lambda表达式、LINQ等等这里都没有列入比较。
当然,.NET和Java两大体系的比较,语言只是一个方面,还有平台、IDE、开源等其他很多方面,这里就不说了。
出处:http://www.cnblogs.com/zhucai/archive/2011/02/16/csharp-compare-java.html
这篇文章对C#与Java做一个语言级的对比,方便C#转Java或Java转C#的人有个大致了解。 这里大致用C#3.0与Java6.0做比较。
写完后得知维基百科里有更加全面得多的比较:
http://en.wikipedia.org/wiki/Comparison_of_C_Sharp_and_Java
.NET(C#) | Java | |
---|---|---|
基本类型 | 基本类型 | |
C#中有无符号数,Java没有。 |
委托,事件 | [无] | |
C#中的委托可以认为是方法的类型化,于是可以将方法放在变量里传递。事件是对委托做了一层包装。 Java通过接口来实现C#中委托和事件的功能,可通过匿名类来达到C#中匿名委托的作用(同样也能实现闭包的功能)。 另,C#中也有匿名类,但C#和Java的匿名类刚好各做各的事情:Java中的匿名类只有方法没有数据;C#中的匿名类只有数据没有方法。 |
非托管 | [无] | |
C#可以有非托管代码,可以有指针等。Java没有。 |
索引器 | [无] | |
C#有索引器,可方便容器类实现类似数组的效果。Java没有,Java的容器基本上用put,get,set等方法达到同样效果。 |
属性 | [无] | |
C#的属性通过在内部定义get/set方法,使外部使用时像是在使用变量字段,但其实是在调用get/set方法,以达到透明的封装数据的目的。 Java没有属性的概念。Java通过约定为字段XX添加getXX,setXX方法达到同样的目的。 |
预编译指令 | [无] | |
C#有预编译指令可方便调试,且有ConditionalAttribute来描述方法。Java没有。 |
操作符重载 | [无] | |
C#可重载操作符。Java没有。 |
内部类 | 内部类 | |
Java的内部类可以直接访问外部类的实例成员。 C#的不行。C#的内部类等同于Java的静态内部类。 |
goto、switch | [goto]、switch | |
C#允许用goto。Java的goto是保留关键字,不能使用。但Java允许有标签,在有嵌套循环时可以在continue、break后面跟标签名。 |
enum | enum | |
C#中的枚举是值类型,且其基于数值类型(默认基于int),可设置枚举项对应的数字,不能在其中添加方法等任何其他成员。 Java中的枚举是引用类型(Java除了基本类型外,任何类型都是引用类型),不是基于数值类型。除了不能继承外,它跟普通类差别不大,可以添加成员方法和成员变量等(当然也就可以重写toString方法)。 |
override | @Override | |
C#能被重写的方法必须添加virtual关键字声明为虚方法,派生类重写子类方法时添加override关键字。 Java默认方法都可被重写,派生类和子类方法签名一样时被认为是重写。要声明不能被重写的方法需在方法前加final关键字。重写时可以在方法前添加标注(即C#中的定制特性)@Override,这样一旦此方法找不到被重写的方法时编译器会报错,以防止拼写错误。 |
定制特性 | 标注 | |
C#用中括号[]将定制特性括起来。Java用@打头,后面跟定制特性的名字。 |
泛型 | 泛型 | |
Java中泛型实现使用的擦除机制,为类型参数传入类型并不导致新类型出现,即传入了类型参数后在运行时仍然完全不知道 类型参数的具体类型,它的目的是为了兼容非泛型(所以可以在泛型和非泛型之间隐式转换,会有编译警告但不会有编译错误,这当然其实并不安全);这同时衍生 了一系列问题:不能定义泛型类型参数的数组如T[],不能通过new T()的方式实例化泛型,等。 Java的泛型不支持值类型(使用的话会被自动包装成引用类型)。 |
C#的泛型在类型参数传入类型后会产生一个新类型(虽然CLR的优化机制会使引用类型共享同样的代码),可以在运行时得到类型参数的类型信息。可以定义泛型数组,可以添加约束使其可以new。C#的泛型可以使用值类型(不会被装箱)。 |
参数引用传递 | [无] | |
C#允许使用关键字out,ref显式指定参数传递方式为引用传递。 Java只有值传递。 |
@字符串 | [无] | |
C#在写字符串时可以在引号前加个@符号来取消/的转义作用。 Java没有。 |
?? | [无] | |
C#的??二元操作符当前面的表达式不为null时返回前面表达式的值,前面表达式为null时返回后面表达式的值。 Java没有。 |
using | import | |
C#可以用using为命名空间或类指定别名。(using还有Dispose的使用方式,与命名空间无关) Java的import可以引入类或包(即C#的命名空间),static import可以引入类的成员。 |
初始化 | 初始化 | |
C#调用基类构造函数的语法为: SubClass() : base() { } Java调用基类构造函数的语法为: SubClass(){ super(); } C#和Java都可以用类似的语法调用同一个类的其他构造函数。(分别将base和super换成this) |
interface | interface | |
Java的接口内允许有内部类、静态字段等。 C#不允许。 |
readonly,const | final | |
C#的const是绝对的常量,必须在声明语句中同时赋值,只有数值、枚举和String可以声明为const。const的值会内联到各个使用的地方。 C#的readonly表示变量在构造函数执行完之后是不能再变化的。它只约束变量本身,而无法约束变量引用(如果它是引用类型或者有成员是引用类型)的对象。 |
[无] | throws | |
Java在可能抛出异常时,除了RuntimeException(包括派生类),都要么捕获,要么在方法声明中用throws关键字声明出来表示继续抛出。 C#没有采用这种强制处理机制。 |
功能相同但语法有差异的 | ||
namespace == package (Java的package对文件结构也有要求;C#没有) |
这个列表里,Java比C#更漂亮的地方基本上只有一处:枚举。Java的枚举更高层一些,更灵活。但内存代价比C#的枚举要高,这可能就是Android里仍然使用常量而不是枚举的原因吧。 所以就从这次比较来讲,C#几乎完胜Java,而C#的新特性像完美的类型推断、动态编程特性、Lambda表达式、LINQ等等这里都没有列入比较。
当然,.NET和Java两大体系的比较,语言只是一个方面,还有平台、IDE、开源等其他很多方面,这里就不说了。
以上是 C#与Java的比较(转) 的全部内容, 来源链接: utcz.com/z/390553.html