2019-11-12

Java String 面试题以及答案

1.判定定义为String类型的st1和st2是否相等,为什么

package string;

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));

  }

}

输出结果:

第一行:true

第二行:true

分析:

       先看第一个打印语句,在Java中==这个符号是比较运算符,它可以基本数据类型和引用数据类型是否相等,如果是基本数据类型,==比较的是值是否相等,如果是引用数据类型,==比较的是两个对象的内存地址是否相等。字符串不属于8中基本数据类型,字符串对象属于引用数据类型,在上面把“abc”同时赋值给了st1和st2两个字符串对象,指向的都是同一个地址,所以第一个打印语句中的==比较输出结果是 true

       然后我们看第二个打印语句中的equals的比较,我们知道,equals是Object这个父类的方法,在String类中重写了这个equals方法,在JDK API 1.6文档中找到String类下的equals方法,点击进去可以看大这么一句话“将此字符串与指定的对象比较。当且仅当该参数不为null,并且是与此对象表示相同字符序列的 String 对象时,结果才为 true。” 注意这个相同字符序列,在后面介绍的比较两个数组,列表,字典是否相等,都是这个逻辑去写代码实现。由于st1和st2的值都是“abc”,两者指向同一个对象,当前字符序列相同,所以第二行打印结果也为true。

2. 下面这句话在内存中创建了几个对象

String st1 = new String(“abc”);

答案是:在内存中创建两个对象,一个在堆内存,一个在常量池,堆内存对象是常量池对象的一个拷贝副本。

       当我们看到了new这个关键字,就要想到,new出来的对象都是存储在堆内存。然后我们来解释堆中对象为什么是常量池的对象的拷贝副本。“abc”属于字符串,字符串属于常量,所以应该在常量池中创建,所以第一个创建的对象就是在常量池里的“abc”。第二个对象在堆内存为啥是一个拷贝的副本呢,这个就需要在JDK API 1.6找到String(String original)这个构造方法的注释:初始化一个新创建的 String 对象,使其表示一个与参数相同的字符序列;换句话说,新创建的字符串是该参数字符串的副本。所以,答案就出来了,两个对象。

3.判定以下定义为String类型的st1和st2是否相等

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”,所以完全相等。

4. 判定以下定义为String类型的st1和st2是否相等

package string;

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”,所以不再创建。所以,不管比较内存地址还是比较字符串序列,都相等。

5.判断以下st2和st3是否相等

package string;

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 和 true

分析:

      上面的答案第一个是false,第二个是true,第二个是true我们很好理解,因为比较一个是“abc”,另外一个是拼接得到的“abc”,所以equals比较,这个是输出true,我们很好理解。那么第一个判断为什么是false,我们很疑惑。同样,下面我们用API的注释说明和内存图来解释这个为什么不相等。

String是最常使用的Java类之一,整理的了一些重要的String知识分享给大家。

作为一个Java新手程序员,对String进行更深入的了解很有必要。如果你是有几年Java开发经验,可以根据目录选择性的阅读以下内容。

1、什么是String,它是什么数据类型?

String是定义在 java.lang 包下的一个类。它不是基本数据类型。

String是不可变的,JVM使用字符串池来存储所有的字符串对象。

2、创建String对象的不同方式有哪些?

和使用其他类一样通过new关键字来创建。

使用这种方式时,JVM创建字符串对象但不存储于字符串池。我们可以调用intern()方法将该字符串对象存储在字符串池,如果字符串池已经有了同样值的字符串,则返回引用。

32507使用双引号直接创建。

】14

String str = new String("abc");

String str1 = "abc";

3、写一个方法来判断一个String是否是回文(顺读和倒读都一样的词)?

回文就是正反都一样的词,如果需要判断是否是回文,只需要比较正反是否相等即可。String类并没有提供反转方法供我们使用,但StringBuffer和StringBuilder有reverse方法。

