Mari kita lihat artikel ini tentang 5 soalan temu bual dalam kategori Java String
Java String
类的 5 道面试题,这五道题,我自己在面试过程中亲身经历过几道题目,本篇就带你了解这些题的答案为什么是这样。
public class Demo2_String { public static void main(String[] args) { String st1 = "abc"; String st2 = "abc"; System.out.println(st1 == st2); System.out.println(st1.equals(st2)); } }
输出结果:
分析
先看第一个打印语句,在Java中==
这个符号是比较运算符,它可以基本数据类型和引用数据类型是否相等,如果是基本数据类型,==
比较的是值是否相等,如果是引用数据类型,==
. Saya sendiri pernah mengalami beberapa daripada lima soalan ini semasa proses temu duga. Artikel ini akan membantu anda memahami mengapa jawapan kepada soalan ini adalah seperti ini.
String st1 = new String(“abc”);
==
Simbol ini ialah operator perbandingan, yang boleh menentukan sama ada jenis data asas dan jenis data rujukan adalah sama. Jika ia adalah jenis data asas, ==
membandingkan sama ada nilai adalah sama. Jika ia adalah jenis data rujukan , ==
membandingkan sama ada alamat memori bagi dua objek adalah sama. 🎜🎜String bukan tergolong dalam 8 jenis data asas objek String tergolong dalam jenis data rujukan Dalam perkara di atas, "abc" ditugaskan kepada dua objek rentetan st1 dan st2 pada masa yang sama, menunjuk ke alamat yang sama. pertama Hasil keluaran perbandingan == dalam pernyataan cetakan adalah benar
Kemudian kita melihat perbandingan equals dalam pernyataan cetakan kedua. 🎜🎜🎜在JDK API 1.6文档中找到String类下的equals方法,点击进去可以看到这么一句话“将此字符串与指定的对象比较。当且仅当该参数不为null,并且是与此对象表示相同字符序列的String 对象时,结果才为 true。” 注意这个相同字符序列,在后面介绍的比较两个数组,列表,字典是否相等,都是这个逻辑去写代码实现。
由于st1和st2的值都是“abc”,两者指向同一个对象,当前字符序列相同,所以第二行打印结果也为true。下面我们来画一个内存图来表示上面的代码,看起来更加有说服力。
内存过程大致如下:
1)运行先编译,然后当前类Demo2_String.class文件加载进入内存的方法区
2)第二步,main方法压入栈内存
3)常量池创建一个“abc”对象,产生一个内存地址
4)然后把“abc”内存地址赋值给main方法里的成员变量st1,这个时候st1根据内存地址,指向了常量池中的“abc”。
5)前面一篇提到,常量池有这个特点,如果发现已经存在,就不在创建重复的对象
6)运行到代码 Stringst2 =”abc”, 由于常量池存在“abc”,所以不会再创建,直接把“abc”内存地址赋值给了st2
7)最后st1和st2都指向了内存中同一个地址,所以两者是完全相同的。
String st1 = new String(“abc”);
答案是:在内存中创建两个对象,一个在堆内存,一个在常量池,堆内存对象是常量池对象的一个拷贝副本。
分析
我们下面直接来一个内存图。
当我们看到了new这个关键字,就要想到,new出来的对象都是存储在堆内存。然后我们来解释堆中对象为什么是常量池的对象的拷贝副本。
“abc”属于字符串,字符串属于常量,所以应该在常量池中创建,所以第一个创建的对象就是在常量池里的“abc”。
第二个对象在堆内存为啥是一个拷贝的副本呢,这个就需要在JDK API 1.6找到String(String original)这个构造方法的注释:初始化一个新创建的 String 对象,使其表示一个与参数相同的字符序列;换句话说,新创建的字符串是该参数字符串的副本。所以,答案就出来了,两个对象。
package string; public class Demo2_String { public static void main(String[] args) { String st1 = new String("abc"); String st2 = "abc"; System.out.println(st1 == st2); System.out.println(st1.equals(st2)); } }
答案:false 和 true 由于有前面两道题目内存分析的经验和理论,所以,我能快速得出上面的答案。
==比较的st1和st2对象的内存地址,由于st1指向的是堆内存的地址,st2看到“abc”已经在常量池存在,就不会再新建,所以st2指向了常量池的内存地址,所以==判断结果输出false,两者不相等。
第二个equals比较,比较是两个字符串序列是否相等,由于就一个“abc”,所以完全相等。
内存图如下
public class Demo2_String { public static void main(String[] args) { String st1 = "a" + "b" + "c"; String st2 = "abc"; System.out.println(st1 == st2); System.out.println(st1.equals(st2)); } }
答案是:true 和 true 分析:“a”,”b”,”c”三个本来就是字符串常量,进行+符号拼接之后变成了“abc”,“abc”本身就是字符串常量(Java中有常量优化机制),所以常量池立马会创建一个“abc”的字符串常量对象,在进行st2=”abc”,这个时候,常量池存在“abc”,所以不再创建。所以,不管比较内存地址还是比较字符串序列,都相等。
public class Demo2_String { public static void main(String[] args) { String st1 = "ab"; String st2 = "abc"; String st3 = st1 + "c"; System.out.println(st2 == st3); System.out.println(st2.equals(st3)); } }
答案:
false
false
true
analisis
Jawapan pertama di atas adalah palsu, yang kedua adalah benar, dan yang kedua adalah benar. , ini adalah output yang benar, kami memahaminya dengan baik. Jadi mengapa penghakiman pertama adalah palsu, kita keliru. Begitu juga, di bawah kami menggunakan ulasan API dan gambar rajah memori untuk menjelaskan mengapa ini tidak sama.Pertama, buka pengenalan String dalam JDK API 1.6 dan cari ayat dalam gambar di bawah.
Intinya ialah ayat dalam bulatan merah Kami tahu bahawa sebarang data dan rentetan tertakluk kepada operasi tanda tambah (+), dan hasil akhir ialah rentetan baharu yang disambung. Apakah sebenarnya yang dilakukan oleh operasi +? Disyorkan untuk dibaca.Komen di atas menerangkan bahawa prinsip penyambungan ini ialah kelas StringBuilder atau StringBuffer dan kaedah tambah di dalam melaksanakan penyambungan, dan kemudian panggil keString() untuk menukar objek yang disambungkan menjadi objek rentetan, dan akhirnya menetapkan alamat daripada objek rentetan kepada pembolehubah . Berdasarkan pemahaman ini, mari kita lukis gambar rajah ingatan untuk analisis.
Mengenai proses memori1) Kolam malar mencipta objek "ab" dan menetapkannya kepada st1, jadi st1 menunjuk kepada "ab" 2) Kolam malar mencipta objek "abc" dan menetapkannya kepada st2, jadi st2 menunjuk kepada "abc"3) Memandangkan kaedah penyambungan + digunakan di sini, langkah ketiga ialah menggunakan kaedah tambah kelas StringBuffer untuk mendapatkan "abc". Pada masa ini, memori 0x0011 mewakili a Objek StringBuffer Perhatikan bahawa ia bukan objek String. 4) Kaedah toString Object dipanggil untuk menggantikan objek StringBuffer menjadi objek String. 5) Tetapkan objek String (0x0022) kepada st3🎜🎜Jadi, hasil penghakiman == bagi st3 dan st2 adalah tidak sama kerana alamat memori kedua-dua objek adalah berbeza. 🎜Soalan wawancara ini memerlukan menguasai beberapa anotasi dan prinsip dalam API JDK, serta analisis grafik memori, untuk mendapatkan hasil yang betul. jawabnya.Atas ialah kandungan terperinci 5 Rentetan soalan temuduga, kurang daripada 10% orang boleh menjawab semuanya dengan betul! (dengan jawapan). Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!