String类型的创建机制
由于是第一篇博客,首先介绍一下自己吧,博主是某高校研究生,方向是医疗软件开发(本科是生物方向),刚开始学编程半年,一直想写博客却又怕才疏学浅,迟迟未敢动笔,但是丑媳妇总要见公婆,今天终于狠狠心写了第一篇博客,如有错误希望大家多多指正。
基本类型和引用类型
首先科普一下JAVA类型的有关知识,JAVA中类型可以分为两大类,基本类型和引用类型,基本类型有9大类:
基本类型 | 大小 | 最小值 | 最大值 | 包装器类型 |
---|---|---|---|---|
boolean | - | - | - | Boolean |
char | 16 bit | Unicode 0 | Unicode 2^16-1 | Character |
byte | 8 bits | -128(-2^7) | 127(2^7-1) | Byte |
short | 16 bits | -2^15 | 2^15-1 | Short |
int | 32 bits | -2^31 | 2^31-1 | Integer |
long | 64 bits | -2^63 | 2^63-1 | Long |
float | 32 bits | - | 有效数字6位 | Float |
double | 64 bits | - | 有效数字15位 | Double |
void | - | - | - | Void |
除了这9个基本类型之外则都是引用类型,基本类型和引用类型的变量都存储于栈上,那么基本类型跟引用类型的区别是什么呢?区别在于基本类型的数据(值)也存储于栈上,基本类型的变量直接存储值;而引用类型的数据(对象)存储于堆上,引用类型的变量存储的是数据的地址;所以基本变量是值传递,引用变量是地址传递(实际上也是值的传递,不过是地址值),这也是为什么基本类型可以直接用“==“比较,而类一般要根据需要重写equals方法来进行的比较。
String类型的创建机制
基本类型并不包括String,说明String是一个引用类型,但是为什么网上有那么多对String类型的争论呢?首先看下面一段代码:
public static void main(String[] args) {
String A = "hello";
String B = "hello";
System.out.println(A==B);//true
B = A;
B ="world";
System.out.println(A);//"hello"
很奇怪是不是?String的表现完全像一个基本类型,这是为什么呢?
首先介绍String对象的两种创建方法:1,String A = “hello
2,String A = new String(“hello”)
那么这两种方法有什么区别呢?方法1要调用字符串常量池,而方法2会忽略掉字符串常量池。
字符串常量池位于堆上,当使用方法1时,当需要创建一个新的字符串的时候(不用new),会自动去常量池找是否有相同的字符串,如果有就直接引用,没有就重新创建一个对象并放入池中。
当创建A时,字符串常量池中没有“hello”,所以就创建了一个字符串对象”hello”;
当创建B时,字符串常量池中有“hello”,所以就直接把B指向字符串对象“hello”;
此时A和B都指向同一个”hello”(即引用变量A,B储存的都是字符串对象”hello”的地址),故A==B为true;
B = A,确保B和A共同引用一个对象;B=“world”,常量池中无“world”字符串对象,重新创建,并将B指向“world”字符串对象。所以A自然还是指向“hello”,如图所示。
那么且看下面一段代码:
String A = new String("hello");
String B = "hello";
System.out.println(A==B);//false
你看,A和B是不是都指向“hello”对象,但是他们却并不是指向同一个“hello”对象,原因就是用了new,new略过了字符串常量池,直接创建“hello”字符串对象,导致B在字符串常量池中没有找到“hello”对象,重新创建了一个,所以A和B并不是指向同一个对象。