专栏首页java金融java采坑之路

java采坑之路

判断相等

字符串判断相等
        String str1 = null;
        String str2 = "java金融";
       // str1.equals(str2);  错误的写法
        str2.equals(str1); // 常量写前面
        Objects.equals(str1, str2);// 借助jdkUtil工具类
        StringUtils.equals(str1,str2); // 自定义工具类
  • 字符串判断相等我们记住一定要常量写前面。
  • 借助jdk提供的util帮助类(Objects)。
  • 自定义工具类,进行判空处理。
包装类判断相等
        Integer n1 = 100;
        Integer n2 = 100;
        System.out.println(n1 == n2);//true
        System.out.println(n1.equals(n2));//true
        Integer n3 = 200;
        Integer n4 = 200;
        System.out.println(n3 == n4);//false
        System.out.println(n3.equals(n4));//true

为什么n3== n4 是false呢?由于包装类的缓存机制。包装类的比较用equals去判断。最推荐的还是用工具类去判断。例如上面的列子如果n3=null的话n3.equals(n4)这时候就会抛出npe了。如果用工具类的话就不会存在这种情况。总之一句话判断相等如果不愿意去判空(偷懒、代码也不好看)就借助工具类。合理使用工具类可以使你的代码减少不必要的npe。

三目运算符

这个常见的坑的话就是由于自动拆箱导致的 NPE 异常。

BigDecimal

禁止使用浮点数double,float的初始化
        double d = 1.001;
        float f = 1.001f;
        BigDecimal bigDecimal1 = new BigDecimal(d);
        BigDecimal bigDecimal2 = new BigDecimal(f);
        System.out.println(bigDecimal1);
        System.out.println(bigDecimal2);

输出结果

1.000999999999999889865875957184471189975738525390625
1.00100004673004150390625

这个结果是不是跟我们所期望的1.001有点不一样。

float和double可以用于工程计算科学计算,他们会有精度丢失,这是由于浮点运算器的结构导致的,但是在金融领域一旦精度出现问题就意味着可能是严重的现实经济损失,所以普通的那些数值型一般不会在这个场景下使用。

所以涉及金钱的计算一定不要使用float和double。使用BigDecimal并且一定要用String来构造。 上面的列子我们可以这样来初始化 new BigDecimal("1.001")。

进行除法运算时必须要设置保留小数位
  BigDecimal a = new BigDecimal("1");
        System.out.println(a.divide(new BigDecimal(3)));

输出

Exception in thread "main" java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.
 at java.math.BigDecimal.divide(BigDecimal.java:1690)
 at com.workit.demo.antisper.Test.main(Test.java:11)

解决办法:使用如下两个函数设置精度

  • divide(num, scale)
  • divide(num, scale, roundingMode)
        BigDecimal a = new BigDecimal("1");
        System.out.println(a.divide(new BigDecimal(3), 2,BigDecimal.ROUND_HALF_UP));

字符串分隔(别忘了转义)

        String str = "java|php|c++";
        String[] split = str.split("|");
        for(String s:split){
            System.out.println(s);
        }

输出结果

j
a
v
a
|
p
h
p
|
c
+
+

