java与c#的区别、两者有什么不同?

Java 的设计者是因为讨厌C++的复杂,于是Java 非常简洁,GC 也让内存管理非常方便,C# 是看中了Java 的GC,和虚拟机技术,希望把微软的几大语言集成到.NET 上来。 因此C#从语言上来讲并不简单甚至可以算的上复杂。

两种语言的设计思路也不一样,Java 是编译解释语言,C#是编译然后编译运行语言。Java 没有委托,C# 有委托。Java 倾向于用Interface 实现委托的功能,而 在C# 中,Abstract Class 比Interface 发挥了更大功能。

Java 遵循camel 命名规则,C#遵循Pascal 命名规则。但是如今越来越多的Java 人开始使用C#, 同时也把camel 命名规则带到了C#,这可能会让C#代码越来越难读。为什么当初C#为什么不遵循camel 呢? 我看不出camel 命名规则有什么不好的。

一. 类名.this 与内部类

在 java 中,经常看到类似类名.this 的用法,this 就是当前对象实例,为什么前面会出现类名呢?对此 C# 程序员会很纳闷。

在 Java 中,很多地方使用到了内部类,甚至可以在在内部类中访问外部类中的成员,这个时候,在内部类中使用 this 的时候,就会出现 this 到底是谁的问题,到底是表示内部类的当前对象实例,还是外部类的当前对象实例问题。

在 Java 中,通过在 this 的前面加上外部类的类名,表示在内部类中使用外部类的当前对象实例。

我们看下面的一个例子。

代码如下:

package core.sisheng;

// 外部类定义
public class OuterClass {

// 内部类定义
    private class InnerClass
    {
        // 内部类中没有定义 id 成员,这里访问外部类中的成员
        public int getId(){    return OuterClass.this.id;    }
        public void setId(int id) { OuterClass.this.id = id;}

// 内部类中定义了 name 成员,直接访问内部类中的成员, 默认 this 访问当前类中的成员
        private String name;
        public String getName() { return this.name;}
        // 可以在 this 的前面加上一个内部类的名称
        public void setName(String name) { InnerClass.this.name = name;}

// 内部类中也可以访问外部类中同名的成员,需要加上外部类的名称
        public String getOuterName() { return OuterClass.this.name;}
        public void setOuterName(String name) { OuterClass.this.name = name;}

@Override
        public String toString()
        {
            return "Id: " + this.getId() + ", Inner Name: " + this.getName() + ", Outer Name: " + this.getOuterName();
        }
    }

// 外部类中定义的成员 id 和 name
    private int id;
    private String name;

private InnerClass innerInstance;
    public OuterClass()
    {
        this.innerInstance = new InnerClass();
        this.innerInstance.setId(20);
        this.innerInstance.setName("Tom");
        this.innerInstance.setOuterName("Alice");
    }

public String toString()
    {
        return this.innerInstance.toString();
    }   
}

在C#中,类区分为嵌套类和非嵌套类,前者是声明在其他数据类型内部的类。后者是直接定义在某一个命名空间的类。C# 中很少定义嵌套类。

非内嵌类只允许使用public和internal的访问控制,而内置类则允许使用所有的五种访问控制符,private, protected , internal protected,public和internal。内部类也可以访问外部类的所有方法,包括instance方法和private方法,但是需要显式 的传递一个外部类的实例。

C#中的内部类能够使用外部类定义的类型和静态方法,但是不能直接使用外部类的实例方法,因此,不存在上面的问题。

在C#中,外部类对于内部类的作用更像是一个命名空间,只要访问控制允许,就可以使用下面的方法创建内部类对象实例。

OuterClass.InnerClass  obj = new OuterClass.InnerClass();这个实例与外部类的任何实例没有任何直接的关系。类似于Java中的静态内部类。

二、类名.class 与类型

在 java 中还经常看到类名.class 的用法,这个用法相当于  C# 中的 typeof( 类名 ),用来获取类型的类型对象实例引用。

java中,每个class都有一个相应的Class对象,当编写好一个类,编译完成后,在生成的.class文件中,就产生一个Class对象,用来表示这个类的类型信息。获得Class实例的三种方式:

