原文: https://howtodoinjava.com/interview-questions/interview-stuff-about-string-class-in-java/

我们所有人都必须经过与 Java 中的字符串类相关的面试问题。 这些字符串面试问题的范围从不变性到内存泄漏问题。 我将在这篇文章中尝试解决此类问题。

  1. Frequently asked String Interview Questions
  2. 1\. Is String keyword in Java?
  3. 2\. Why are strings immutable?
  4. 3\. What is String constant pool?
  5. 4\. Keyword 'intern' usage
  6. 5\. Matching Regular expressions?
  7. 6\. String comparison with equals() and '=='?
  8. 7\. Memory leak issue in String class
  9. 8\. How does String work in Java?
  10. 9\. What are different ways to create String Object?
  11. 10\. How to check if String is Palindrome.
  12. 11\. How to remove or replace characters from String.
  13. 12\. How to make String upper case or lower case?
  14. 13\. How to compare two Strings in java program?
  15. 14\. Can we use String in the switch case?
  16. 15\. Write a program to print all permutations of String?
  17. 16\. Write a java program to reverse each word of a given string??
  18. 17\. How to Split String in java?
  19. 18\. Why is Char array preferred over String for storing password?
  20. 19\. Is String thread-safe in Java
  21. 20\. Why String is popular HashMap key in Java
  22. 21\. Difference between String, StringBuffer and StringBuilder?
  23. 22\. How to concatenate multiple strings.
  24. 23\. How many objects will be created with string initialization code?
  25. 24\. How do you count the number of occurrences of each character in a string?
  26. 25\. Write a java program to reverse a string?

1. Java 中的String是关键字吗?

没有。 String不是 Java 保留的关键字。 它是派生类型数据类型,即类。

  1. public class StringExample
  2. {
  3. public static void main(String[] args)
  4. {
  5. Integer String = 10;
  6. System.out.println(String); //Prints 10
  7. }
  8. }

2.为什么字符串是不可变的?

我们都知道 java 中的字符串是不可变的。 如果您想知道什么是不变性以及如何实现? 按照这篇文章:如何使 java 类不可变

这里的问题是为什么? 为什么一成不变? 让我们分析一下。

  1. 我能想到的第一个原因是性能提升。 开发 Java 语言是为了加快应用程序开发,因为以前的语言并没有那么快。 JVM 设计人员必须足够聪明,才能识别出现实世界中的应用程序将主要由标签,消息,配置,输出等多种形式的字符串组成。
    看到这种过度使用,他们想像到不正确使用字符串会带来多大的危险。 因此,他们提出了字符串池的概念(下一部分)。 字符串池不过是一些大多数唯一字符串的集合。 字符串池背后的最基本思想是重新创建字符串。 这样,如果一个特定的字符串在代码中创建了 20 次,则应用最终将只有一个实例。

  2. 第二个原因是安全注意事项。 字符串是 Java 编程各个方面中最常用的参数类型。 无论是加载驱动程序还是打开 URL 连接,您都需要以字符串形式将信息作为参数传递。 如果字符串不是最终的,那么它们就打开了一个安全问题的潘多拉盒。我们所有人都必须通过与 Java 中的String类相关的面试问题。 这些问题的范围从不变性到内存泄漏问题。 我将在这篇文章中尝试解决此类问题。

除了上述两个原因外,我没有找到任何令人信服的答案。 如果您有任何吸引人的内容,请与我分享。

3.字符串池概念

字符串池是一个特殊的内存区域,与存储这些字符串常量的常规堆内存分开。 这些对象在应用的生命周期中称为字符串变量。

在 Java 中,可以通过多种方式创建String。 让我们了解它们:

1)字符串赋值

  1. String str = "abc";

上面的代码使 JVM 验证是否已经有一个字符串"abc"(相同的字符序列)。 如果存在这样的字符串,则 JVM 仅将现有对象的引用分配给变量str,否则,将创建一个新的对象"abc",并将其引用分配给变量str

2)使用new新关键字

  1. String str = new String("abc");

