stringbuilder在头和末尾加字符_面试宝典:java中字符串,你真的了解吗?

面试中,经常会被问到:String、StringBuffer、StringBuilder有什么区别?你心中有答案了吗?

54d44ebe71586f8d15b2927cf78630f2.png

比较典型的回答

String是java语言中比较重要的类,他是典型的Immutable类,被声明为 final class ,并且它的所有属性也被 final 。也正是由于它的不可变性,像字符串的拼接都会产生新的String对象。于是在 java1.5 中新增 StringBuffer,为了解决拼接中产生太多中间对象。

在StringBuffer中可以用append或者add方法,将字符串添加到字符串的末尾或者指定的位置。但是由于StringBuffer是一个线程安全,它在方法定义前面都会有synchronize关键字,这样也带来了额外的性能开销,于是乎又提供了StringBuilder

我们唯一需要注意的是:StringBuffer是线程安全的,StringBuilder是线程不安全的。

应用场景:

【A】在字符串内容不经常发生变化的业务场景优先使用String类,如果有大量的字符串内容拼接,优先使用StringBuilder,但如果需要考虑线程的安全的情况下,优先使用StringBuffer

【B】在频繁进行字符串的运算,并且在多线程环境下,建议使用StringBuffer,如:XML解析、Http参数解析封装等

【C】在频繁字符串的运算,且在单线程环境下,建议使用StringBuilder,如:SQL语句拼接

从面试官的角度分析考察点

【1】通过String相关的类,考察基本的线程安全问题

【2】String类的演进,如:String到StringBuffer,再到StringBuilder,甚至考察在java9中实现的巨大变化

【3】考察JVM对象的缓存机制和理解,以及如何更好的使用

知识点的扩展

【1】 为什么这样设计String ?

String 是典型的 Immutable类,这样保证了基础的线程安全,因为这样你就无法对它的内部数据进行修改,在拷贝时也不需要额外的赋值数据。

再来看下StringBuffer实现,它是线程安全的,在各个方法都加上 synchronize 关键字。虽然比较粗暴,但非常适合常见的线程安全的类实现,有的人可能担心 synchronize 的性能问题,其实没有必要,考虑到代码的可靠性、正确性,这是大多数开发者的手段。

【2】String的缓存机制

由于java中的String使用的比较频繁,所以为了避免在一个系统中产生大量的String对象,引入了字符串常量池。它的运行机制,创建一个字符串时,首先检查池中是否有值相同的字符串对象,如果有,则不需要创建直接从池中刚查找到的对象引用;如果没有则新建字符串对象,返回对象引用,并且将新创建的对象放入池中。

注意:通过new方法创建的String对象是不检查字符串池的,而是直接在堆区或栈区创建一个新的对象,也不会把对象放入池中。

举例: //通过直接量赋值方式,放入字符串常量池String str1 = "我和初恋的故事"; //通过new方式赋值方式,不放入字符串常量池String str2 = new String("我和初恋的故事"); 

你会了今天讨论的面试题了吗?

由于自身的能力有限,还有很多与字符串相关的问题没有深入探讨,但大家可以在讨论区继续回复讨论,写下你们对这个面试题的思考,或者分享下你在操作字符串时掉过的坑?谢谢~~~

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
可以使用字符串来表示长整数,然后进行减运算。具体步骤如下: 1. 将两个字符串表示的长整数补齐到相同长度,即在较短的字符串前面 0,使得两个字符串的长度相等。 2. 从字符串末尾开始,逐位相,同时考虑上一位的进位。如果某一位相的结果大于等于 10,则需要进位。 3. 将相的结果逆序输出,得到最终的结果。 下面是 Java 代码实现: ```java public static String add(String s1, String s2) { // 将两个字符串表示的长整数补齐到相同长度 int len = Math.max(s1.length(), s2.length()); s1 = leftPad(s1, len); s2 = leftPad(s2, len); StringBuilder sb = new StringBuilder(); int carry = 0; // 进位 for (int i = len - 1; i >= 0; i--) { int sum = carry + s1.charAt(i) - '0' + s2.charAt(i) - '0'; // 逐位相 sb.append(sum % 10); // 计算当前位的值 carry = sum / 10; // 计算进位 } if (carry != 0) { sb.append(carry); // 如果最高位有进位,需要补上 } return sb.reverse().toString(); // 将相的结果逆序输出 } public static String subtract(String s1, String s2) { // 将两个字符串表示的长整数补齐到相同长度 int len = Math.max(s1.length(), s2.length()); s1 = leftPad(s1, len); s2 = leftPad(s2, len); StringBuilder sb = new StringBuilder(); int borrow = 0; // 借位 for (int i = len - 1; i >= 0; i--) { int diff = s1.charAt(i) - borrow - s2.charAt(i); // 逐位相减 if (diff < 0) { // 需要借位 diff += 10; borrow = 1; } else { borrow = 0; } sb.append(diff); // 计算当前位的值 } while (sb.charAt(sb.length() - 1) == '0' && sb.length() > 1) { sb.deleteCharAt(sb.length() - 1); // 去掉结果的前导 0 } return sb.reverse().toString(); // 将相减的结果逆序输出 } private static String leftPad(String s, int len) { StringBuilder sb = new StringBuilder(s); while (sb.length() < len) { sb.insert(0, '0'); // 在字符串前面补 0 } return sb.toString(); } ``` 这里使用了 StringBuilder 类来保存间结果,leftPad 方法用于在字符串前面补 0,subtract 方法还需要去掉结果的前导 0。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值