通过调用对象实例的 getClass() 方法获取该对象的Class实例。

使用Class的静态方法forName(),用类的名字获取一个Class实例。Class.forName(xxx.xx.xx) 返回的是一个类, 作用是要求JVM查找并加载指定的类,也就是说JVM会执行该类的静态代码段。

类名.calss的方式获取Class实例,对基本数据类型的封装类,还可以采用.TYPE来获取对应的基本数据类型的Class实例。

C# 中获取类型对象实例的方式更加简单明确一些。

通过调用数据实例的 GetType() 方法获取,这个方法继承自Object,所以C#中任何对象都具有GetType()方法,x.GetType(),其中x为变量名。

typeof(x)中的x,必须是具体的类名、类型名称等,不可以是变量名称。

通过 System.Type 的静态方法 System.Type.GetType()。

 三、匿名类

在 java 中,匿名类的使用也比较多,比如在 Android 中,实现按钮的监听,经常会看到类似这样的代码。

代码如下:

this.listener0 = new OnClickListener() {

@Override
    public void onClick(View arg0) {
        Intent intent = new Intent( MainActivity.this, ActivityFrameLayout.class);
        setTitle("FrameLayout");
        startActivity( intent );
    }
};

在这里,OnClickListenter 实际上是一个接口,接口能用来创建对象实例吗?当然不能。

所以,java 在这里自动创建一个实现接口的匿名类,我们创建出来的实际上就是这个匿名类的对象实例。

这样做的好处就是我们没有必须再定义一个只使用一次的类,然后再通过这个类创建对象实例,简化了程序的开发。

比如,我们有下面的一个接口。


代码如下:

public interface myInterface {
    void onClick();
}

就可以通过接口创建一个实现接口的匿名类的对象实例,然后使用这个对象实例。

代码如下:

myInterface instance = new myInterface(){
    @Override
    public void onClick()
    {
        System.out.println("Clicked!");
    }
};

instance.onClick();在 C# 中,我们根本就不会使用这种形式,通过委托,可以非常简单地实现同样的功能。

注意,java 中是没有委托的。

如果我们输出一下这个实例的类型,你会看到这个匿名类的实际类型的。

代码如下:

System.out.println( instance.getClass());
// class core.sisheng.Study1$1四、属性(Property)

属性的概念对大家来说应该是很熟悉的,类成员函数可以自由地访问本类中的任何属性成员。不过若要从一个类中去访问另一个类中的属性,那就比较麻烦了,所以很多时候我们使用Getxxx和Setxxx方法,这样看起来显得极不自然,比如用Java或者C++,代码是这样的:

代码如下:

foo.setSize (getSize () + 1);
label.getFont().setBold (true);

但是,在C#中,这样的方法被“属性化”了。同样的代码,在C#就变成了:  
foo.size++;  
label.font.bold = true; 
    可以看出来,C#显然更容易阅读和理解。我们从这个“属性方法”的子程序代码中,也可以看到类似情况:

Java/C++:

代码如下:

public int getSize()  
{  
return size;  
}  
public void setSize (int value)  
{  
size = value;  
}

C#:

代码如下:

public int Size  
{  
 get{return size;}  
 set{size = value;}  
}

为了区分这种属性化的方法和类的属性成员,在C#中把属性成员称作“域(field)”,而“属性”则成为这种“属性化的方法”专用的名词。顺便说一句,其实这样的属性化方法在VB和DELPHI中是经常碰到的,在VB中它也就叫属性。另外,在C#中Get和Set必须成对出现,一种属性不能只有Get而没有Set(在Java和C++中就可以只有Get或者只有Set),C#中这样做的好处在于便于维护,假如要对某种属性进行修改,就会同时注意Get和Set方法,同时修改,不会改了这个忘了那个。

五、对象索引机制(Indexer)

C#中引入了对象索引机制。说得明白点,对象索引其实就是对象数组。这里和上一节中的属性联系起来讲一下,属性需要隐藏Get和Set方法,而在索引机制中,各个对象的Get或者Set方法是暴露出来的。比如下面的例子就比较清楚地说明了这一点。以上介绍C#和Java不同点


