《Java解惑》书摘
例子1:关于char数组的输出
System.out.println("H" + "a");//输出:Ha
System.out.println('H' + 'a');//输出:169
System.out.println("" + 'H' + 'a');//输出:Ha
System.out.println("//////////////////");
System.out.println("2 + 2 = " + 2 + 2);//输出:2 + 2 = 22
System.out.println("2 + 2 = " + (2 + 2));//输出:2 + 2 = 4
System.out.println("//////////////////");
char[] char1 = {'1', '2', '3'};
Object char2 = new char[] {'1', '2', '3'};
System.out.println(char1);//输出:123
System.out.println(char2);//输出:[C@55e83f9
System.out.println("test-" + char1);//输出:test-[C@55e83f9
System.out.println("test-" + char1.toString());//输出:test-[C@55e83f9
System.out.println("test-" + String.valueOf(char1));//输出:test-123
System.out.println("//////////////////");
例子2:
Random random = new Random();
StringBuffer sBuffer = null;
switch(random.nextInt(2)) {
case 1:sBuffer = new StringBuffer('A');
case 2:sBuffer = new StringBuffer('B');
default:sBuffer = new StringBuffer('C');
}
sBuffer.append('a');
sBuffer.append('b');
sBuffer.append('c');
System.out.println(sBuffer);//输出:abc
}
这段程序有三个BUG:
- Random.nextInt(int n)方法的使用错误:
先看下这个方法的API:
public int nextInt(int n)
返回一个伪随机数,它是取自此随机数生成器序列的、在 0(包括)和指定值(不包括)之间均匀分布的 int 值。
所以,此处应改为:switch(random.nextInt(3)) { - case子句中没有break语句:
导致前面的赋值被后面的赋值覆盖。 - StringBuffer类的构造方法使用错误:
StringBuffer不支持传入参数为字符(char)的构造方法,应改为传入字符串(String),
所以,应改为:sBuffer = new StringBuffer("A");
例子3:会死循环的代码
//死循环1
int start1 = Integer.MAX_VALUE-1;
for(int start7=start1; start7<=start1+1; start7++) {
System.out.println("loop..");
}
//死循环2
double start2 = 1.0 / 0.0;
double start3 = Double.POSITIVE_INFINITY;//与上句等价,可替代
while(start2 == (start2+1)) {
System.out.println("loop..");
}
//死循环3
double start4 = 0.0 / 0.0;
double start5 = Double.NaN;//与上句等价,可替代
while(start4 != start4) {
System.out.println("loop..");
}
//死循环4
String start6 = "test";
while(start6 != (start6+0)) {
System.out.println("loop..");
}
//死循环5
byte start7 = -1;//死循环
// short start7 = -1;//死循环
// int start7 = -1;//循环32次
// long start7 = -1;//循环64次
int count = 0;//循环次数
while(start7 != 0) {
start7 >>>= 1;
System.out.println("loop..");
count++;
}
System.out.println("循环次数 = " + count);
//死循环6
int start8 = Integer.MIN_VALUE;
// Long start8 = Long.MIN_VALUE;
while(start8 != 0 && start8 == -start8) {
System.out.println("loop..");
}
补充说明:
- 关于Infinity和NaN(死循环2和3)
double d = 1.0 / 0; System.out.println(d); //输出:Infinity System.out.println(d + 1); //输出:Infinity System.out.println(d == d + 1); //输出:true d = 0.0 / 0; System.out.println(d); //输出:NaN System.out.println(d + 1); //输出:NaN System.out.println(d == d + 1); //输出:false System.out.println(Double.NaN == Double.NaN);//输出:false Double a = new Double(Double.NaN); Double b = new Double(Double.NaN); System.out.println(a.equals(b));//输出:true
- 关于Java的自动窄化原生类型转换(死循环5)
分析 short start7 = -1; 和 start7 >>>= 1;这两句代码:
在执行移位操作时,第一步是将start7提升为start7nt类型,因为所有的算术操作都会对short,byte,char类型的操作数执行这样的提升;
这种提升是拓宽了原生类型,没有信息的损失,执行的是符号扩展,start7的值提升后是0xffffffff,start7>>>1之后得到的是0x7fffffff;
将这个值再存入start7中的时候,为了将int数值存入short变量,Java自动执行了可怕的窄化原生类型,直接把高16位截掉,剩下(short)0xffff,又回到了起点,从而导致了死循环。
这提醒了我们,不要在short,byte,char类型上使用复合赋值操作符,很容易出问题,而且这种隐性的窄化类型转换,在丢失信息的同时,结果也可能是灾难性的。 - 关于Java中的特殊值(死循环6)
Java使用2的补码的算术运算是不对称的。
对于每一种有符号的整数类型(byte、short、int和long),负的数值总是比正的数值多一个,这个多出来的数值总是这个类型所能表示的最小值;
对Integer.MIN_VALUE取负值并不会改变它的值,对这个值取负值将会产生溢出,但是Java自动忽略了这种溢出;同理对Long.MIN_VALUE等也是如此。
《Java解惑》书摘的更多相关文章
- Code Simplicity–The Science of Software Development 书摘
Chapter1 Introduction That is the art and talent involved in programming—reducing complexity to simp ...
- 《CODE》书摘
2016-11-08 14:59:16 可以说英语词汇就是一种编码. 2016-11-08 15:19:04 实际上任何两种不同的东西经过一定的组合都可以代表任何种类的信息. 2016-11-08 1 ...
- 《C Elements of Style》 书摘
<C Elements of Style> 书摘 学完C语言和数据结构后,虽然能解决一些问题,但总觉得自己写的程序丑陋,不专业.这时候看到了Steve Oualline写的<C El ...
- Visual Studio Code 代理设置
Visual Studio Code (简称 VS Code)是由微软研发的一款免费.开源的跨平台文本(代码)编辑器,在十多年的编程经历中,我使用过非常多的的代码编辑器(包括 IDE),例如 Fron ...
- 我们是怎么做Code Review的
前几天看了<Code Review 程序员的寄望与哀伤>,想到我们团队开展Code Review也有2年了,结果还算比较满意,有些经验应该可以和大家一起分享.探讨.我们为什么要推行Code ...
- Code Review 程序员的寄望与哀伤
一个程序员,他写完了代码,在测试环境通过了测试,然后他把它发布到了线上生产环境,但很快就发现在生产环境上出了问题,有潜在的 bug. 事后分析,是生产环境的一些微妙差异,使得这种 bug 场景在线下测 ...
- 从Script到Code Blocks、Code Behind到MVC、MVP、MVVM
刚过去的周五(3-14)例行地主持了技术会议,主题正好是<UI层的设计模式——从Script.Code Behind到MVC.MVP.MVVM>,是前一天晚上才定的,中午花了半小时准备了下 ...
- 在Visual Studio Code中配置GO开发环境
一.GO语言安装 详情查看:GO语言下载.安装.配置 二.GoLang插件介绍 对于Visual Studio Code开发工具,有一款优秀的GoLang插件,它的主页为:https://github ...
- 代码的坏味道(14)——重复代码(Duplicate Code)
坏味道--重复代码(Duplicate Code) 重复代码堪称为代码坏味道之首.消除重复代码总是有利无害的. 特征 两个代码片段看上去几乎一样. 问题原因 重复代码通常发生在多个程序员同时在同一程序 ...
- http status code
属于转载 http status code:200:成功,服务器已成功处理了请求,通常这表示服务器提供了请求的网页 404:未找到,服务器未找到 201-206都表示服务器成功处理了请求的状态代码,说 ...
随机推荐
- form-validation-engine中的正则表达式
form-validation-engine是一个不错的表单验证,可以玩玩. (function($) { $.fn.validationEngineLanguage = function() {}; ...
- Linux系统编程(20)——信号基本概念
信号及信号来源 信号是在软件层次上对中断机制的一种模拟,在原理上,一个进程收到一个信号与处理器收到一个中断请求可以说是一样的.信号是异步的,一个进程不必通过任何操作来等待信号的到达,事实上,进程也不知 ...
- Spring(三)——AOP
AOP全名为Aspect-Oriented Programming,意思是面向横切面编程,前边我们有过介绍 面向横切面编程AOP的理解 ,我们通过这种编程思想很容易的扩展我们的应用程序. 一,如何 ...
- JVM工作原理和特点(一些二逼的逼神面试官会问的问题)
作为一种阅读的方式了解下jvm的工作原理 ps:(一些二逼的逼神面试官会问的问题) JVM工作原理和特点主要是指操作系统装入JVM是通过jdk中Java.exe来完毕,通过以下4步来完毕JVM环境. ...
- Redis 命令参考
Redis 命令参考 http://redis.readthedocs.org/en/latest/index.html
- Java - 反射机制(Reflection)
Java - 反射机制(Reflection) > Reflection 是被视为 动态语言的关键,反射机制允许程序在执行期借助于 Reflection API 取得任何类的 ...
- OS X EI Capitan 安装mysql-5.7.9
先去下载mysql-5.7.9-osx10.9-x86_64.dmg安装(一直下一步,输入密码即可)mysql-5.7.9-osx10.9-x86_64.pkg好了,启动MySQL服务. Untitl ...
- WebApi 文件上传
1. 注意给form表单加上enctype = "multipart/form-data" 属性,否则会导致Action的参数HttpPostedFileBase 对象接收不到文件 ...
- 归并排序java
import java.util.Arrays; public class MergeSort { public static void main(String[] args) { MergeSort ...
- OCMOCM
14年,OCM考试费12000 15年,考试费19800 对于我来说,1,2年之后是否换工作还是个未知数 在本单位考这个貌似没什么用处,工资也不会突然就涨很多 跳槽的话,专门做数据库感觉压力还挺大 年 ...