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=24*60*60*1000*1000;
       final long MILLIS_PER_DAY=24*60*60*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=24L*60*60*1000*1000;
       final long MILLIS_PER_DAYS=24L*60*60*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);
  }
}
相关文章
|
Java
Java一些常见的坑
总是觉得自己Java基础还是不行,需要恶补。今天偶然mark了一本《Java解惑》,其中以端程序的方式罗列了95个即常见又不常见的xian(坑)jing(儿),拿来瞻仰一下。
65 0
|
Java
Java常见的坑(二)
你猜上述程序输出的是什么? 是 ABC easy as 123 吗? 你执行了输出操作,你才发现输出的是 ABC easy as [C@6e8cf4c6 ,这么一串丑陋的数字是什么鬼? 实际上我们知道字符串与任何数值的相加都会变为字符串,上述事例也不例外, numbers输出其实实际上是调用了Object.toString()方法,让numbers转变为'[c' + '@' + 无符号的十六进制数。
68 0
Java常见的坑(二)
你猜上述程序输出的是什么? 是 ABC easy as 123 吗? 你执行了输出操作,你才发现输出的是 ABC easy as [C@6e8cf4c6 ,这么一串丑陋的数字是什么鬼? 实际上我们知道字符串与任何数值的相加都会变为字符串,上述事例也不例外, numbers输出其实实际上是调用了Object.toString()方法,让numbers转变为'[c' + '@' + 无符号的十六进制数。
|
8天前
|
安全 Java 测试技术
Java并行流陷阱:为什么指定线程池可能是个坏主意
本文探讨了Java并行流的使用陷阱,尤其是指定线程池的问题。文章分析了并行流的设计思想,指出了指定线程池的弊端,并提供了使用CompletableFuture等替代方案。同时,介绍了Parallel Collector库在处理阻塞任务时的优势和特点。
|
17天前
|
安全 Java
java 中 i++ 到底是否线程安全?
本文通过实例探讨了 `i++` 在多线程环境下的线程安全性问题。首先,使用 100 个线程分别执行 10000 次 `i++` 操作,发现最终结果小于预期的 1000000,证明 `i++` 是线程不安全的。接着,介绍了两种解决方法:使用 `synchronized` 关键字加锁和使用 `AtomicInteger` 类。其中,`AtomicInteger` 通过 `CAS` 操作实现了高效的线程安全。最后,通过分析字节码和源码,解释了 `i++` 为何线程不安全以及 `AtomicInteger` 如何保证线程安全。
java 中 i++ 到底是否线程安全?
|
5天前
|
安全 Java 开发者
深入解读JAVA多线程:wait()、notify()、notifyAll()的奥秘
在Java多线程编程中,`wait()`、`notify()`和`notifyAll()`方法是实现线程间通信和同步的关键机制。这些方法定义在`java.lang.Object`类中,每个Java对象都可以作为线程间通信的媒介。本文将详细解析这三个方法的使用方法和最佳实践,帮助开发者更高效地进行多线程编程。 示例代码展示了如何在同步方法中使用这些方法,确保线程安全和高效的通信。
25 9
|
8天前
|
存储 安全 Java
Java多线程编程的艺术:从基础到实践####
本文深入探讨了Java多线程编程的核心概念、应用场景及其实现方式,旨在帮助开发者理解并掌握多线程编程的基本技能。文章首先概述了多线程的重要性和常见挑战,随后详细介绍了Java中创建和管理线程的两种主要方式:继承Thread类与实现Runnable接口。通过实例代码,本文展示了如何正确启动、运行及同步线程,以及如何处理线程间的通信与协作问题。最后,文章总结了多线程编程的最佳实践,为读者在实际项目中应用多线程技术提供了宝贵的参考。 ####
|
4天前
|
监控 安全 Java
Java中的多线程编程:从入门到实践####
本文将深入浅出地探讨Java多线程编程的核心概念、应用场景及实践技巧。不同于传统的摘要形式,本文将以一个简短的代码示例作为开篇,直接展示多线程的魅力,随后再详细解析其背后的原理与实现方式,旨在帮助读者快速理解并掌握Java多线程编程的基本技能。 ```java // 简单的多线程示例:创建两个线程,分别打印不同的消息 public class SimpleMultithreading { public static void main(String[] args) { Thread thread1 = new Thread(() -> System.out.prin
|
8天前
|
Java
JAVA多线程通信:为何wait()与notify()如此重要?
在Java多线程编程中,`wait()` 和 `notify()/notifyAll()` 方法是实现线程间通信的核心机制。它们通过基于锁的方式,使线程在条件不满足时进入休眠状态,并在条件满足时被唤醒,从而确保数据一致性和同步。相比其他通信方式,如忙等待,这些方法更高效灵活。 示例代码展示了如何在生产者-消费者模型中使用这些方法实现线程间的协调和同步。
22 3
|
6天前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。