今天在牛客网上做题,基础题部分竟然做的很差。其中有一道题就是字符串的创建和存储机制。然后查阅了相关资料,将知识点做一个整理。
先看一下博客上的题目:
然后我就写了一个测试程序代码:
程序中定义了两个字符串常量,字符串对象实例,以及字符数组,。将使用 == 和 equals()对这两个对象进行比较。结果如下:
下面就说一下原理:
在java中字符串的声明与初始化有如下几种方式:
1. String a = new String("abc");
2. String b = "abc"
这两种方式结果都是一样的,但是实现机制却不相同,接着往下看:
String a = "hello";
String b = "hello"
- 与
String c = new String("hello");
String d = new String ("hello");
请问
a==b true or false? c==d true or false?
a.equlas(b) true or false? c.equals(d) true or false?
简而言之,java中在new时,总会产生一个新的对象,而新对象则需要分配新的内存,所以即使两个对象的内容是相同的,但是在内存中的首地址是不同的,即c,d中存储的数值是不相同的,所以表达式c==d将返回false。
而a==b=true 是因为在JVM中存在着一个字符串池,其中保存着很多String对象,并且可以被共享使用,a,b引用的是同一个常量池中的对象,当创建一个字符串常量时,会首先在字符串常量池中查找是否已经有相同的字符串被定义(其判断依据是String类的equals方法的返回值),若已经定义则直接获取对其的引用,此时不需要创建对象;若没有定义,则需要创建这个对象,然后将其加入到字符串池中,再将它的引用返回。(由于String是不可变类,一旦创建好了就不能被修改,因此String对象可以被共享且不会导致程序的混乱)。
equals是Object类提供的方法之一。每一个java类都继承Object类,所以每个对象都有equals这个方法。Obejct类中定义的equals方法是直接使用”==”运算符比较两个对象,所以在没有覆盖equlas方法的时候,equals与“==”运算符一样,比较的是引用。
因为a与b,c与d对象中的内容是相同的。
现在想想
a==c ? true or false?
a.equals(c) ? true or false?
看一张图:
a与c指的是同一个东西吗? a与c内容是一样的吗?答案显而易见,a==c为false a.equals(c)为true
可以人为地分为两个过程,第一个过程是新建对象的过程,new String("hello")
; 第二个过程是赋值的过程,即String c =
。
在进行第一个过程new String("hello")
时,等价于”hello”与new String()
两个操作。若在字符串池中不存在”hello”则会创建一个字符串常量”hello”,并将其添加到字符串池中;若存在,则不进行创建,然后new String()
会在堆中创建一个新的对象。
问题:
new String(“hello”)创建了几个对象?
答案:一个或者两个。如果常量池中有”hello”,那么只创建一个对象;如果没有,就会创建两个对象