转自:http://blog.163.com/pilgrim_yang/blog/static/55631481201111542151582/

Java对byte 的 + - * / >> >>> << & | ^ (加,减,乘,除,右移,左移,无符号右移,位与,位或,位异或)操作,均会是首先将byte转化为int, 再行运算。这一事实可能导致多种问题:

 
 
假设我们想进行如下byte运算: 1111 1000 右移1位,再与0000 0001 或运算,得 0111 1101。
直觉写程序如下:
byte b = 0xf8;
byte b2 =  b >> 1 | 0x01;  
 
这个写法里有多重错误,现逐个纠正:
1 编译器报错,int无法直接自动转化为byte 
为解决此问题,加强制转化。
byte b = (byte)0xf8;
byte b2 = (byte)( (b >> 1) | 0x01);  
 
 
2 输出为 1111 1101 不是我们想要的 0011 1101
原因是>> 是有符号右移,当符号位为1时,左侧补的是1而非0。
修改为使用 >>> 无符号右移:
byte b = (byte)0xf8;
byte b2 = (byte)( (b >>> 1) | 0x01);  
 
3 运行后发现输出依然为 1111 1101 
原因是byte在运算前先转化为int再行位运算,因此分解后的运算步骤如下:
b 转化为int  1111 1000 转化为      11111111 11111111 11111111 11111000
无符号右移1位                      01111111 11111111 11111111 11111100
与 0x01 按位或                  01111111 11111111 11111111 11111101
强制转化回byte     11111101
解决方案,在右移运算前先 位与 0xff
byte b = (byte)0xf8;
byte b2 = (byte)( ((b & 0xff )>>> 1) | 0x01);  //注意必须加括号,因为 >>> 的优先级高于 &
 
 
4 运行后发现输出为我们想要的结果  0111 1101。运算步骤分解如下:
b 转化为int  1111 1000 转化为   11111111 11111111 11111111 11111000
和0xff 进行 & 操作              00000000 00000000 00000000 11111000
无符号右移1位   00000000 00000000 00000000 01111100
与 0x01 按位或  00000000 00000000 00000000 01111101
强制转化回byte  01111101
 
 
 
5 关于System.out.println();
byte b = (byte)0xf8;
System.out.println(b);   --最终输出为-8
运算步骤为:
b 转化为int  1111 1000 转化为   11111111 11111111 11111111 11111000
取符号位 -                      -1111111 11111111 11111111 11111000
-1取返(因为是按逆补码运算)      -0000000 00000000 00000000 00001000
输出  -8
 
 
 
 
 
最终结论:
1 区分使用 >> 和 >>>
2 在 >> 操作前要首先 & 0xff
3 注意符号优先级,正确使用括号。
4 需要强烈注意的一点是 & 的优先级小于 + . 因此 a = b & 0xff + 2000 的结果 可能不是你想要的
 
 
 
 
 
 
附:
打印byte,int 每个bit值的函数。
 
public static void printByte(byte b){
    for(int i = 7; i >=0 ; i --){
        int shiftleft = (b >> i) & 0x01;
        System.out.print(shiftleft);
    }
    System.out.println();
}
 
public static void printInt(int b){
    for(int i = 31; i >=0 ; i --){
        int shiftleft = (b >> i) & 0x01;
        System.out.print(shiftleft);
    }
    System.out.println();
}

最新文章

  1. 深入浅出JavaScript之this
  2. TCP流量控制与拥塞控制
  3. iOS_XCode7_Launch Image 的初使用
  4. windows 录音程序(二)
  5. Do带你解析:原生APP与web APP的区别
  6. wamp虚拟主机的配置 .
  7. 【leetcode】Find Peak Element ☆
  8. notepad++中的python缩进问题
  9. 用IO流发送Http请求
  10. 网站引导页flash动画跳转js脚本
  11. CP343-1 扩展ProfibusCPU 314C-2DP
  12. orchestrator HTTP接口forget-cluster误下线集群问题
  13. ACM学习&lt;二&gt;
  14. uoj35 后缀排序
  15. 使用方法拦截机制在不修改原逻辑基础上为 spring MVC 工程添加 Redis 缓存
  16. c/c++指针详解(一)
  17. WPF INotifyPropertyChanged 通过特性减少代码量
  18. MySQL在创建数据表的时候创建索引
  19. [作业] Python入门基础---九九乘法表
  20. NIO与Socket

热门文章

  1. 小草的Trouble学生信息管理系统
  2. 2018中国大学生程序设计竞赛 - 网络选拔赛 1010 YJJ&#39;s Salesman 【离散化+树状数组维护区间最大值】
  3. Hibernate注解开发示例
  4. &lt;jsp:include&gt;和&lt;%@include file=&quot;&quot;%&gt;的区别(简单了解)
  5. c语言描述的链队列的基本操作
  6. 数据库——MySQL——单表查询
  7. Python基础—10-常用模块:time,calendar,datetime
  8. JetBrains 授权服务器(License Server):
  9. 【CodeForces 915 C】Permute Digits(思维+模拟)
  10. python核心编程2 第十五章 练习