一、前言:
当我们创建一个对象时,就会调用它的构造函数来开辟空间,将对象存到堆内存中,与此同时,在栈内存中生成对应的引用。当我们在后续代码中调用的时候用的都是栈中的引用。需要值得注意的是,基本数据类型是存储在栈内存中的。
二、区别:
(int等基本类型是不支持equals()方法的。)
对于自定义的引用类型来说。
==和equals最大的区别是一个是运算符一个是方法。在Java中,两者比较的都是物理地址,而不是值的比较。调用的是Object类的equals方法。(并没有对其进行重写)
首先看一下下面的例子。
两种比较最终显示都为false。这是为什么呢?就是因为他们比较的不是对象中的字段或者或对象本身,真正比较的是物理地址。
对于String类型来说。
equals方法比较的是两个字符串的值是否一样。(不比较物理地址)。
String a=new String();
String b=new String();
System.out.println(a.equals(b));
最终输出结果为:true
(a和b都是空串,即零个字符的串,也就是说串的长度为0)
(空串和空格串是不一样的:空格串是只包含空格的串,空格串是有内容有长度的,而且可以不止一个空格。空格对应的ascii码是32)
String d=new String("abc");
String h=new String("abc");
System.out.println(d.equals(h));
最终输出结果为:true
String o="";
String z=new String("");
String g=new String();
System.out.println(o.equals(z))
System.out.println(o.equals(g))
最终输出结果为
:true true三、原因:
这是为何呢?原因何在?其实看一下源码,问题就一目了然啦。我们知道,所以类的父类都是java.lang. Object类。
Object类的equals源码如下:
public boolean equals(Object obj) {
return (this == obj);
}
由源码可知,a.equals(b)等价于a==b。
而String类重写了equals方法,源码如下:
private final char value[];
public boolean equals(Object anObject) {
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
由源码可知,首先声明字符类型的数组value[]。然后比较两个字符串的字符长度是否相等。最后比较每一个对应的字符是否相等。
四、扩展:
String aa="abc";
String bb="abc";
System.out.println(aa==bb);
System.out.println(aa.equals(bb));
最后的输出结果为:
true true
这是为何?
所有的字符串都是String对象。由于字符串的大量使用,java中为了节省时间,在编译阶段,会把字符串文字放在字符串常量池中。
如把"abc"字符串放到常量池中,当再次需要"abc"字符串时,首先会去常量池中查找是否存在。也就是说引用变量a、b指向的是同一个对象。同一块空间
1、java中创建字符串对象的几种形式:
①通过new方法如 String s=new String("abc");
②通过字面量的形式如 String s="abc";
③字面量+字面量 如 String s="ac"+"c";
④字面量+变量 如 String s1="ab"+s;
(String是final(不可变)类型的,任何时候出现新的String一定是重新创建的。例如,String a=“123”;a=“456”
是创建了两个对象。而不是把对象“123”直接修改为对象“456”)
2、String s = new String("abc")创建了几个对象?
3、字符串常量池:
字符串在java程序中被大量使用,为了避免每次都创建相同的字符串对象及内存分配,JVM内部对字符串对象的创建做了一定的优化。在方法区有一块区域专门用来存储字符串常量的。