private static boolean isPalindrome(String str) {

       if (str == null)

           return false;

       StringBuilder strBuilder = new StringBuilder(str);

       strBuilder.reverse();

       return strBuilder.toString().equals(str);

   }

假设面试官让你不使用任何其他类来实现的话,我们只需要首尾一一对比就知道是不是回文了。

private static boolean isPalindromeString(String str) {

       if (str == null)

           return false;

       int length = str.length();

       System.out.println(length / 2);

       for (int i = 0; i < length / 2; i++) {

           if (str.charAt(i) != str.charAt(length - i - 1))

               return false;

       }

       return true;

   }

4、如何让一个字符串变成小写或大写形式?

使用toUpperCase 和 toLowerCase 方法让一个字符串变为 大写或小写。

5、如何比较两个字符串?

String内部实现了Comparable接口,有两个比较方法:compareTo(String anotherString) 和compareToIgnoreCase(String str)。

compareTo(String anotherString)

与传入的anotherString字符串进行比较,如果小于传入的字符串返回负数,如果大于则返回证书。当两个字符串值相等时,返回0.此时eqauls方法会返回true。

equalsIgnoreCase(String str)

该方法与compareTo方法类似,区别只是内部利用了Character.toUpperCase等方法进行了大小写转换后进行比较。

6、如何将String转换为char,反过来呢?

这是一个误导题,String是一系列字符,所有我们没法转换成一个单一的char,但可以调用toCharArray() 方法将字符串转成字符数组。

String str = "Java interview";


   //string to char array

   char[] chars = str.toCharArray();

   System.out.println(chars.length);

7、如何将String转换为byte array,反过来呢?

使用String的getBytes()方法将String转成byte数组,使用String的构造方法 new String(byte[] arr) 将byte数据转为String。

public class StringToByteArray {

   public static void main(String[] args) {

       String str = "PANKAJ";

       byte[] byteArr = str.getBytes();

       // print the byte[] elements

       System.out.println("String to byte array: " + Arrays.toString(byteArr));

   }

}

public class ByteArrayToString {

   public static void main(String[] args) {

       byte[] byteArray = { 'P', 'A', 'N', 'K', 'A', 'J' };

       byte[] byteArray1 = { 80, 65, 78, 75, 65, 74 };

       String str = new String(byteArray);

       String str1 = new String(byteArray1);

       System.out.println(str);

       System.out.println(str1);

   }

}

<div id="question8"></div>

8、浅谈一下String, StringBuffer,StringBuilder的区别?

String是不可变类,每当我们对String进行操作的时候,总是会创建新的字符串。操作String很耗资源,所以Java提供了两个工具类来操作String - StringBuffer和StringBuilder。

StringBuffer和StringBuilder是可变类,StringBuffer是线程安全的,StringBuilder则不是线程安全的。所以在多线程对同一个字符串操作的时候,我们应该选择用StringBuffer。由于不需要处理多线程的情况,StringBuilder的效率比StringBuffer高。

9、String是不可变的有什么好处?

String是不可变类有以下几个优点

由于String是不可变类,所以在多线程中使用是安全的,我们不需要做任何其他同步操作。

String是不可变的,它的值也不能被改变,所以用来存储数据密码很安全。

因为java字符串是不可变的,可以在java运行时节省大量java堆空间。因为不同的字符串变量可以引用池中的相同的字符串。如果字符串是可变得话,任何一个变量的值改变,就会反射到其他变量,那字符串池也就没有任何意义了。

10、如何分割一个String?

public String[] split(String regex):

根据传入的正则字符串进行分割,注意,如果最后一位刚好有传入的字符,返回数组最后一位不会有空字符串。

String s = "abcaada";

System.out.println(Arrays.toString(s.split("a")));

//以上代码输出为  [, bc, , d].

public String[] split(String regex, int limit):

限制分割结果数组中有几个字符串。传入2,则结果分割后数组长度为2。

String s = "Y,Kunming,Yunnan";

String[] data = s.split(",", 2);

System.out.println("Name = "+data[0]); //Y

