前言
当你路过一些商场或者地铁口的时候,有没有被千篇一律的"xx健身,了解一下" 所烦到。
无论在什么编程语言里面,字符串类型一直都是我们使用频率非常高的一个类型,在Java语言里面也不例外,今天我们不打广告而是重新认识一下我们的老朋友String类。
String类被封装在java.lang包里面,在Java里面每一个创建出来的字符串它的类型都是String,它最大的特点就是不可变(immutable ),这意味String类一旦创建就不能再修改,如果看过其源码就会发现String Class类是用关键字final修饰的并且其主要的成员变量也都是使用final修饰的。
什么是不可变对象
不可变对象一旦创建它的状态就不能再修改,我们常用的基本类型的包装类如Integer,Byte, Short, Float, Double都是不可变的。
下面看一个不可变类的例子:
在上面这个例子中MyString类就是不可变的,一旦实例化之后str的值就不能变化。
当然Java里面的String底层是用char数组+UTF-16编码存储的,这个在后面会提到。
创建String类的常用方法
(1)使用字面量创建,是我们最常用的方法
(2)通过一个String对象
(3)使用new关键词
(4)使用+号操作符
创建一个String类时发生了什么
每当我们创建一个String字面量时,jvm(java虚拟机)都会先检查字符串池(string pool)里面是否已经存在该字符串,如果已经存在,jvm会返回这个实例的引用,如果在池里不存在那么新的字符串就会被创建,然后添加到string池中,这个string池我们可以理解它是一块常量的字符串池,在JDK7之后是被放在java堆内存里面,当然避免创建重复的实例是jvm层面对字符串创建的一个优化。
下面看下字符串对象时怎么被存储的:
上面的这段代码创建了一个字符串字面量,如果是第一次创建,这个实例会被存放到字符串常量吃里面:
紧接着,当我们再次创建一个的字符串时,jvm实际上会把已经存在实例的引用赋值给新的字符串:
当然,如果我们改变字符串的内容,那么他的引用也会对应变化:
字符串连接
这里有2种方法可以连接两个或者更多的字符串:
(1)使用 concat() 方法
(2)使用 + 操作符
字符串比较
字符串比较这里有3种方式:
(1)使用equals方法
注意equals方法比较特殊,因为这个Object类里面的方法,Java里面所有的类都直接或者间接的继承了Object类,如果继承之后没有重写equals方法,那么默认比较的两个对象的内存地址。
通过观察源码我们能发现String里面的equals方法已经被重写过,它比较的是字符串的内容。
JDK8中的源码如下:
根据源码,我们知道内容相等返回true,否则就是false
(2)使用==方法
==方法比较的是两个对象的引用,也就是内存地址,如果内存地址一样,就返回true,否则就返回false
下面看个例子:
细心的朋友可能已经看出来s1和s3有一样的内容,但是却返回false,是因为他们的内存地址不一样,上面说过基于字符串字面量是放在字符串常量池里面,但是如果使用new出来的实例是放在堆内存里面,所以他们的内存指针是不一样的。
(3)使用compareTo方法
compareTo方法是基于自然顺序(通常指字母表的顺序)比较两个字符串的先后,它返回是int类型的值分别是:-1(排名靠前),0(排名相等),1(排名靠后)
总结:
本篇文章介绍了Java语言里面String类的特点和一些常见的基础用法,此外还介绍了创建一个String实例时其在JVM里面的是如何优化执行和存储的,这里面主要涉及关于字符串常量池的知识,这个会在下一篇文章中单独介绍。
领取专属 10元无门槛券
私享最新 技术干货