Java变量、数据类型与运算符
提示:此篇Java变量、数据类型与运算符内容并不完整,仅记录了自己需注意的点。
目录
1.字节
- 在 Java 中, 一个 int 变量占 4 个字节,和操作系统没有直接关系。
- 字节是计算机中表示空间大小的基本单位。计算机使用二进制表示数据。8 个二进制位(bit) 为一个字节(Byte)。
- 例如:我们平时的计算机为 8GB 内存, 意思是 8G 个字节,其中 1KB = 1024 Byte, 1MB = 1024 KB, 1GB = 1024 MB,所以 8GB 相当于 80 多亿个字节。
- int变量 4 个字节表示的数据范围是 -2^31 -> 2^31-1 , 也就大概是 -21亿 到 +21亿.
2.字符
- 计算机中的字符本质上是一个整数, 在 C 语言中使用 ASCII 表示字符, 而 Java 中使用 Unicode 表示字符。 因此一个字符占用两个字节,表示的字符种类更多, 包括中文。
3.boolean类型
- boolean 类型有些 JVM 的实现是占 1 个字节, 有些是占 1 个比特位, 这个没有明确规定。
4.常量
常量指的是运行时类型不能发生改变。常量主要有以下两种体现形式:
①字面值
10 // int 字面值常量(十进制) 010 // int 字面值常量(八进制) 由数字 0 开头. 010 也就是十进制的 8
0x10 // int 字面值常量(十六进制) 由数字 0x 开头. 0x10 也就是十进制的 16 10L // long 字面值常量.
也可以写作 10l (小写的L)
1.0 // double 字面值常量. 也可以写作 1.0d 或者 1.0D
1.5e2 // double 字面值常量. 科学计数法表示. 相当于 1.5 * 10^2
1.0f // float 字面值常量, 也可以写作 1.0F true // boolen 字面值常量, 同样的还有 false ‘a’ // char 字面值常量, 单引号中只能有一个字符 “abc” // String 字面值常量, 双引号中可以有多个字符.
②final 关键字修饰的常量
final int a = 10; a = 20; // 编译出错. 提示 无法为最终变量a分配值
常量不能在程序运行过程中发生修改.
5.类型转换
-
Java 作为一个强类型编程语言, 当不同类型之间的变量相互赋值的时候, 会有较严格的校验. 因此像 && 之类的运算操作数必须是boolean。
-
int 和 boolean 是毫不相干的两种类型, 不能相互赋值。
-
int字面值常量给 byte 赋值
byte a = 100; // 编译通过 byte b = 256; // 编译报错, 提示 从int转换到byte可能会有损失 //注意: byte 表示的数据范围是 -128 -> +127, 256 已经超过范围, 而 100 还在范 围之内.
结论:使用字面值常量赋值的时候, Java 会自动进行一些检查校验, 判定赋值是否合理.
-
类型转换小结:
1.不同数字类型的变量之间赋值, 表示范围更小的类型能隐式转换成范围较大的 类型.
2.如果需要把范围大的类型赋值给范围小的, 需要强制类型转换, 但是可能精度丢失.
3.将一个字面值常量进行赋值的时候, Java 会自动针对数字范围进行检查.
4.强制类型转换不是一定能成功, 互不相干的类型之间无法强转.(例如:int和boolean不可强转)
6.数值提升
-
byte 和 byte 的运算
byte a = 10; byte b = 20; byte c = a + b; System.out.println(c); // 编译报错 Test.java:5: 错误: 不兼容的类型: 从int转换到byte可能会有损失 byte 和 byte 都是相同类型, 但是出现编译报错. 原因是, 虽然 a 和 b 都是 byte, 但是计算 a + b 会先将 a 和 b 都提升成 int, 再进行计算, 得到的结果也是 int, 这时赋给 c, 就会出现上述错误. //正确的写法: byte a = 10; byte b = 20; byte c = (byte)(a + b); System.out.println(c);
-
类型提升小结:
1.不同类型的数据混合运算, 范围小的会提升成范围大的.
2.对于 short, byte 这种比 4 个字节小的类型, 会先提升成 4 个字节的 int , 再运算.
7.int 和 String 之间的相互转换
-
int 转成 String
int num = 10; // 方法1 String str1 = num + ""; // 方法2 String str2 = String.valueOf(num);
-
String 转成 int
String str = "100"; int num = Integer.parseInt(str);
8.自增/自减运算符(++、- -)
int a = 10;
int b = ++a;
System.out.println(b);
int c = a++;
System.out.println(c);
- 如果不取自增运算的表达式的返回值, 则前置自增和后置自增没有区别.
- 如果取表达式的返回值, 则前置自增的返回值是自增之后的值, 后置自增的返回值是自增之前的值.
9.短路求值
-
&& 和 || 遵守短路求值的规则
System.out.println(10 > 20 && 10 / 0 == 0); // 打印 false System.out.println(10 < 20 || 10 / 0 == 0); // 打印 true //计算 10 / 0 会导致程序抛出异常. 但是上面的代码却能正常运行, 说明 10 / 0 并没有真正被求值.
结论:
1.对于 && , 如果左侧表达式值为 false, 则表达式的整体的值一定是 false, 无需计算右侧表达式.
2.对于 ||, 如果左侧表达式值为 true, 则表达式的整体的值一定是 true, 无需计算右侧表达式.
-
& 和 | 不支持短路求值
System.out.println(10 > 20 & 10 / 0 == 0); // 程序抛出异常 System.out.println(10 < 20 | 10 / 0 == 0); // 程序抛出异常
& 和 | 如果操作数为 boolean 的时候, 也表示逻辑运算. 但是和 && 以及 || 相比, 它们不支持短路求值。
10.位运算符(& 、|、 ~、 ^)
-
当 & 和 | 的操作数为整数(int, short, long, byte) 的时候, 表示按位运算, 当操作数为 boolean的时候, 表示逻辑运算.
-
按位取反:
int a = 0xf; System.out.printf("%x\n", ~a) /** 1. 0x 前缀的数字为 十六进制 数字. 十六进制可以看成是二进制的简化表示方式. 一个十六进制数字对应 4 个二进制位. 2. 0xf 表示 10 进制的 15, 也就是二进制的 1111 3. printf 能够格式化输出内容, %x 表示按照十六进制输出. 4. \n 表示换行符 */
11.移位运算(<<、 >>、 >>>)
移位运算符有三个:<< >> >>>,都是按照二进制位来运算.
-
左移 <<: 最左侧位不要了, 最右侧补 0.
int a = 0x10; System.out.printf("%x\n", a << 1); // 运行结果(注意, 是按十六进制打印的) 20
-
右移 >>: 最右侧位不要了, 最左侧补符号位(正数补0, 负数补1)
int a = 0x10; System.out.printf("%x\n", a >> 1); // 运行结果(注意, 是按十六进制打印的) 8 int b = 0xffff0000; System.out.printf("%x\n", b >> 1); // 运行结果(注意, 是按十六进制打印的) ffff8000
-
无符号右移 >>>: 最右侧位不要了, 最左侧补 0.
int a = 0xffffffff; System.out.printf("%x\n", a >>> 1); // 运行结果(注意, 是按十六进制打印的) 7fffffff
-
注意:
1.左移 1 位, 相当于原数字 * 2. 左移 N 位, 相当于原数字 * 2 的N次方.
2. 右移 1 位, 相当于原数字 / 2. 右移 N 位, 相当于原数字 / 2 的N次方.
3. 由于计算机计算移位效率高于计算乘除, 当某个代码正好乘除 2 的N次方的时候可以用移位运算代替.
4. 移动负数位或者移位位数过大都没有意义.