由一次代码调试引发的对JAVA数值包装类型的使用的思考

简介: 由一次代码调试引发的对JAVA数值包装类型的使用的思考

问题的背景是这样的,判断数据库表中的一列是否有值以及该值是否是0,该列是数值型的,JAVA类中的类型是Long,注意这里使用的包装类。实际业务中的代码实现如下:

if(list!=null&&list.size()>0){
for(PoObjectobj:list){
if(obj.getFlowId()!=null&&!obj.getFlowId().equals(0)){
obj.setUrl(url.replace("aaa",String.valueOf(obj.getFlowId())));
        }
    }
}

从数据库表中查询出数据列表,然后遍历,判断是否flowId不为空且不为0的时候,对url属性做一个replace。url属性是显示到前端做跳转使用的。数据库表里flowId对应的列值都是0,启动项目的时候,发现页面上url地址都有值,这样就都做了跳转。本意是为0了,前端的url属性就是空的,提示暂无链接。那么问题出现在哪里了呢,就是!obj.getFlowId().equals(0)这个判断,这个判断对于flowId为0的时候返回是正确的,这就奇怪了,为啥0!=0是对的呢?其实是错用了equals方法。正确的写法应该是

!obj.getFlowId().equals(0L)。JDK中对于Long的equals的实现如下:

/*** Compares this object to the specified object.  The result is* {@code true} if and only if the argument is not* {@code null} and is a {@code Long} object that* contains the same {@code long} value as this object.** @param   obj   the object to compare with.* @return  {@code true} if the objects are the same;*          {@code false} otherwise.*/publicbooleanequals(Objectobj) {
if (objinstanceofLong) {
returnvalue== ((Long)obj).longValue();
        }
returnfalse;
    }

0不是属于Long的,而0L是属于Long的,同样要注意的是Float的类型,0f和0的类型在Float的equals中也不是一样的概念。除了!obj.getFlowId().equals(0L)可以以外,还可以使用obj.getFlowId().compareTo(0L)==0去判断。这里面有个要注意的点就是Long的compareTo方法参数是Long类型的,所以如果你写compareTo(0)会报编译错误,强制你使用0L,而equeals方法的参数是Object类型的,所以不会报编译错误。由这点可以看出来编译器暴露出问题来的重要性了。可以用如下的测试用例测试一下几个包装类的比较:

publicstaticvoidmain(String [] args) {
Longid=0L;
Floata=0f;
longid2=0;
System.out.println(id2==0); //trueSystem.out.println(a.equals(0)); //falseSystem.out.println(a.equals(0f)); //trueSystem.out.println(id.equals(0)); //falseSystem.out.println(id.equals(0L)); //trueSystem.out.println(id.compareTo(0L)); //0    }

虽然是个小问题,但是刚入行的同学找了半天没找出来,还感觉很奇怪。也不是什么大问题,更不要怀疑JDK会出bugs或者会给你模棱两可的答案,从自己提供的参数去考虑,以及使用的方法执行的逻辑输出的结果是否是预期的结果这方面的入手。看看源码确实是个不错的成长途径。可能以后会遇到Integer的比较的时候会存在小于还是大于127的缓存的问题,希望能避坑。

目录
相关文章
|
9天前
|
存储 Java 开发者
Java 中 Set 类型的使用方法
【10月更文挑战第30天】Java中的`Set`类型提供了丰富的操作方法来处理不重复的元素集合,开发者可以根据具体的需求选择合适的`Set`实现类,并灵活运用各种方法来实现对集合的操作和处理。
|
9天前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
20 2
|
1月前
|
Java 编译器
Java“返回类型为 void 的方法不能返回一个值”解决
在 Java 中,如果一个方法的返回类型被声明为 void,那么该方法不应该包含返回值的语句。如果尝试从这样的方法中返回一个值,编译器将报错。解决办法是移除返回值语句或更改方法的返回类型。
|
1月前
|
设计模式 Java
Java“不能转换的类型”解决
在Java编程中,“不能转换的类型”错误通常出现在尝试将一个对象强制转换为不兼容的类型时。解决此问题的方法包括确保类型间存在继承关系、使用泛型或适当的设计模式来避免不安全的类型转换。
|
30天前
|
Java
Java 中锁的主要类型
【10月更文挑战第10天】
|
1月前
|
安全 Java
Java“不兼容类型” 错误怎么查找解决
在 Java 中遇到“不兼容类型”错误时,首先理解错误信息,它表明试图将一种类型赋给不兼容的类型。检查代码中类型不匹配的赋值、方法调用参数类型不匹配、表达式类型不兼容及泛型类型不匹配等问题。解决方法包括进行类型转换、修改代码逻辑、检查方法参数和返回类型以及处理泛型类型不匹配。通过这些步骤,可以有效解决“不兼容类型”错误,确保代码类型兼容性良好。
|
1月前
|
Java 程序员 编译器
Java中的异常类型
Java中的异常类型
23 3
|
1月前
|
Java 开发者
Java“类 Y 中的方法 X 不能应用于给定类型”解决
在Java中遇到“类Y中的方法X无法应用于给定类型”的错误时,通常是因为方法调用时的参数类型与定义不符。解决此问题需检查方法签名,确保传递的参数类型正确无误,或使用显式类型转换以匹配方法所需的参数类型。这种错误提示帮助开发者及时修正类型不匹配的问题。
|
1月前
|
Java 编译器
Java“无效的方法声明;需求返回类型”怎解决
要解决Java中的“无效的方法声明;需要返回类型”错误,需为方法指定正确的返回类型。检查方法签名,添加如`void`、`int`、`String`等类型,并确保方法体内正确使用`return`语句。这能帮助Java编译器理解和验证方法的行为。遵守这些规则,可以避免语法错误并确保程序正常运行。
|
1月前
|
Java
java包装器类型
java包装器类型
17 0