结果并不是我们所期待的,java、php、c++。解决办法我们对|进行转义分割,代码改为 String[] split = str.split("\|");结果就正确了。String的split方法需要转义的字符串:. $ | ( ) [ { ^ ? * + \ 共12个特殊字符,遇到以这些字符进行分割字符串的时候,需要在这些特殊字符前加双反斜杠\ \。

Arrays.asList 需要谨慎使用

下面列举一些常用但是却与我们所期待的结果不一样的用法。

将基本类型数组作为asList的参数
   int[] array = {1,2,3};
        List list = Arrays.asList(array);
        System.out.println(list.size()); //1

输出的结果是1不是3哦是不是跟想象的有点不一样?原因如下: 由于Arrays.ArrayList参数为可变长泛型,而基本类型是无法泛型化的,所以它把int[] array 数组当成了一个泛型对象,所以集合中最终只有一个元素array 。

将数组作为asList参数后,修改数组或List
        String[] array = {"欢迎","关注","java金融"};
        List list = Arrays.asList(array);
        array[0] ="修改数组第一个元素";
        list.set(2,"修改集合第三个元素");
        System.out.println(Arrays.toString(array));
        System.out.println(list.toString());

输出结果

[修改数组第一个元素, 关注, 修改集合第三个元素]
[修改数组第一个元素, 关注, 修改集合第三个元素] 

是不是也与我们所期待的不一样。修改了数组奥了的值居然影响到了集合里面的值。原因如下: 由于asList产生的集合元素是直接引用作为参数的数组,所以当外部数组或集合改变时,数组和集合会同步变化,这在平时我们编码时可能产生莫名的问题。

数组转换为集合后,进行增删元素。
         String[] array = {"欢迎","关注","java金融"};
        List list = Arrays.asList(array);
        list.add("java金融");
        System.out.println(list.toString());

输出结果:

Exception in thread "main" java.lang.UnsupportedOperationException
 at java.util.AbstractList.add(AbstractList.java:148)
 at java.util.AbstractList.add(AbstractList.java:108)

抛出异常原因:由于asList产生的集合并没有重写add,remove等方法,所以它会调用父类AbstractList的方法,而父类的方法中抛出的却是异常信息。当我们使用Arrays.asList 产生的集合时候,需要谨慎的去使用。如果需要对集合进行操作的时候我们可以通过 List list = new ArrayList(Arrays.asList(array)); 来进行使用。

currenHaseMap注意 key和value的null值
       String key = "java金融";
        Map<String,Object> map = new ConcurrentHashMap<>();
        map.put("1","2");
        map.put(key,null);// Exception in thread "main" java.lang.NullPointerException

记得刚开始工作的时候,我负责的一个管理系统里面有一个关于省份的缓存,用HashMap来存的。大概就是项目一起动,然后就从db里面把省份信息加载到HashMap里面,以后需要用到省份信息直接从HashMap里面取,HashMap不是线程不安全吗?然后我反手就把它改成了currenHaseMap。测试环境测试没问题,然后就跟着其他功能上线。上完线之后也没有去回归关于省份的这一块内容,然后就下班了。第二天上班运营反映有部分注册用户的省份信息没了。leader就找我昨天有没有改过关于省份的代码,我说就改了一个currenHaseMap。leader先让昨天上线代码回退,一回退省份信息就有了。后面经过仔细排查原来生产数据库有一条省份信息是空的。然后加载那条空的省份信息到currenHaseMap就报空指针了,在这条空记录后面信息就没加载到currenHaseMap了。幸好是内网管理系统没有造成太大的影响 。

string.valueof

    String  userName= String.valueOf(parmMap.get("userName"));
        if(StringUtils.isNotBlank(userName)) {
            sql.append(" and tt.userName like %").append(userName);
        }

这里的 parmMap.get("userName") 如果是 null , 那么这里的 userName就是 “null” ,这是一个值为 null 的字符串。导致数据会拼接到SQL 中,导出出错。为什么会这样我们看下源码就知道了。

 public static String valueOf(Object obj) {
        return (obj == null) ? "null" : obj.toString();
    }

所以转字符串的时候我们要根据实际的情况来选择合适的方法。

总结

本文列举了一些对于java常见的一些可能稍微不注意就会采坑的一些知识点。还有其他更多需要注意的知识点也欢迎大家来补充。其实这些常见的采坑基本上只要去看下源码都能够避免的。

本文分享自微信公众号 - java金融(java4299),作者:java金融

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2020-05-17

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 在项目中随手把haseMap改成了currenHaseMap差点被公司给开除了。

    所以涉及金钱的计算一定不要使用float和double。使用BigDecimal并且一定要用String来构造。 上面的列子我们可以这样来初始化 new Big...

    java金融
  • 本地缓存高性能之王Caffeine

    随着互联网的高速发展,市面上也出现了越来越多的网站和app。我们判断一个软件是否好用,用户体验就是一个重要的衡量标准。比如说我们经常用的微信,打开一个页面要十几...

    java金融
  • 本地缓存性能之王Caffeine

    随着互联网的高速发展,市面上也出现了越来越多的网站和app。我们判断一个软件是否好用,用户体验就是一个重要的衡量标准。比如说我们经常用的微信,打开一个页面要十几...

    java金融
  • 在项目中随手把haseMap改成了currenHaseMap差点被公司给开除了。

    所以涉及金钱的计算一定不要使用float和double。使用BigDecimal并且一定要用String来构造。 上面的列子我们可以这样来初始化 new Big...

    java金融
  • SAPJCO3升级3.1后报错java.lang.UnsatisfiedLinkError: sapjco3.dll解决

    SAPJCO官方站点 The SAP Java Connector 3.1 requires a JDK/JRE 8 or 11. In addition, ...

    路过君
  • 类A是公共的,应在名为A.java的文件中声明错误

    第一种!!! “类A是公共的,应在名为A.java的文件中声明”这句话需要分两步来理解: 1、如果类A被声明为公共的(public),那么必须将类A保存在名为...

    Angel_Kitty
  • 【SAP PI系列】IDOC发送状态03,PI没有收到消息的解决办法

    matinal
  • Java基础-常用类(二)

    版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 ...

    cwl_java
  • 你真的会用ABAP, Java和JavaScript里的constructor么?

    如果constructor里调用了一个成员方法,这个方法被子类override了,当初始化一个子类实例时,父类的构造函数被的调用,此时父类构造函数的上下文里调用...

    Jerry Wang
  • 你真的会用ABAP, Java和JavaScript里的constructor么?

    如果constructor里调用了一个成员方法,这个方法被子类override了,当初始化一个子类实例时,父类的构造函数被的调用,此时父类构造函数的上下文里调用...

    Jerry Wang

扫码关注云+社区

领取腾讯云代金券