今天看到成员变量是在堆中的,那么用==比较成员变量的时候比较是什么地方的东西,如果是局部变量就是比较栈区内容如果是对象就比较栈里存的地址,可是成员变量在堆里,那我们比较的难道是堆,我记得这个只能比较栈空间的内容这是什么原理
是这样,java中除了基本类型,其他的都是引用类型,==比较的是什么跟是否是成员变量没有关系,你的成员变量如果是基本类型,他比较的就是两者的值,如果是引用则比较的是地址值,也就是两者是否指向同一对象,这跟在栈中或是堆中没关系,在栈中比的就是栈中的地址或值,堆中比的就是堆中的地址或值
这个要看成员变量的类型吧。如果是引用类型:比较的是内存地址,如果是简单数据类型:比较的是值
对于引用类型,比较的是引用地址。而且除了判断==null,其他基本上不存在用==来比较两个对象的情况。
如果是基本数据类型比较的是对象的值,如果是引用数据类型比较的是地址值。equse不重写和==一样。
对于这个问题,首先搞懂基本数据类型和引用数据类型的存储结构就很好理解了。基本类型存储在堆中(存放的就是变量值),而引用类型在堆中存的是
一个地址,该地址指向栈中的一块内存(存放的引用类型的值)。而“==”比较的是堆里的内容。当然,普通的equals也是比较堆,而String不同是因为
String重写了equals方法。
==操作符专门用来比较两个变量的值是否相等,也就是用于比较变量所对应的内存中所存储的数值是否相同,要比较两个基本类型的数据或两个引用变量是否相等,只能用==操作符。
如果一个变量指向的数据是对象类型的,那么,这时候涉及了两块内存,对象本身占用一块内存(堆内存),变量也占用一块内存,例如Objet obj = new Object();变量obj是一个内存,new Object()是另一个内存,此时,变量obj所对应的内存中存储的数值就是对象占用的那块内存的首地址。对于指向对象类型的变量,如果要比较两个变量是否指向同一个对象,即要看这两个变量所对应的内存中的数值是否相等,这时候就需要用==操作符进行比较。
equals方法是用于比较两个独立对象的内容是否相同,就好比去比较两个人的长相是否相同,它比较的两个对象是独立的。例如,对于下面的代码:
String a=new String("foo");
String b=new String("foo");
两条new语句创建了两个对象,然后用a,b这两个变量分别指向了其中一个对象,这是两个不同的对象,它们的首地址是不同的,即a和b中存储的数值是不相同的,所以,表达式a==b将返回false,而这两个对象中的内容是相同的,所以,表达式a.equals(b)将返回true。
在实际开发中,我们经常要比较传递进行来的字符串内容是否等,例如,String input = …;input.equals(“quit”),许多人稍不注意就使用==进行比较了,这是错误的,随便从网上找几个项目实战的教学视频看看,里面就有大量这样的错误。记住,字符串的比较基本上都是使用equals方法。
如果一个类没有自己定义equals方法,那么它将继承Object类的equals方法,Object类的equals方法的实现代码如下:
boolean equals(Object o){
return this==o;
}
这说明,如果一个类没有自己定义equals方法,它默认的equals方法(从Object 类继承的)就是使用==操作符,也是在比较两个变量指向的对象是否是同一对象,这时候使用equals和使用==会得到同样的结果,如果比较的是两个独立的对象则总返回false。如果你编写的类希望能够比较该类创建的两个实例对象的内容是否相同,那么你必须覆盖equals方法,
Object的equals方法实际上就是用==判断的,
所以一般我们自定义的类都需要重写Object的equals方法,实际上Java的API中各种类都重写了equals方法,例如String等,
当用equals进行两个引用变量a和b的比较时,实际上是比较这两个引用变量所指向的对象内容是否相同,一般指这两个引用变量所指向的两个对象的各个属性值是否相同。如果这两个引用变量所指向的两个对象的内容相同则返回true,否则返回false,
当用==比较引用变量a和b时,实际上是比较a和b的直接值(在内存中的栈中的值),a的直接值是a所指向的对象的首地址(堆中),b的直接值是b所指向的对象的首地址(堆中),加入a==b返回true则说明a和b指向同一个对象,否则a和b指向不同的对象。