System.out.println("Address = "+data[1]); //Kunming,Yunnan

实际上第一个方法调用了第二个方法,只不过不限制返回的数组长度了。

public String[] split(String regex) {

   return split(regex, 0);

}

11、如何判断两个String是否相等?

有两种方式判断字符串是否相等,使用"=="或者使用equals方法。当使用"=="操作符时,不仅比较字符串的值,还会比较引用的内存地址。大多数情况下,我们只需要判断值是否相等,此时用equals方法比较即可。

还有一个equalsIgnoreCase可以用来忽略大小写进行比较。

String s1 = "abc";

       String s2 = "abc";

       String s3= new String("abc");

       System.out.println("s1 == s2 ? "+(s1==s2)); //true

       System.out.println("s1 == s3 ? "+(s1==s3)); //false

       System.out.println("s1 equals s3 ? "+(s1.equals(s3))); //true

12、什么是字符串池?

顾名思义,字符串常量池就是用来存储字符串的。它存在于Java 堆内存。

下图解释了字符串池在java堆空间如何存在以及当我们使用不同方式创建字符串时的情况。

以下是上图的一个编程例子

public class StringPool {


   public static void main(String[] args) {

       String s1 = "Cat";

       String s2 = "Cat";

       String s3 = new String("Cat");


       System.out.println("s1 == s2 :"+(s1==s2));

       System.out.println("s1 == s3 :"+(s1==s3));

   }

}

运行以上代码,输出如下:

s1 == s2 :true

s1 == s3 :false

一些java题中,可能会问一段代码中有几个字符串被创建,例如:

String str = new String("Cat");

上面一行代码将会创建1或2个字符串。如果在字符串常量池中已经有一个字符串“Cat”,那么就智慧创建一个“Cat”字符串。如果字符串常量池中没有“Cat”,那么首先会在字符串池中创建,然后才在堆内存中创建,这种情况就会创建2个对象了。

13、String的intern()方法

当intern()方法被调用,如果字符串池中含有一个字符串和当前调用方法的字符串eqauls相等,那么就会返回池中的字符串。如果池中没有的话,则首先将当前字符串加入到池中,然后返回引用。

14、String是线程安全的吗?

String是不可变类,一旦创建了String对象,我们就无法改变它的值。因此,它是线程安全的,可以安全地用于多线程环境中。

15、为什么我们在使用HashMap的时候总是用String做key?

因为字符串是不可变的,当创建字符串时,它的它的hashcode被缓存下来,不需要再次计算。因为HashMap内部实现是通过key的hashcode来确定value的存储位置,所以相比于其他对象更快。这也是为什么我们平时都使用String作为HashMap对象。

16、String编程题

1、下面的代码输入什么

String s1 = new String("abc");

String s2 = new String("abc");

System.out.println(s1 == s2);

输入false

2、下面的代码输入什么

String s1 = "abc";

StringBuffer s2 = new StringBuffer(s1);

System.out.println(s1.equals(s2));

输入false,因为s2不是String类型,String的equals方法进行了类型判断。

3、下面的代码输入什么

String s1 = "abc";

String s2 = new String("abc");

s2.intern();

System.out.println(s1 ==s2);

输出false,intern()方法将返回从字符串池中的字符串对象的引用,但因为我们没有分配到S2,S2没有变化,如果该第三行代码为s2 =

s2.intern(),则输入true。

4、下面的代码将创建几个字符串对象。

String s1 = new String("Hello");  

String s2 = new String("Hello");

答案是3个对象.

第一,行1 字符串池中的“hello”对象。

第二,行1,在堆内存中带有值“hello”的新字符串。

第三,行2,在堆内存中带有“hello”的新字符串。这里“hello”字符串池中的字符串被重用

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,923评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,154评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,775评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,960评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,976评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,972评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,893评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,709评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,159评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,400评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,552评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,265评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,876评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,528评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,701评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,552评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,451评论 2 352

推荐阅读更多精彩内容