数据类型转换

当数据类型不一样时,将会发生数据类型转换。

自动类型转换(隐式)
	1. 特点:代码不需要进行特殊处理,自动完成。
	2. 规则:数据范围从小到大。
public class Demo {
	public static void main(String[] args){
		System.out.println(1024); // 这是一个整数,默认就是int类型
		System.out.println(3.14); // 就是一个浮点数,默认就是double类型
		
		// 左边是long类型,右边是默认int类型,左右不一样
		// 一个等号代表赋值,将右侧int常量,交给左侧的long变量进行存储
		// int --> long,符合了数据范围从小到大的要求
		// 这一行代码发生了自动类型转换
		long num1 = 100;
		System.out.println(num1); // 100
		
		// 左边是double类型,右边是float类型。左右不一样
		// float --> double 符合从小到大的规则
		// 也发生了自动类型的转换
		double num2 = 2.5F;
		System.out.println(num2); // 2.5
		
		// 左边是float类型,右边是long类型。左右不一样
		// long--> float 符合从小到大的规则
		// 也发生了自动类型的转换
		float num3 = 30L;
		System.out.println(num3); // 30.0
	}
}
强制类型转换(显式)
	1. 特点:代码需要进行特殊格式处理,不能自动完成。
	2. 格式:范围小的类型, 范围小的变量名 = (范围小的类型)原本范围大的数据

注意事项

  1. 强制类型转换一般不推荐使用,因为有可能发生精度损失、数据溢出。
  2. byte/short/char这三种类型都可以发生数学运算,例如“+”
  3. byte/short/char这三种类型在运算的时候,会被首先提升为int类型,再计算
  4. boolean类型不能发生数据类型转换
public class Demo {
	public static void main(String[] args) {
		// 左边int类型,右边long类型,不一样
		// long --> int 不是从小到大
		int num = (int) 100L;
		System.out.println(num); // 100

		// long 强制转换为int类型
		int num2 = (int) 6000000000L;
		System.out.println(num2); // 1705032704 // 数据溢出
		
		// double --> int  强制类型转换
		int num3 = (int) 3.99;
		// 这并不是四舍五入,所有的小数都会被舍掉
		System.out.println(num3); // 3   // 精度损失

		char zifu1 = 'A'; //这是一个字符型变量,里面是大写字母A
		System.out.println(zifu1 + 1); // 66,大写字母A被当做65处理
		// 计算机的底层会用一个数字(二进制)来代表字符A,就是65 (ASCII编码表)
		// 一旦char类型进行了数学运算,那么字符就会按照一定的规则翻译成为一个数字
		
		byte num4 = 40; // 注意!右侧的数值大小不能超过左侧类型的范围
		byte num5 = 50;
		// byte + byte --> int + int = int
		int result1 = num4 + num5;
		System.out.println(result1); // 90
		
		short num6 = 60;
		// byte + short  -->  int + int  = int
		int result2 = num4 + num6;
		System.out.println(result2); // 100
		// int强制转换为short;注意:必须保证逻辑上真实大小本来就没有超过short范围,否则会发生数据溢出
		short result3 = (short) (num4 + num6);
		System.out.println(result3); // 100
		
		
	}
}

java 隐形转换和显示转换 java数据类型隐式转换_数据


java 隐形转换和显示转换 java数据类型隐式转换_数据_02

// 数字和字符的对照关系表(编码表):
// ASCII码表:American Standard Code for Information Interchange,美国信息交换标准代码
// Unicode码表:万国表。也是数字和符号的对照关系,开头0-127部分和ASCII完全一样,但是从128开始包含有更多字符
// 48 -> '0'
// 65 -> 'A'
// 97 -> 'a'
对于byte、short、char三种类型来说,如果右侧赋值的数值没有超过范围,
那么javac编译器将会自动隐含的为我们补上(byte)(short)(char)
	1. 如果没有超过左侧的范围,编译器补上强转
	2. 如果右侧超过了左侧的范围,那么直接编译器报错
public class Demo {
	public static void main(String[] args) {
		// 右侧确实是一个int数字,但是没有超过左侧的范围,就是正确的
		// int --> byte  不是自动类型转换
		byte num1 = /*(byte)*/ 30; // 右侧没有超过左侧的范围
		System.out.println(num1); // 30
		
		// byte num2 = 128; // 报错 右侧超过了左侧的范围 

		// int --> char,没有超过范围
		// 编译器将会自动补上一个隐含的(char)
		char zifu = /*(char)*/ 65;
		System.out.println(zifu); // A
	}
}
在给变量进行赋值的时候,如果右侧的表达式当中全都是常量,没有任何变量,
那么在编译器javac将会直接将若干个常量表达式计算得到结果。
short result = 5 + 8; // 等号右边全都是常量,没有任何变量参与运算
编译之后,得到的。class字节码文件当中相当于【直接就是】:
short result = 13;
右侧的常量结果数值,没有超过左侧范围,所以正确。

这称为“编译器的常量优化”

但是注意:一旦表达式当中又变量参与,就不能进行这种优化了
public class Demo {
	public static void main(String[] args) {
		short num1 = 10; // 正确写法,右侧没有超过左侧的范围

		short a = 5;
		short b = 8;
		// short + short --> int + int --> int
		// short result = a + b; // 错误写法!左侧需要的是int类型
		
		// 右边不用变量,而是采用常量,而且只有两个常量,没有别人
		short result = 5 + 8;
		System.out.println(result); // 13
	}
}