Java之字符串实践

这篇具有很好参考价值的文章主要介绍了Java之字符串实践。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

功能概述

  • 字符串是Java编程中常用的数据类型,本文对String部分常见功能做了对应实践以及分析。

功能实践

场景1:字符串比较

用例代码
@Test
public void test_string_compare() {
    String s1 = "abc";
    String s2 = s1;
    String s5 = "abc";
    String s3 = new String("abc");
    String s4 = new String("abc");
    System.out.println("s1 == s5: " + (s1 == s5));
    System.out.println("s1 == s2: " + (s1 == s2));
    System.out.println("s1.equals(s2): " + s1.equals(s2));
    System.out.println("s3 == s4: " + (s3 == s4));
    System.out.println("s1.equals(s4): " + s1.equals(s4));
    System.out.println("s3.equals(s4): " + s3.equals(s4));
}
运行结果
s1 == s5: true
s1 == s2: true
s1.equals(s2): true
s3 == s4: false
s1.equals(s4): true
s3.equals(s4): true
结果分析
  • “==” 比较基本类型时,判断值是否相等,比较对象类型时,判断是否指向同一内存地址。
  • equals是比较两个对象是否一样(即所有成员的值是否相同)。
  • 字符串,如"abc"是放在常量池中的,在内存中只存在一份副本,所以s1 == s5,同一个字符串,只有一个内存地址(会在常量池中检查是否存在已有的字符串,存在时,返回原有字符串的内存地址,就不会新建新的内存地址了)。
  • s3、s4指向的对象都是通过new创建的新对象,内存地址不一样,所以比较为false。

场景2:字符串intern方法使用

用例代码
@Test
public void test_intern_v1() {

    // 未使用intern时
    String s1 = new String("aaa777");
    String s2 = "aaa777";
    System.out.println(s1 == s2);

    // 使用intern时
    String s3 = new String("aaa777");
    s3 = s3.intern();
    String s4 = "aaa777";
    System.out.println(s3 == s4);
}
运行结果
false
true
结果分析
  • s1是指向堆中对象的引用,s2是指向常量池中字符串的引用,所以两者的内存地址不一样。
  • intern()会查找常量池中是否存在"aaa777"的引用,若存在返回,否则把String对象添加到池中,再返回在池中的引用。

场景3:String、StringBuffer、StringBuilder性能比较

用例代码
@Test
public void test_String_StringBuffer_StringBuilder_efficiency() {
    testString();
    testStringBuffer();
    testStringBuilder();
}

private void testString() {
    String s = "Hello";
    String s1 = "World";
    long start = System.currentTimeMillis();
    for (int i = 0; i < 10000; i++) {
        s = s + s1;
    }
    long end = System.currentTimeMillis();
    long runtime = (end - start);
    System.out.println("testString runtime:" + runtime);
}

private void testStringBuilder() {
    StringBuilder s = new StringBuilder("Hello");
    String s1 = "World";
    long start = System.currentTimeMillis();
    for (int i = 0; i < 10000; i++) {
        s.append(s1);
    }
    long end = System.currentTimeMillis();
    long runtime = (end - start);
    System.out.println("testStringBuilder runtime:" + runtime);
}

private void testStringBuffer() {
    StringBuffer s = new StringBuffer("Hello");
    String s1 = "World";
    long start = System.currentTimeMillis();
    for (int i = 0; i < 10000; i++) {
        s.append(s1);
    }
    long end = System.currentTimeMillis();
    long runtime = (end - start);
    System.out.println("testStringBuffer runtime:" + runtime);
}
运行结果
testString runtime:646
testStringBuffer runtime:1
testStringBuilder runtime:0
结果分析
  • 在执行方面来看:StringBuilder最高、StringBuffer次之、String最低。

场景4:StringTokenizer字符串分割

用例代码
@Test
public void test_StringTokenizer() {
    StringTokenizer st2 = new StringTokenizer("Hello&World&!","&"); //指定自定义分隔符
    while (st2.hasMoreTokens()) {
        System.out.println(st2.nextToken());
    }
}
运行结果
Hello
World
!
结果分析
  • StringTokenizer可按指定分隔符,第字符串进行分隔。

场景5:字符串与字符数组比较

用例代码
@Test
public void test_string_with_char_array_compare() {
    String s = "hello";
    String t = "hello";
    char c[] = {'h','e','l','l','o'};
    System.out.println(s.equals(t));
    System.out.println(t.equals(c));
    System.out.println(s == t);
    System.out.println(t.equals(new String("hello")));

    String s1 = "he" + "llo";
    System.out.println(s == s1);
}
运行结果
true
false
true
true
true
结果分析
  • s.equals(t)值为true:因为"hello"会存在堆中的字符串常量池中,并且按照享元模式,判断字符串是否存在,不存在才创建,所以s、t是同一个字符串引用
  • t.equals©值为false:按照String重写的equals方法,会判断入参是否为String类型,若不是则返回false
  • s == t值为true:==比较的是引用,因为s、t是同一个引用,所以相等
  • t.equals(new String(“hello”))值为true:因为String的比较逻辑,会去字符串中的字符数组来比较,字符都相等,则返回true
  • s == s1值为true:因为"he" + “llo"在编译器就会转化"hello”,存在常量区,因为s已经放在常量池中,与s1不会创建新的字符串,s与s1是同一个字符串引用

功能总结

  • String常量池采用了享元模式(Flyweight)即会共享字符串,判断字符串是否存在,若存在则使用,否则新创建。常量池:它是一个由数组组成的表,用来存储程序中使用的各种常量,包括Class、String、Integer等各种基本的Java数据类型

  • intern()方法分析:(intern:助理、实习生)

    • intern方法主要把字符串放入字符串常量池中。有两种情况,会将字符串放在常量池中

      • 直接使用双引号声明的String对象会直接存储在常量池中,如:String s1 = “aa” 。
      • 通过调用String提供的intern方法把字符串放在常量池中,会检查字符串在常量池中是否存在,存在则不处理,若不存在则放入字符串常量池中。
    • Returns a canonical representation for the string object. (jdk中的描述)

      • 根据jdk中描述:intern()会返回字符串对象的规范标识,即会从常量池中查找指定字符串,若能找到则返回对应引用,否则把String对象添加到池中,再返回在池中的引用。
    • 注明

      • a)字符串放入字符串常量池,需要满足上面所说的两种情况。
      • b)并不是调用了intern就会将字符串放入字符串常量池中,会先检查字符串是否存在,若存在则不处理。若不存在,要看字符串在堆中是否已存在,已存在只存对象引用,减少对象的创建 c)字符串常量池,在jdk1.7后就从Perm区迁移到堆中,这样对于字符串常量池中可以存对象引用,减少了对象的创建,就大大减少了字符串所占的空间了。
  • String、StringBuilder、StringBuffer使用总结:文章来源地址https://www.toymoban.com/news/detail-677667.html

    • 如果要操作的数据量比较小,优先使用String类(量多的话,String本质是使用StringBuilder处理的,会产生许多临时对象,触发垃圾回收,影响性能)
    • 如果是单线程下处理大量数据,优先使用StringBuilder类(StringBuilder是线程不安全的,单线程下可以使用,减少线程同步的开销)
    • 如果是多线程下处理大量数据,优先使用StringBuffer类(因为是线程安全的)

到了这里,关于Java之字符串实践的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包