《Java解惑》系列——01表达式之谜——谜题06:多重转型
来源:互联网 发布:鸟倦飞而知还什么意思 编辑:程序博客网 时间:2024/05/19 23:25
知识点:
注意java中的类型转换引起的:类型拓宽、窄化,符号扩展等。
问题:
下面程序会打印什么??
结果是不是出乎大家的意料呢?的确如此。
产生这个结果的原因:
转型为 byte,之后转型为 char,最后转型回 int。第一个转型将数值从32 位窄
化到了8位,第二个转型将数值从8位拓宽到了 16位,最后一个转型又将数值
从16位拓宽回了32 位。这个数值最终是回到了起点吗?如果你运行该程序,你
就会发现不是。它打印出来的是65535,但是这是为什么呢?
该程序的行为紧密依赖于转型的符号扩展行为。Java 使用了基于 2 的补码的二
进制运算,因此 int类型的数值-1 的所有32 位都是置位的。从int到byte的
转型是很简单的,它执行了一个窄化原始类型转化(narrowing primitive
conversion),直接将除低8位之外的所有位全部砍掉。这样做留下的是一个8
位都被置位了的 byte,它仍旧表示-1。
从byte 到 char 的转型稍微麻烦一点,因为 byte 是一个有符号类型,而 char
是一个无符号类型。在将一个整数类型转换成另一个宽度更宽的整数类型时,通
常是可以保持其数值的,但是却不可能将一个负的 byte 数值表示成一个 char。
因此,从byte 到 char 的转换被认为不是一个拓宽原始类型的转换,而是一个拓
宽并窄化原始类型的转换(widening and narrowing primitive conversion):
byte 被转换成了 int,而这个 int又被转换成了 char。
所有这些听起来有点复杂,幸运的是,有一条很简单的规则能够描述从较窄的整
型转换成较宽的整型时的符号扩展行为:如果最初的数值类型是有符号的,那么
就执行符号扩展;如果它是char,那么不管它将要被转换成什么类型,都执行
零扩展。了解这条规则可以使我们很容易地解决这个谜题。
因为byte是一个有符号的类型,所以在将byte数值-1转换成char时,会发生
符号扩展。作为结果的char数值的16个位就都被置位了,因此它等于216-1,
即 65535。从char到int的转型也是一个拓宽原始类型转换,所以这条规则告
诉我们,它将执行零扩展而不是符号扩展。作为结果的int数值也就成了65535,
这正是程序打印出的结果。
尽管这条简单的规则描述了在有符号和无符号整型之间进行拓宽原始类型时的
符号扩展行为,你最好还是不要编写出依赖于它的程序。如果你正在执行一个转
型到 char 或从char 转型的拓宽原始类型转换,并且这个 char 是仅有的无符号
整型,那么你最好将你的意图明确地表达出来。
解决方法:
转换之前,明确你自己的意图。否则就容易得到你意想不到的结果。
如果你在将一个 char 数值 c 转型为一个宽度更宽的类型,并且你不希望有符号
扩展,那么为清晰表达意图,可以考虑使用一个位掩码,即使它并不是必需的:
int i = c & 0xffff;
或者,书写一句注释来描述转换的行为:
int i = c; //不会执行符号扩展
如果你在将一个 char数值 c 转型为一个宽度更宽的整型,并且你希望有符号扩
展,那么就先将 char 转型为一个 short,它与 char 具有同样的宽度,但是它是
有符号的。在给出了这种细微的代码之后,你应该也为它书写一句注释:
int i = (short) c; //转型将引起符号扩展
如果你在将一个byte数值b转型为一个char,并且你不希望有符号扩展,那么
你必须使用一个位掩码来限制它。这是一种通用做法,所以不需要任何注释:
char c = (char) (b & 0xff);
扩展,那么为清晰表达意图,可以考虑使用一个位掩码,即使它并不是必需的:
int i = c & 0xffff;
或者,书写一句注释来描述转换的行为:
int i = c; //不会执行符号扩展
如果你在将一个 char数值 c 转型为一个宽度更宽的整型,并且你希望有符号扩
展,那么就先将 char 转型为一个 short,它与 char 具有同样的宽度,但是它是
有符号的。在给出了这种细微的代码之后,你应该也为它书写一句注释:
int i = (short) c; //转型将引起符号扩展
如果你在将一个byte数值b转型为一个char,并且你不希望有符号扩展,那么
你必须使用一个位掩码来限制它。这是一种通用做法,所以不需要任何注释:
char c = (char) (b & 0xff);
总结:
这个教训很简单:如果你通过观察不能确定程序将要做什么,那么它做的就很有
可能不是你想要的。要为明白清晰地表达你的意图而努力。尽管有这么一条简单
的规则,描述了涉及有符号和无符号整型拓宽转换的符号扩展行为,但是大多数
程序员都不知道它。如果你的程序依赖于它,那么你就应该把你的意图表达清楚。
0 0
- 《Java解惑》系列——01表达式之谜——谜题06:多重转型
- 《Java解惑》系列——01表达式之谜——谜题01:奇偶性
- 《Java解惑》系列——01表达式之谜——谜题02:找零时刻
- 《Java解惑》系列——01表达式之谜——谜题03:长整除
- 《Java解惑》系列——01表达式之谜——谜题04:初级问题
- 《Java解惑》系列——01表达式之谜——谜题05:十六进制趣事
- 《Java解惑》系列——01表达式之谜——谜题07:互换内容
- 《Java解惑》系列——01表达式之谜——谜题08:Dos Equis
- 《Java解惑》系列——01表达式之谜——谜题09:半斤
- 《Java解惑》系列——01表达式之谜——谜题10:八两
- java解惑--谜题6:多重转型
- 《java解惑》读书笔记1——表达式之谜
- java解惑------多重转型(解惑)
- java解惑(一)—表达式之谜
- 《Java解惑》系列——03循环谜题——谜题25:无情的增量操作(表达式中变量赋值不要超过一次)
- 《Java解惑》系列——02字符谜题——谜题16:打印程序
- 《Java解惑》系列——02字符谜题——谜题17:嗯??
- 《Java解惑》系列——02字符谜题——谜题20、21的感想
- cannot bind to local 8600 for debugger
- 高效生成差异的脚本
- 【opencv】图像和轮廓的匹配(hu矩)
- struct stat结构体的详解和用法
- 数据结构_经典查找算法(1)
- 《Java解惑》系列——01表达式之谜——谜题06:多重转型
- 用Node.js做了一个iOS icon自动化生成工具
- c++ 内部模板类如何继承
- hadoop2.3.0错误之Configured Capacity: 0 (0 B)Present Capacity: 0 (0 B) DFS Remaining: 0 (0 B) DFS Used:
- SQL Server 存储过程
- 5、ARP协议的用途 及算法、在哪一层上会使用arp ?
- assert()函数用法总结
- 函数命名规则及调用约定:__stdcall,__cdecl,__fastcall,thiscall,naked call
- MySQL(基础篇)之Table操作