代码如下:

public class Skyscraper 

 Story[] stories; 
 public Story this [int index] { 
get { 
 return stories [index]; 

set { 
 if (value != null) {

stories [index] = value; 
 } 

 } 
... 
}

以上介绍C#和JAVA得不同点,希望对你了解C#和JAVA有所帮助。

(0)

相关推荐

  • C#命名空间与java包的区别分析

    本文实例分析了C#命名空间与java包的区别.分享给大家供大家参考.具体分析如下: 相同点: 1.都是为了重用性(reusebility)--软件工程中一个非常重要的目标. 2.C#里面的命名空间和java中的包都属于访问权限的控制机制. 不同点: 1.C#命名空间只是一种逻辑上的结构,它允许所组织的类文件的物理存放位置与逻辑结构不一致,而Java中类文件的物理结构必须与逻辑结构一致. 2.Java的package本身没有子包的概念,所有package都是并列的关系,没有谁包含谁的问题.比如:o

  • js substr,substring与java substring和C# substring的区别解析

    js substr(start[,length])表示从start位置开始取length个字符串 js substring(start,end)表示从start,到end之间的字符串,包括start位置的字符但是不包括end位置的字符 java sbustring(start,end)表示从start,到end之间的字符串,包括start位置的字符它包括end位置的字符 c# Substring(start[,length])表示从start位置开始取length个字符串

  • java与c#的语法区别详细介绍

    由C#转入Java一段时间了,总结下个人认为的Java同C#语法之间的不同之处,有不同意见之处还望各位海涵 刚学Java时觉得语法同C#大致是相同的(应该说C#同Java大致相同,毕竟人家微软的C#是有意模仿Java的语法习惯的) 比尔.盖茨曾经说过:"Java是最卓越的程序设计语言" 言归正传,下面探讨Java同C#的语法不同之处... 1,命名空间与包 C#为了把实现相似功能的类组织在一起,引入了命名空间的概念(namespace) Java中与此对应的东西叫做包(package)

  • C#和Java有什么区别和联系

    由于公司同是使用.NET和JAVA,而且各个服务集使用接口来进行通信,因此某些例如清算系统.收银台之类的安全性比较高的系统会使用RSA进行加密.因此会涉及到秘钥的转换.所以大概看了下C#的秘钥跟JAVA的秘钥区别. RSA对于程序本身是没有区别的,其格式都是相同的.对于不同的程序来说,存储使用的语法(包装的类)会有所不同. RSA语法和语法标准有很多,大的类型大概分为ASN.1.PKCS.X.509. RSA语法介绍 ASN.1.PKCS是最初的,也是最主要的RSA公钥和私钥的语法标准,被RSA

  • java 栈和堆区别详细介绍

    java 栈和堆区别 1, 垃圾回收机制仅仅作用于堆内存,与栈内存无关; 2, 栈:stack 栈的存取速度比堆快,效率高 保存局部变量和对象的引用值 3, 堆:保存较大的变量 4, 栈有一个很重要的特殊性,就是存在栈中的数据可以共享 编译器处理int a = 3 时,首先会在栈中创建一个变量为a的引用,然后查找栈中是否存在3这个值,如果不存在,就将3存放进来: 处理b = 3时,也是这样,因为之前将3存放进来了,所以此时编译器只需将创建b这个变量再指向3就行了(有利于节省空间). 这个时候就出

  • Java Array与ArrayList区别详解

    Java Array与ArrayList区别 1)精辟阐述: 可以将 ArrayList想象成一种"会自动扩增容量的Array". 2)Array([]):最高效:但是其容量固定且无法动态改变: ArrayList:  容量可动态增长:但牺牲效率: 3)建议: 基于效率和类型检验,应尽可能使用Array,无法确定数组大小时才使用ArrayList! 不过当你试着解决更一般化的问题时,Array的功能就可能过于受限. 4)Java中一切皆对象,Array也是对象.不论你所使用得Array

  • java.util.Date与java.sql.Date的区别

    我数据库里用到了日期类型.用java编程的时候同时import了java.util.*和java.sql.*,发现直接申明Date类型 Date dt; 会报错,查了一下才发现有java.util.Date和java.sql.Date,在定义日期类型的时候要使用全名,就是像这样: java.util.Date udt; java.sql.Date sdt; 然后我就查java.util.Date和java.sql.Date的区别和用法,这方面网上资料挺全,我就不赘述了,大致区别就是java.ut

  • JavaScript与Java正则表达式写法的区别介绍

    Js验证写法:(转义符\) var str = "待验证文本"; var regular = new RegExp(/这里是正则表达式/); if (regular.test(str)) { console.log("符合条件"); } else { console.log("不符合条件"); } //或者 var str = "待验证文本"; if (/这里是正则表达式/.test(str)) { console.log(&

  • 浅谈java中Math.random()与java.util.random()的区别

    今天突然想起来,java产生随机数的问题,上机试了一下,找到了一点区别,在这里总结一下: 直接调用Math.random()是产生一个[0,1)之间的随机数, 如果用 java.util.Random random=new Random();random.nextInt() 这样产生一个长整型的随机数并且与上一次是一样的,如果过一会再产生就不会一样了,例如: for (int i = 0; i < 10; i++) { Random random=new Random(); Thread.sle

  • C#和Java中二维数组区别分析

    本文实例讲述了C#和Java中二维数组区别,分享给大家供大家参考.具体分析如下: 在Java中使用二维数组可以如下代码: 复制代码 代码如下: public class Array2D{     public static void main(String[] args){         int myInt[][]=new int[5][10];         //遍历,给数组中的每一个数组赋值         for(int i=0;i<myInt.length;i++){        

  • Java集合与数组区别简介及相互转换实例

    数组Array和集合的区别: (1)数组是大小固定的,并且同一个数组只能存放类型一样的数据(基本类型/引用类型) (2)JAVA集合可以存储和操作数目不固定的一组数据.(3)若程序时不知道究竟需要多少对象,需要在空间不足时自动扩增容量,则需要使用容器类库,array不适用. 联系:使用相应的toArray()和Arrays.asList()方法可以回想转换. List和ArrayList的区别 1.List是接口,List特性就是有序,会确保以一定的顺序保存元素. ArrayList是它的实现类

  • Java 位运算符>>与>>>区别案例详解

    下图是java教程中对于>>和>>>区别的解释,但是介绍的并不详细,因为这两种运算符是以补码二进制进行运算的. 1.学习过计算机原理的都知道,数字是以补码的形式在计算机中存储的,那么源码,反码,补码之间的关系是如下所示: **正整数**的原码.反码和补码都一样: **负数部分**: 1.原码和反码的相互转换:符号位不变,数值位按位取反 2.原码和补码的相互转换:符号位不变,数值位按位取反,末位再加1 3.已知补码,求原码的负数的补码:符号位和数值位都取反,末位再加1 2.了解

  • Java getRealPath("/")与getContextPath()区别详细分析

    一.以下是在servlet中测试的. 二.运行结果如下(其中myfile文件夹是不存在的). 三.总结. 1.getRealPath("/")方法返回的是项目在服务器的绝对路径,而getRealPath("WEB-INF/myfile")返回的是包含一个给定虚拟路径的绝对路径,其中/myfile是虚拟的. 2.getContextPath()方法返回的是项目相对路径,并不会返回其绝对的路径,就是说并不知道项目在哪个位置. 到此这篇关于Java getRealPath

  • java compare compareTo方法区别详解

    1,compareTo(Object o)方法是java.lang.Comparable<T>接口中的方法,当需要对某个类的对象进行排序时,该类需要实现Comparable<T>接口的,必须重写public int compareTo (T o)方法,比如MapReduce中Map函数和Reduce函数处理的<key,value>,其中需要根据key对键值对进行排序,所以,key实现了WritableComparable<T>接口,实现这个接口可同时用于序列

随机推荐