Java字符串类(java.lang.String)是Java中使用最多的类,也是最为特殊的一个类,很多时候,我们对它既熟悉又陌生。
一.从根本上认识java.lang.String类和String池
首先,建议先看看String类的源码实现,这是从本质上认识String类的根本出发点。从中可以看到:
1.String类是final的,不可被继承。public final class String。
2.String类是的本质是字符数组char[], 并且其值不可改变。private final char value[];
然后打开String类的API文档,可以发现:
3.String类对象有个特殊的创建的方式,就是直接指定比如String x = "abc","abc"就表示一个字符串对象。而x是"abc"对象的地址,也叫做"abc"对象的引用。
4.String对象可以通过“+”串联。串联后会生成新的字符串。也可以通过concat()来串联,这个后面会讲述。
5.Java运行时会维护一个String Pool(String池),JavaDoc翻译很模糊“字符串缓冲区”。String池用来存放运行时中产生的各种字符串,并且池中的字符串的内容不重复。而一般对象不存在这个缓冲池,并且创建的对象仅仅存在于方法的堆栈区。下面是个系统内存示意图:
a.使用new关键字创建字符串,比如String s1 = new String("abc");
b.直接指定。比如String s2 = "abc";
c.使用串联生成新的字符串。比如String s3 = "ab" + "c";
String对象的创建也很讲究,关键是要明白其原理。
原理1:当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个s在String池中找是否存在内容相同的字符串对象,如果不存在,则在池中创建一个字符串s,否则,不在池中添加。
原理2:Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。
原理3:使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。
原理4:使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。
另外,String的intern()方法是一个本地方法,定义为public native String intern(); intern()方法的价值在于让开发者能将注意力集中到String池上。当调用 intern 方法时,如果池已经包含一个等于此String 对象的字符串(该对象由equals(Object)方法确定),则返回池中的字符串。否则,将此String对象添加到池中,并且返回此String对象的引用。
实例1:
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { /* * "sss111"是编译期常量,编译时已经能确定它的值,在编译 好的class文件中它已经在String Pool中了,此语句会在 * String Pool中查找等于"sss111"的字符串(用equals(Object)方法确定), * 如果存在就把引用返回,付值给s1.不存在就会创建一个"sss111"放在 String Pool中,然后把引用返回,付值给s1. */ String s1 = "sss111"; // 此语句同上 String s2 = "sss111"; /* * 由于String Pool只会维护一个值相同的String对象 上面2句得到的引用是String Pool中同一个对象,所以 他们引用相等 */ System.out.println(s1 == s2); // 结果为true } }
实例2:
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { /* * 在java中,使用new关键字会创建一个新对象,在本例中,不管在 String Pool中是否已经有值相同的对象,都会创建了一个新的 * String对象存储在heap中,然后把引用返回赋给s1. 本例中使用了String的public String(String * original)构造函数. */ String s1 = new String("sss111"); /* * 此句会按照例1中所述在String Pool中查找 */ String s2 = "sss111"; /* * 由于s1是new出的新对象,存储在heap中,s2指向的对象 存储在String Pool中,他们肯定不是同一个对象,只是 * 存储的字符串值相同,所以返回false. */ System.out.println(s1 == s2); // 结果为false } }
实例3:
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { String s1 = new String("sss111"); /* * 当调用intern方法时,如果String Pool中已经包含一个等于此String对象 的字符串(用 * equals(Object)方法确定),则返回池中的字符串.否则,将此 String对象添加到池中,并返回此String对象在String * Pool中的引用. */ s1 = s1.intern(); String s2 = "sss111"; /* * 由于执行了s1 = s1.intern(),会使s1指向String Pool中值为"sss111" * 的字符串对象,s2也指向了同样的对象,所以结果为true */ System.out.println(s1 == s2); } }
实例4:
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { String s1 =new String("111"); String s2 ="sss111"; /* * 由于进行连接的2个字符串都是常量,编译期就能确定连接后的值了, * 编译器会进行优化直接把他们表示成"sss111"存储到String Pool中, * 由于上边的s2="sss111"已经在String Pool中加入了"sss111", * 此句会把s3指向和s2相同的对象,所以他们引用相同.此时仍然会创建出 * "sss"和"111"两个常量,存储到String Pool中. */ String s3 ="sss"+"111"; /* * 由于s1是个变量,在编译期不能确定它的值是多少,所以 * 会在执行的时候创建一个新的String对象存储到heap中, * 然后赋值给s4. */ String s4 ="sss"+ s1; System.out.println(s2 == s3); //true System.out.println(s2 == s4); //false System.out.println(s2 == s4.intern()); //true } }
实例5:
StringTest.java
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { String hello = "Hello", lo = "lo"; System.out.println((hello == "Hello") + ""); //true System.out.println((Other.hello == hello) + ""); //true System.out.println((com.bijian.study.other.Other.hello == hello) + ""); //true System.out.println((hello == ("Hel" + "lo")) + ""); //true System.out.println((hello == ("Hel" + lo)) + ""); //false System.out.println(hello == ("Hel" + lo).intern()); //true } } class Other { static String hello = "Hello"; }
Other.java
package com.bijian.study.other; public class Other { public static String hello ="Hello"; }
四.认识trim()、concat()
实例:
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { // 在池中和堆中分别创建String对象"abc",s1指向堆中对象 String s1 = new String("abc"); // s2直接指向池中对象"abc" String s2 = "abc"; // 在堆中新创建"abc"对象,s3指向该对象 String s3 = new String("abc"); // 在池中创建对象"ab" 和 "c",并且s4指向池中对象"abc" String s4 = "ab" + "c"; // c指向池中对象"c" String c = "c"; // 在堆中创建新的对象"abc",并且s5指向该对象 String s5 = "ab" + c; String s6 = "ab".concat("c"); String s7 = "ab".concat(c); System.out.println("------------实串-----------"); System.out.println(s1 == s2); // false System.out.println(s1 == s3); // false System.out.println(s2 == s3); // false System.out.println(s2 == s4); // true System.out.println(s2 == s5); // false System.out.println(s2 == s6); // false System.out.println(s2 == s7); // false } }
五.认识空格、空串、null
实例:
package com.bijian.study.string; public class StringTest { public static void main(String[] args) { String b1 = new String(""); String b2 = ""; String b3 = new String(""); String b4 = "".intern(); String b5 = "" + ""; String b6 = "".concat(""); String b7 = " ".trim(); String b8 = " "; String b9 = " ".trim(); System.out.println("------------空串-----------"); System.out.println(b1 == b2); //false System.out.println(b1 == b3); //false System.out.println(b2 == b3); //false System.out.println(b2 == b4); //true System.out.println(b2 == b5); //true System.out.println(b2 == b6); //true System.out.println(b2 == b7); //false System.out.println("-----a----"); System.out.println(b2.equals(b7)); //true System.out.println(b7 == b8); //false System.out.println(b7 == b9); //false System.out.println(b7.equals(b9)); //true System.out.println(b9 == null);//false System.out.println("b8.trim():"); for (byte b : b8.getBytes()) { System.out.print(">>>" + (int) b + " "); //>>>32 >>>32 } System.out.println("\nb8.trim():"); for (byte b : b8.trim().getBytes()) { System.out.print(">>>" + (int) b + " "); // } System.out.println("\nb9.trim():"); for (byte b : b9.trim().getBytes()) { System.out.print(">>>" + (int) b + " "); // } } }
运行结果:
------------空串----------- false false false true true true false -----a---- true false false true false b8.trim(): >>>32 >>>32 b8.trim(): b9.trim():
六.String的常见用法
1.字符串重编码
这个问题说来比较简单,转码就一行搞定,不信你看看,但究竟为什么要转码,是个很深奥的问题,看实例:
package com.bijian.study.string; import java.io.UnsupportedEncodingException; public class StringTest { public static void main(String[] args) throws UnsupportedEncodingException { System.out.println("转码前,输出Java系统属性如下:"); System.out.println("user.country:" + System.getProperty("user.country")); System.out.println("user.language:" + System.getProperty("user.language")); System.out.println("sun.jnu.encoding:" + System.getProperty("sun.jnu.encoding")); System.out.println("file.encoding:" + System.getProperty("file.encoding")); System.out.println("---------------"); String s = "字符串测试"; String s1 = new String(s.getBytes(), "UTF-8"); String s2 = new String(s.getBytes("UTF-8"), "UTF-8"); String s3 = new String(s.getBytes("UTF-8")); String s4 = new String(s.getBytes("UTF-8"), "GBK"); String s5 = new String(s.getBytes("GBK")); String s6 = new String(s.getBytes("GBK"), "GBK"); System.out.println(s1); System.out.println(s2); System.out.println(s3); System.out.println(s4); System.out.println(s5); System.out.println(s6); } }
运行结果:
转码前,输出Java系统属性如下: user.country:CN user.language:zh sun.jnu.encoding:GBK file.encoding:UTF-8 --------------- 字符串测试 字符串测试 字符串测试 瀛楃涓叉祴璇� �ַ���� 字符串测试
小结:
a.转一个码,又用该码来构建一个字符串,是绝对不会出现乱码的,----你相当于没转。
b.转码与否,与字符串本身编码有关,字符串本身的编码与谁有关?----文件编码,或者你的IDE设置的编码有关。
在此,我用的IDEA开发工具,默认是UTF-8编码,但操作系统使用的是GBK,但没有问题,我只要按照UTF-8来读取我的字符串就不会有乱码。但是文件已经是UTF-8了,你非要转为GBK,不乱才怪!那有什么办法呢?在Windows下,用记事本或者Editplus打开后另存为(并修改编码方式即可)。
2.字符比较
不就是个匹配关系吗?String类的API有一些可以做比较,如果不行,可以寻求正则表达式来解决。
3.获取某个字符
获取一个字符序列toCharArray() ,然后就随便玩去吧,中文就乱了。
4.字符串的截取
substring()
5.字符串的替换与查找
6.开始结束判断
startsWith()/endWith()
7.字符串的排序比较
compareTo(String anotherString):按字典顺序比较两个字符串。
compareToIgnoreCase(String str):不考虑大小写,按字典顺序比较两个字符串。
8.字符串的equals()和hashCode()
已经实现了好了,直接调用,不用重写
9.字符串的类型转化
太多了,String.valueOf()系列很多。
类似的Long.parseLong(String s)
10.字符串的复制
copyValueOf()
11.大小写转换
toLowerCase()
toUpperCase()
12.正则匹配
相关推荐
本教程"深入学习C++_String2.1版"旨在帮助开发者更全面、深入地理解`std::string`类及其在实际编程中的应用。以下是对这个主题的详细探讨: 一、`std::string`类的基本概念 `std::string`是一个容器类,用于存储可...
### 深入学习C++ String #### 一、C++的string的使用 ##### 1.1 C++ string简介 在C++中,`std::string`是用于处理文本数据的标准库的一部分,它提供了丰富的功能来管理和操作字符串。与传统的C风格字符串不同,`...
深入学习C++ `string`类是非常有价值的,因为它涉及到许多实际编程场景,如文本处理、输入/输出操作等。在这个主题中,我们将深入探讨C++ `string`的特性、操作和常见用法。 1. **`string`类的基本概念**: - `std...
**Stringboot 空白项目示例及jar包详解** Stringboot 是一个假设性的轻量级框架,可能是基于Java开发的,旨在简化Web应用...通过运行和修改这个示例,你可以深入学习Stringboot框架,并将其应用到实际的项目开发中。
在以后的学习中,我们可以继续深入学习 String 类的实现原理,并掌握更多的字符串操作方法。 附录 在本实验报告中,我们提供了详细的实现过程和结果,我们可以通过这些结果来了解 String 类的实现原理,并掌握基于...
《深入学习C++ string》2.1版详细解读 C++中的`std::string`是C++标准库中一个非常重要的容器,它用于存储和操作文本字符串。在本篇文章中,我们将深入探讨`std::string`的使用,涵盖其成员函数、构造、重载运算符...
在这个主题中,我们将深入探讨一些关键的`string`函数,如`strcpy`、`strcat`、`strcmp`、`strlen`等,并了解它们的用法和注意事项。 首先,`strcpy`函数用于将一个字符串完全复制到另一个字符串中。它的原型是`...
为了更好地应对这些问题,我们需要深入学习string类的内部工作机制,包括它的内存管理、迭代器操作、大小调整策略等。例如,了解何时会发生动态内存分配以及COW策略的具体实现,可以帮助我们避免潜在的性能瓶颈和...
这个简单的`string`类介绍将带我们回顾基础,深入理解其内部机制和常用方法。 首先,`std::string`是C++标准库中的一个类模板,它表示可变长度的字符序列。在声明时,你可以通过构造函数初始化字符串,例如: ```...
在深入探讨`String`类之前,先要理解C#与SQL Server数据库技术的结合,因为这两个领域的知识是现代软件开发中的基础。`String`类在处理SQL查询时经常扮演关键角色,例如构建SQL语句、存储和检索数据库中的文本数据。...
本篇将基于提供的`String1.java`代码示例,深入探讨如何创建并操作字符串对象。 #### 一、程序概述 `String1.java`程序的主要功能是: 1. 接受用户的输入,并将其存储为字符串。 2. 对该字符串执行两个不同的方法...
在C++编程中,`std::string`是一个...通过阅读和理解源代码,开发者可以学习到如何实现自定义的字符串格式化功能,以及如何编写测试用例来验证其正确性。这对于提升C++编程技能和深入理解字符串处理是非常有价值的。
在本教程中,我们将深入探讨STRING数据库的主要功能、使用方法以及在生物信息学研究中的应用。 STRING数据库的核心功能是提供蛋白质之间的相互作用信息。这些相互作用包括物理相互作用和功能关联,涵盖了多种物种,...
本篇文章将深入探讨如何在Android中进行Bitmap与String之间的转换,以及这些转换在实际应用中的场景。 首先,我们要理解Bitmap与String转换的原理。Bitmap是Android系统中用来表示图像的类,它包含了图像的所有像素...
通过实际运行和调试这些代码,可以更深入地理解`byte[]`与`String`之间的转换过程。 综上所述,理解和熟练掌握`byte[]`与`String`之间的转换,是Android开发中的一项基础技能,能帮助开发者处理各种数据交换和存储...
Java中的`String`类是编程中最常用的类之一,它在Java的`java.lang`包中,无需显式导入即可使用。`String`类代表不可变的字符序列,这意味着...通过深入学习和实践,你可以更好地利用`String`类来处理各种文本数据。
`std::string`类不仅在日常编程中频繁使用,而且在很多高级功能如正则表达式、文本处理等方面都有重要应用,因此了解其构造方法对于深入理解C++字符串处理至关重要。 首先,让我们来看看`std::string`的几个主要...
在计算机科学领域,字符编码是数据处理和存储中不可或缺的一部分,UTF-8和字符串(string)之间的转换是常见的操作。UTF-8是一种广泛使用的Unicode字符编码,它可以表示Unicode字符集中的所有字符,而字符串则通常指的...