Java一些常见的坑

简介: 总是觉得自己Java基础还是不行,需要恶补。今天偶然mark了一本《Java解惑》,其中以端程序的方式罗列了95个即常见又不常见的xian(坑)jing(儿),拿来瞻仰一下。

前言:写公号差不多也有一个月了,没有做到每天坚持(真的太难了),但是养成了会有新知识就会记录下来的习惯,感谢读者(你们),也欣赏一下我自己。

总是觉得自己Java基础还是不行,需要恶补。今天偶然mark了一本《Java解惑》,其中以端程序的方式罗列了95个即常见又不常见的xian(坑)jing(儿),拿来瞻仰一下。

解惑一、奇数性判断

判断一个数是否为奇数


public class SolutionOne{

   /**
    * 判断下列方法是否能够很好的运转?
    *
    * 不能
    * 因为int整型中有一半是负数,也就是说
    * i 为负数,那么返回值永远为false
    */
   public static boolean isOdd(int i){
       return i % 2 == 1;
  }

   /**
    * 这个方法能够很好的运转,
    * 因为不管对正数和负数来说,余数为0 -- 偶数
    * 不为0 -- 奇数
    */
   public static boolean isNew(int i){
       return i % 2 != 0;
  }

   /**
    * 使用位操作符 来替代取余操作符,能够改善性能
    * note:
    * 5转换为二进制:0000 0000 0000 0000 0000 0000 0000 0101
    * 3转换为二进制:0000 0000 0000 0000 0000 0000 0000 0011
    * -------------------------------------------------------------------------------------
    * 1转换为二进制:0000 0000 0000 0000 0000 0000 0000 0001
    * 位与:第一个操作数的的第n位于第二个操作数的第n位如果都是1,那么结果的第n为也为1,否则为0
    */
   public static boolean isChange(int i){
       return(i & 1)  != 0;
  }

   public static void main(String[] args) {
       System.out.println(isOdd(3));
       System.out.println(isOdd(-1));
       System.out.println(isOdd(-2));
       System.out.println("------------------");
     
       System.out.println(isNew(3));
       System.out.println(isNew(-1));
       System.out.println(isNew(-2));
       System.out.println("------------------");
     
       System.out.println(isNew(3));
       System.out.println(isNew(-1));
       System.out.println(isNew(-2));
  }
}


解惑二、找零时刻

试问:能否用2.00美元买到1.10美元的热狗?


public class SolutionTwo{


public static void main(String[] args) {
// 请验证一下以下输出 输出 0.8999999999999999
System.out.println(2.00-1.10);

// 解决办法一、输出 0.90
System.out.printf("%.2f%n ",2.00-1.10);

// 解决办法二、使用正数类型,int or long 输出 0.90
System.out.println((200-110) +" cents");

// 解决办法三、使用更精确精度的BigDecimal,一定要用BigDecimal(String) 而不要用BigDecimal(Double)
System.out.println(new BigDecimal("2.00").subtract(new BigDecimal("1.10")));
}
}


解惑三、长整除


public class SolutionThree{

public static void main(String[] args) {

// 打印输出以下语句,会输出什么呢?
// 并不是你预料的1000, 而是 5
final long MICROS_PER_DAY=24606010001000;
final long MILLIS_PER_DAY=246060*1000;
System.out.println(MICROS_PER_DAY/MILLIS_PER_DAY);

/**
* 发生了什么呢? 为什么差着1000倍两个常量的数值相除会得到 5?
*
* 因为上述常数MICROS_PER_DAY的计算发生溢出,虽然计算的结果适合
* 放入long中,并且其空间还有富余,但是这个结果并不适合放入int中,
* 所以计算的结果是按着int来执行的。运算完成之后,转换为long类型
* 但此时已经太迟,计算已经溢出,它返回了一个小于200倍的数值。从
* int提升为long是一种拓宽原生类型转换,它保留了不正确的计算数值
* 于是整除的结果为5
*
*/

// 下面的结果是正确的结果
final long MICROS_PER_DAYS=24L606010001000;
final long MILLIS_PER_DAYS=24L6060*1000;
System.out.println(MICROS_PER_DAYS/MILLIS_PER_DAYS);

}
}


解惑四、初级问题


public class SolutionFour{

public static void main(String[] args) {

/**
* 下面的问题简直太简单了,打印出来肯定是66666,
* 但是是吗?输出的结果确实17777,你会感到惊讶,
* 为什么是17777呢?请注意 + 号右面的数字,是5432 l 而不是1
* 注意到差别了吗? 所以以后为了产生这种微小的误差
* 建议把l -> L
*/
System.out.println(12345+5432l);

/**
* 此外,还要避免使用 l 作为对象
*/
List<String> l = new ArrayList<>();
l.add("Foo");
System.out.println(l);
}
}


解惑五、最后的笑声(对应书中的解惑十一)


public class SolutionEleven{

public static void main(String[] args) {
/**
* 下面的程序将打印什么?
* 第一个输出语句将打印 Ha
* 第二个输出语句打印 169
* 为什么会这样呢?
*
* 第一个输出语句是 两个字符串进行拼接,拼接的结果就是 Ha
* 第二个输出语句是 两个字符型的常量的加和,所以使用的是加法
* 而不是拼接, 对于'H',char的数值是72,对于'a',char的数值是97
* 所以二者的加和是 169
*
*/
System.out.println("H"+"a");
System.out.println('H'+'a');

/**
* 改变一、字符串与任何数的计算都会转换为字符串
*/
System.out.println(""+'H'+'a');

// 这条输出语句真的打印的是2 + 2 = 4 吗?
System.out.println("2 + 2 = "+2+2);
}
}
            </div>
目录
相关文章
|
存储 安全 Java
Java 总结
Java 总结
72 0
|
分布式计算 JavaScript Java
学JAVA,从现在开始-day01
每日记录自己学习java的心得和体会~
|
存储 消息中间件 负载均衡
JAVA问答6
JAVA问答6
124 0
|
分布式计算 Java API
赶快看看Java11,不然你就out了!
由于直接从Java8跨越到Java11,所以特性介绍就把Java9-Java11的部分特性一起介绍一下。想要了解Java8特性的朋友可以去我的博客找「Java8系列」。
656 3
赶快看看Java11,不然你就out了!
AbstractDemo.java
AbstractDemo.java
71 0
1105 链表合并(JAVA)
给定两个单链表 L1​=a1​→a2​→⋯→an−1​→an​ 和 L2​=b1​→b2​→⋯→bm−1​→bm​。如果 n≥2m,你的任务是将比较短的那个链表逆序,然后将之并入比较长的那个链表,得到一个形如 a1​→a2​→bm​→a3​→a4​→bm−1​⋯ 的结果。例如给定两个链表分别为 6→7 和 1→2→3→4→5,你应该输出 1→2→7→3→4→6→5。
1105 链表合并(JAVA)
|
Java 测试技术
1070 结绳(JAVA)
给定一段一段的绳子,你需要把它们串成一条绳。每次串连的时候,是把两段绳子对折,再如下图所示套接在一起。这样得到的绳子又被当成是另一段绳子,可以再次对折去跟另一段绳子串连。每次串连后,原来两段绳子的长度就会减半。
1070 结绳(JAVA)
|
SQL 负载均衡 Oracle
[一起学Java]
一、批量添加品牌信息到数据表 一、业务实现类实现
165 0
|
数据安全/隐私保护 Android开发
java32-巩固练习
java32-巩固练习
108 0
java32-巩固练习
|
Java
java24-if..else...if
java24-if..else...if
103 0
java24-if..else...if