此版本最终在内存中创建两个对象。 字符串池中的一个对象具有char序列"abc",第二个对象在堆存储器中由变量str引用,并且具有与"abc"相同的char序列。

正如 java 文档所说的: 除非需要显式的原始副本,否则不需要使用此构造器,因为字符串是不可变的。

4.关键字“实习生”的用法

java 文档 最好地描述了这一点:

调用intern()方法时,如果池已经包含一个与equals(Object)方法确定的等于String对象的字符串,则返回池中的字符串。 否则,将此String对象添加到池中,并返回对此String对象的引用。

  1. String str = new String("abc");
  2. str.intern();

因此,对于st中的任何两个字符串,当且仅当s.equals(t)true时,s.intern() == t.intern()才为true。 意味着如果st都是不同的字符串对象,并且具有相同的字符序列,则在这两个变量上调用intern()将导致两个变量引用的单个字符串池文本。

5.匹配正则表达式

如果您还没有探索的话,它并不是那么秘密,但很有用。 您必须已经看到PatternMatcher用于正则表达式匹配的用法。 字符串类提供了自己的快捷方式。 直接使用。 此方法还使用函数定义内的Pattern.matches()

  1. String str = new String("abc");
  2. str.matches("<regex>");

6.使用equals()==进行字符串比较

面试中另一个最喜欢的领域。 通常有两种比较对象的方法

  • 使用==运算符
  • 使用equals()方法

==运算符比较对象引用,即内存地址相等性。 因此,如果两个字符串对象引用字符串池中的相同文本或堆中的相同字符串对象,则s==t将返回true,否则返回false

equals()方法在String类中被覆盖,它验证字符串对象持有的char序列。 如果它们存储相同的字符序列,则s.equals(t)将返回true,否则返回false

7.内存泄漏问题

到目前为止,我们已经完成了基本的工作。 现在发生了严重的事情。 您是否尝试过从字符串对象创建子字符串? 我敢打赌,是的。 您知道 Java 中子字符串的内部吗? 他们如何造成内存泄漏?

Java 中的子字符串是使用方法substring(int beginIndex)和此方法的其他一些重载形式创建的。 所有这些方法都会创建一个新的String对象,并更新在本文开头看到的offsetcount变量。

原始值数组不变。 因此,如果您创建一个包含 10000 个字符的字符串,并创建 100 个每个包含 5-10 个字符的子字符串,则所有 101 个对象将具有大小为 10000 个字符的相同字符数组。 毫无疑问,这是内存浪费。

让我们使用程序来看看:

  1. import java.lang.reflect.Field;
  2. import java.util.Arrays;
  3. public class SubStringTest {
  4. public static void main(String[] args) throws Exception
  5. {
  6. //Our main String
  7. String mainString = "i_love_java";
  8. //Substring holds value 'java'
  9. String subString = mainString.substring(7);
  10. System.out.println(mainString);
  11. System.out.println(subString);
  12. //Lets see what's inside mainString
  13. Field innerCharArray = String.class.getDeclaredField("value");
  14. innerCharArray.setAccessible(true);
  15. char[] chars = (char[]) innerCharArray.get(mainString);
  16. System.out.println(Arrays.toString(chars));
  17. //Now peek inside subString
  18. chars = (char[]) innerCharArray.get(subString);
  19. System.out.println(Arrays.toString(chars));
  20. }
  21. }
  22. Output:
  23. i_love_java
  24. java
  25. [i, _, l, o, v, e, _, j, a, v, a]
  26. [i, _, l, o, v, e, _, j, a, v, a]

显然,两个对象都存储有相同的char数组,而subString只需要四个字符。

