《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都表示服务器成功处理了请求的状态代码,说 ...
随机推荐
- hdu 4619 Warm up 2 ( 二分图最大匹配 )
题目:Warm up 2 题意:有横竖两种方式放着的多米诺骨牌,相同方向的不可能重叠,但是横放和竖放 的牌可能重叠.移走重叠的牌使剩下的牌最多. 分析:二分图匹配:最大独立集= ...
- 【sql语句】好用的sql语句之项目数据库学习总结
转载请注明出处:http://blog.csdn.net/pearyangyang/article/details/41115491 这几天学习公司系统的数据流向.主要涉及到几个表的数据. 可是表中的 ...
- HDU 4978 A simple probability problem
A simple probability problem Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K ( ...
- 执行此安装程序之前,必须安装 32 位 Windows 映像处理组件(WIC)解决的方法
我们在Windows Service 2003上安装 Microsoft .NET Framework4.0时常常出现以下的报错 执行此安装程序之前,必须安装 32 位 Windows 映像处理组件( ...
- 武道释义 · 零散
若是你防御你的左边.你的右边必定是弱点: 若是你防御你的前面,你的后面必定是弱点. 若是你处处小心防御,则必定处处都是弱点." 有些武术尽管先声夺人.但却如喝渗水之酒,令人越瞧越觉无味: 但 ...
- poj1184 聪明的打字员(BFS剪枝)
http://poj.org/problem?id=1184 用字符串s存下数字,并把光标位置做一个字符加到s末尾,用map做标记状态是否出现过,然后bfs即可. 不剪枝是过不了的,考虑的两种交换操作 ...
- [置顶] SSO单点登录系列6:cas单点登录防止登出退出后刷新后退ticket失效报500错
这个问题之前就发现过,最近有几个哥们一直在问我这个怎么搞,我手上在做另一个项目,cas就暂时搁浅了几周.现在我们来一起改一下你的应用(client2/3)的web.xml来解决这个2b问题,首先看下错 ...
- iOS 字体设置
使用无衬线字体 body { font-family: "Helvetica Neue", Helvetica, STHeiTi, sans-serif; } iOS 4 ...
- [生成树][Uva1395][Slim Span]
代码: #include <set> #include <queue> #include <cmath> #include <cstdio> #incl ...
- [裸KMP][HDU1711][Number Sequence]
题意 找到子串在母串出现的第一个位置 解法 裸的KMP 特别的地方 第一次不看模板自己敲的KMP #include<stdio.h> const int maxn=100000; cons ...