java记录一下字符串比较以及引申的设计模式缓冲区内容

java简单记录一下字符串比较的相关

本文在 内容空("")和 引用空(null)以及 双等号(==)和 equals 方法上进行记录

1.内容空和引用空

① 顾名思义,内容空即为String本身地址中没有存放任何东西,即没有内容,但是String是实例化的,是有地址的

例如:String str=new String(""); 其中str是真正指向一个存放String的地址的,只不过这个地址存放的String是空的

② 很好理解,与之对应的就是引用空,即String声明的引用只是声明了,并没有实例化,也就是没有指向任何地址

例如:String str;这时候的str就是引用空,只是声明了一个引用,他并没有指向实际的一个地

 

2.双等号和equals

理解了上述的一个简单的前提,再来说明最基础的字符串判断

先来看一个例子:

        String str1=new String("abc");
        String str2=new String("abc");
		
	System.out.println(str1==str2);
	System.out.println(str1.equals(str2));

以上声明并实例化了两个String字符串,并使其内容均为"abc",然后对比两种比较的输出结果

很明显,结果说明了一切(这里浪漫小白也想对大家说,有什么问题弄不懂,直接上代码跑一下结果,然后分析)

①双等号判断返回值false,也就是两者不等,原因很简单,双等号判断的是地址是否一致(顺带着也就判断了内容,哈哈。滑稽,地址都一样,内容能不一样?),而两个实例化的str1和str2显然是分配的两个不同的地址空间来存放abc字符串

②这就简单的说明了,equals判断的其实才是抛开地址后的字符串内容的比较,也就是地址不同,但内容相同的匹配

所以在判断字符串是否内容匹配的时候要用equals

这里需要说明的是:此处的equals只是针对string类来说的,为什么呢?

原因很简单,equals本身是Object的方法,而其在object的方法定义中,其逻辑就是双等号,有兴趣可以直接找源码看一下。

所以说,默认的equals其实也是和双等号一样的逻辑判断,但是我们经常使用到的jdk所提供的类库中的大部分都是重写了equals的逻辑判断,将其改为了内容判断,比如上述中的string类。这也说明了我们经常直接使用java所提供的类库中的equals大部分都是对equals进行过重写的!

理解了上述,这里多记录一种情况,也很好理解

就是,当一个字符串引用为空的时候,即String str=null的时候,是不能够用equals的,这个细节每个人都懂,因为没有实例化不能使用非静态方法,但是,作者周围的朋友却经常会遇到这一类问题,就是实际编程过程中,在判断字符串是否为空的时候,经常就直接用str.equals(""),要知道你的str不一定是声明的时候就创建的,很可能来源于其他任何可以使用字符串引用的地方!这时候要留意一下你的str是否是空指针

以上的说明,很简单,很基础,却很频繁,然后说明一下设计模式缓冲区问题

java本身的关于字符串初始化时候缓冲区相关的解释说明

还是看例子,

public class Test {

	public static void main(String a[])
	{
		//String str1=new String("abc");
		//String str2=new String("abc");
		String str1="abc";
		String str2="abc";
		
		System.out.println(str1==str2);
		System.out.println(str1.equals(str2));
	}
}

然后看结果,

哎,这一看就看出了关键,这时候Str1 == Str2 返回值是 true,用我们刚才理解的双等号判断的含义,就知道了,此时str1和str2

的指针指向是一致的,就是说这两个引用的实际地址是一样的

那这本质的原因是什么呢?其实就是一种缓存区的设计模式。

java中直接声明的实际字符串"abc"是存在在常量池位置的,并且是唯一的,像例子中的一样,str1=“abc”的时候,这时候abc其实本身就是一个直接使用的字面量,只不过又让str1这个引用指向了他(这个字符串放在了常量区)。str2又等于“abc”的时候,其实也是指向的刚才的那个常量池中abc的位置!

换句话说,就类似于现在有了一个实际的String类型的str3,然后新声明String str1=str3,String str2=str3,那么str1和str2当然都是指向了一个地址(str3)了!但是所谓的不同就是,这里的str3 是实例化的String,他的地址就是String应该在的地址,而我们例子中直接使用的“abc”字符串是放在常量池中的。

而对比之前的例子String str=new String();这样的方式,就没有这个问题了,因为new关键字,就实例化了一个真正的String,有自己的地址空间(在堆中),然后把abc这个内容放进这个地址空间中。

总结来说,就是直接使用(字面量)的字符串形式是存放在一种缓冲区的模式(常量池)中的,且相同内容的存放是唯一的。

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值