让我们使用自己的代码解决此问题:

  1. import java.lang.reflect.Field;
  2. import java.util.Arrays;
  3. public class SubStringTest
  4. {
  5. public static void main(String[] args) throws Exception
  6. {
  7. //Our main String
  8. String mainString = "i_love_java";
  9. //Substring holds value 'java'
  10. String subString = fancySubstring(7, mainString);
  11. System.out.println(mainString);
  12. System.out.println(subString);
  13. //Lets see what's inside mainString
  14. Field innerCharArray = String.class.getDeclaredField("value");
  15. innerCharArray.setAccessible(true);
  16. char[] chars = (char[]) innerCharArray.get(mainString);
  17. System.out.println(Arrays.toString(chars));
  18. //Now peek inside subString
  19. chars = (char[]) innerCharArray.get(subString);
  20. System.out.println(Arrays.toString(chars));
  21. }
  22. //Our new method prevents memory leakage
  23. public static String fancySubstring(int beginIndex, String original)
  24. {
  25. return new String(original.substring(beginIndex));
  26. }
  27. }
  28. Output:
  29. i_love_java
  30. java
  31. [i, _, l, o, v, e, _, j, a, v, a]
  32. [j, a, v, a]

现在,子字符串只有它需要的字符,并且可以无用地收集用于创建正确子字符串的中间字符串,从而不占用任何内存。

8. String如何在 Java 中工作?

Java 中的字符串类似于任何其他编程语言,都是字符序列。 这更像是用于该char序列的工具类。 此char序列在以下变量中维护:

  1. /** The value is used for character storage. */
  2. private final char value[];

要在不同的情况下访问此数组,请使用以下变量:

  1. /** The offset is the first index of the storage that is used. */
  2. private final int offset;
  3. /** The count is the number of characters in the String. */
  4. private final int count;

10.如何检查回文中的字符串?

如果字符串的值在反转时相同,则称其为回文。 要检查回文,只需反转字符串并检查原始字符串和受尊敬的字符串的内容。

  1. public class StringExample
  2. {
  3. public static void main(String[] args)
  4. {
  5. String originalString = "abcdcba";
  6. StringBuilder strBuilder = new StringBuilder(originalString);
  7. String reverseString = strBuilder.reverse().toString();
  8. boolean isPalindrame = originalString.equals(reverseString);
  9. System.out.println(isPalindrame); //true
  10. }
  11. }

11.如何从String中删除或替换字符?

要替换或删除字符,请使用String.replace()String.replaceAll()。 这些方法有两个参数。 第一个参数是要替换的字符,第二个参数是将放置在字符串中的新字符。

如果要删除字符,请在第二个参数中传递空白字符。

  1. String originalString = "howtodoinjava";
  2. //Replace one character
  3. System.out.println( originalString.replace("h", "H") ); //Howtodoinjava
  4. //Replace all matching characters
  5. System.out.println( originalString.replaceAll("o", "O") ); //hOwtOdOinjava
  6. //Remove one character
  7. System.out.println( originalString.replace("h", "") ); //owtodoinjava
  8. //Remove all matching characters
  9. System.out.println( originalString.replace("o", "") ); //hwtdinjava

12.如何使String大写或小写?

使用String.toLowerCase()String.toUpperCase()方法将字符串转换为小写或大写。

  1. String blogName = "HowToDoInJava.com";
  2. System.out.println(blogName.toLowerCase()); //howtodoinjava.com
  3. System.out.println(blogName.toUpperCase()); //HOWTODOINJAVA.COM

13.如何在 Java 程序中比较两个字符串?

始终使用equals()方法来验证字符串是否相等。 切勿使用"=="运算符。 双重等于运算符始终检查内存中的对象引用。 equals()方法检查字符串内容。

  1. String blogName = "HowToDoInJava.com";
  2. String anotherString = new String("HowToDoInJava.com");
  3. System.out.println(blogName == anotherString); //false
  4. System.out.println(blogName.equals(anotherString)); //true

14.我们可以在开关盒中使用 String 吗?

是的,自 Java7 起,您可以在switch语句中使用String。在 Java7 之前,这是不可能的,您必须使用if-else语句才能实现类似的行为。

  1. String number = "1";
  2. switch (number)
  3. {
  4. case "1":
  5. System.out.println("One"); //Prints '1'
  6. break;
  7. case "2":
  8. System.out.println("Two");
  9. break;
  10. default:
  11. System.out.println("Other");
  12. }

15.编写一个程序以打印String的所有排列?

