java怎么判断数据类型_Java 数据类型及其判断

一、Java中的数据类型,可分为两类:

1.基本数据类型,也称原始数据类型:

byte、short、char、int、long、float、double、boolean它们之间的比较,应该用双等号(==)比较的是它们的值。

7d4ba86ff885

2.引用数据类型:

JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较变量(栈)内存中存放的对象的(堆)内存地址,用来判断两个对象的地址是否相同,即是否是指相同一个对象。比较的是真正意义上的指针操作。

但在一些类库当中这个方法被覆盖掉了,如String、Integer、Date。在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。

对于引用数据类型之间进行equals比较,在没有重写equals方法的情况下,它们之间的比较还是基于它们在内存中的存放位置的地址值的,因为Object的equals方法也是用双等号进行比较的,所以比较后的结果跟双等号的结果相同。

注意:

比较的是操作符两端的操作数是否是同一个对象。

两边的操作数必须是同一类型的(可以是父子类之间)才能编译通过。

比较的是地址。如果是具体的阿拉伯数字的比较,值相等则为true,如:

int a=10 与 long b=10L 与 double c=10.0都是相同的(为true),因为他们都指向地址为10的堆。

String s=“abc"是一种非常特殊的形式,和new 有本质的区别。它是java中唯一不需要new就可以产生对象的途径。以 String s=“abc”; 形式赋值在java中叫直接量,它是在常量池中而不是象new一样放在压缩堆中。这种形式的字符串,在JVM内部发生字符串拘留,即当声明这样的一个字符串后,JVM会在常量池中先查找有有没有一个值为"abc"的对象。如果有,就会把它赋给当前引用。即原来那个引用和现在这个引用指向了同一对象。如果没有,则在常量池中新创建一个"abc”,下一次如果有 String s1 = “abc”; 又会将s1指向"abc"这个对象,即以这种形式声明的字符串,只要值相等,任何多个引用都指向同一对象。

而 String s = new String(“abcd”); 和其它任何对象一样,每调用一次就产生一个对象,只要它们调用。

也可以这么理解: String str = “hello”; 先在内存中找是不是有"hello"这个对象,如果有,就让str指向那个"hello"。如果内存里没有"hello",就创建一个新的对象保存"hello"。String str=new String (“hello”) 就是不管内存里是不是已经有"hello"这个对象,都新建一个对象保存"hello"。

具体可以看下面的代码:

public class test1 {

public static void main(String[] args) {

String a = new String("ab"); // a 为一个引用

String b = new String("ab"); // b为另一个引用,对象的内容一样

String aa = "ab"; // 放在常量池中

String bb = "ab"; // 从常量池中查找

if (aa == bb){

System.out.println("aa==bb");// true

}

if (a == b){

System.out.println("a==b");// false,非同一对象

}

if (a.equals(b)){

System.out.println("aEQb");// true

}

if (42 == 42.0) {

System.out.println("true");15 // true

}

}

}

二、equals和==的区别

equals方法最初是在所有类的基类Object中进行定义的,源码:

public boolean equals(Object obj) {

return (this == obj);

}

由equals的源码可以看出这里定义的equals与双等号是等效的(Object类中的equals与双等号没什么区别)。不同的原因就在于有些类(像String、Integer等类)对equals进行了重写,但是没有对equals进行重写的类(比如我们自己写的类)就只能从Object类中继承equals方法,其equals方法与==就也是等效的,除非在此类中重写equals。

对equals重新需要注意五点:

1️⃣自反性:对任意引用值X,x.equals(x)的返回值一定为true。

2️⃣对称性:对于任何引用值x、y,当且仅当y.equals(x)返回值为true时,x.equals(y)的返回值一定为true。

3️⃣传递性:如果x.equals(y)=true,y.equals(z)=true,则x.equals(z)=true。

4️⃣一致性:如果参与比较的对象没任何改变,则对象比较的结果也不应该有任何改变。

5️⃣非空性:任何非空的引用值X,x.equals(null)的返回值一定为false 。

String类对equals的重写如下:

public boolean equals(Object var1) {

if (this == var1) {

return true;

} else {

if (var1 instanceof String) {

String var2 = (String) var1;

int var3 = this.value.length;

if (var3 == var2.value.length) {

char[] var4 = this.value;

char[] var5 = var2.value;

for (int var6 = 0; var3-- != 0; ++var6) {

if (var4[var6] != var5[var6]) {

return false;

}

}

return true;

}

}

return false;

}

}

另外,双等号比"equals"运行速度快,因为双等号"=="只是比较引用。

三、经典案例

如果Integer a = 3;Integer b = 3;那么【a==b】的结果是什么?-----true

如果Integer a = 273;Integer b = 273;那么【a==b】的结果是什么?-----false

原因:对于Integer var = ?在-128至127范围内的赋值,Integer对象是在 IntegerCache.cache产生,会复用已有对象,这个区间内的Integer值可以直接使用==进行判断,但是这个区间之外的所有数据,都会在堆上产生,并不会复用已有对象,推荐使用equals方法进行判断。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值