排列是对字符的有序列表的元素进行重新排列,以使每个排列相对于其他排列都是唯一的。 例如,以下是字符串“ABC”的排列 – ABC ACB BAC BCA CBA CAB

长度为N的字符串具有N!N的阶乘)个排列。

  1. import java.util.HashSet;
  2. import java.util.Set;
  3. public class StringExample
  4. {
  5. public static void main(String[] args)
  6. {
  7. System.out.println(getPermutations("ABC"));
  8. //Prints
  9. //[ACB, BCA, ABC, CBA, BAC, CAB]
  10. }
  11. public static Set<String> getPermutations(String string)
  12. {
  13. //All permutations
  14. Set<String> permutationsSet = new HashSet<String>();
  15. // invalid strings
  16. if (string == null || string.length() == 0)
  17. {
  18. permutationsSet.add("");
  19. }
  20. else
  21. {
  22. //First character in String
  23. char initial = string.charAt(0);
  24. //Full string without first character
  25. String rem = string.substring(1);
  26. //Recursive call
  27. Set<String> wordSet = getPermutations(rem);
  28. for (String word : wordSet) {
  29. for (int i = 0; i <= word.length(); i++) {
  30. permutationsSet.add(charInsertAt(word, initial, i));
  31. }
  32. }
  33. }
  34. return permutationsSet;
  35. }
  36. public static String charInsertAt(String str, char c, int position)
  37. {
  38. String begin = str.substring(0, position);
  39. String end = str.substring(position);
  40. return begin + c + end;
  41. }
  42. }

16.编写一个 Java 程序来反转给定字符串的每个单词?

要分别反转每个单词,首先,对字符串进行标记,并在数组中将所有单词分开。 然后对每个单词应用反向单词逻辑,最后连接所有单词。

  1. String blogName = "how to do in java dot com";
  2. //spilt on white space
  3. String[] tokens = blogName.split(" ");
  4. //It will store reversed words
  5. StringBuffer finalString = new StringBuffer();
  6. //Loop all words and reverse them
  7. for (String token : tokens) {
  8. String reversed = new StringBuffer(token).reverse().toString();
  9. finalString.append(reversed);
  10. finalString.append(" ");
  11. }
  12. //Check final string
  13. System.out.println(finalString.toString()); //woh ot od ni avaj tod moc

17.如何在 Java 中拆分字符串?

使用String.split()方法可在给定正则表达式的匹配项附近中断给定字符串。 也称为基于定界符的获取字符串令牌

split()方法返回字符串数组。 数组中的每个字符串都是单独的标记。

  1. String numbers = "1,2,3,4,5,6,7";
  2. String[] numArray = numbers.split(",");
  3. System.out.println(Arrays.toString(numArray)); //[1, 2, 3, 4, 5, 6, 7]

18.为什么用Char数组而不是String来首选Char数组来存储密码?

我们知道字符串存储在 Java 的常量池中。 一旦在字符串池中创建了一个字符串,它将一直保留在该池中,直到收集到垃圾为止。 这时,任何恶意程序都可以访问物理内存位置中的内存位置,也可以访问字符串。

如果我们将密码存储为字符串,那么它也将存储在 spring 池中,并且在内存中的可用时间比要求的更长,因为垃圾收集周期是不可预测的。 这使得敏感的密码字符串容易遭到黑客攻击和数据盗窃

使用后可以将String留空吗? 不,我们不可以。 我们知道,一旦创建了字符串,我们将无法对其进行操作,例如,您不能更改其内容。 字符串是最终的且不可变的。

但是char数组是可变的,使用后它们的内容可以被覆盖。 因此,您的应用应使用char[]存储密码文本,并在使用密码后,将数组内容替换为空白。

  1. String password = "123456"; //Do not use it
  2. char[] passwordChars = new char[4]; //Get password from some system such as database
  3. //use password
  4. for(char c : passwordChars) {
  5. c = ' ';
  6. }

19. Java 中的字符串线程安全吗?

是的,字符串是线程安全的。 它们是不可变的,默认情况下,java 中所有不可变的实例都是线程安全的。

20.为什么String是 Java 中流行的HashMap键?

在 Java 中,必须在Map中使用的键是不可变的,并且应遵循equals()hashCode()方法之间的约定。 String类满足这两个条件。

另外,String类提供了许多有用的方法来比较,排序,标记化或小写。 在Map上执行 CRUD 操作时可以使用这些方法。 这使它成为在Map中使用而不是创建自己的类的非常有用的类。

21. StringStringBufferStringBuilder之间的区别?

  • String类表示字符序列,并提供了使用字符的有用方法。 字符串类实例是不可变的。 因此,每次使用字符串类执行字符串连接时,都会使用连接的字符串创建一个新对象。

  • StringBuilder类用于以更节省内存的方式执行字符串连接操作。 它在内部维护一个char数组并仅操作该数组中的内容。
    执行完所有操作后需要获取完整的连接字符串时,它将创建一个具有字符数组内容的新String

  • StringBufferStringBuilder类非常相似。 唯一的区别是它是线程安全的。 所有方法都是synchronized

22.如何连接多个字符串?

根据您是否需要线程安全来使用StringBufferStringBuilder类。 在两个类中都使用append()方法来连接字符串。

  1. StringBuffer buffer = new StringBuffer();
  2. buffer.append("how")
  3. .append("to")
  4. .append("do")
  5. .append("in")
  6. .append("java")
  7. .append(".")
  8. .append("com");
  9. String blogName = buffer.toString();
  10. System.out.println(blogName); //howtodoinjava.com

23.使用字符串初始化代码将创建多少个对象?

  1. String s1 = "howtodoinjava.com";
  2. String s2 = "howtodoinjava.com";
  3. String s3 = new String("howtodoinjava.com");
  1. 上面的代码将创建 2 个对象
  2. 第一个语句将在字符串池中创建第一个对象。
  3. 第二条语句不会创建任何新对象,并且s2将引用与s1相同的字符串常量。
  4. 第三条语句将在堆内存中创建一个新的字符串对象。

24.如何计算字符串中每个字符的出现次数?

为了找到给定字符串中每个字符的出现次数,我们使用了HashMap,并将该字符作为键,并将其出现作为值。 每次出现新的字符时,我们将增加该字符的值。

  1. String blogName = "howtodoinjava.com";
  2. HashMap<Character, Integer> occurancesMap = new HashMap<Character, Integer>();
  3. char[] strArray = blogName.toCharArray();
  4. for (char c : strArray)
  5. {
  6. if(occurancesMap.containsKey(c))
  7. {
  8. occurancesMap.put(c, occurancesMap.get(c)+1);
  9. }
  10. else
  11. {
  12. occurancesMap.put(c, 1);
  13. }
  14. }
  15. System.out.println(occurancesMap);
  16. //{a=2, c=1, d=1, h=1, i=1, j=1, m=1, n=1, .=1, o=4, t=1, v=1, w=1}

25.编写一个 Java 程序来不使用StringBuilderStringBuffer反转字符串?

反转字符串的最佳方法肯定是StringBuffer.reverse()StringBuilder.reverse()方法。 不过,面试官可能会要求您编写自己的程序,以检查您的技能水平。

使用下面给出的基于递归的示例来反转字符串。 该程序从字符串中获取第一个字符,并将其放置在字符串中的最后一个位置。 它将替换字符串中的所有字符,直到验证整个字符串为止。

  1. public class StringExample
  2. {
  3. public static void main(String[] args)
  4. {
  5. String blogName = "howtodoinjava.com";
  6. String reverseString = recursiveSwap(blogName);
  7. System.out.println(reverseString);
  8. }
  9. static String recursiveSwap(String str)
  10. {
  11. if ((null == str) || (str.length() <= 1))
  12. {
  13. return str;
  14. }
  15. return recursiveSwap(str.substring(1)) + str.charAt(0);
  16. }
  17. }

我可以想到这些常见的String面试问题将为您的下一次面试提供帮助。 如果您还有其他关于String类的问题,请分享。

学习愉快!