Java基础语法(二)_深入理解数据类型转换-corejava-4

7 篇文章 0 订阅
6 篇文章 0 订阅

目标:
1.自动转换
2.强制类型转换

一、类型转换

1.原因:Java程序中要求参与的计算的数据,必须要保证数据类型的一致性。

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

3.分类:自动类型转换和强制类型转换

二、自动类型转换

1.定义:将取值范围小的类型自动提升为取值范围大的类型 。

2. 特点:代码不需要进行特殊处理,自动完成。

3. 规则:范围小的类型向范围大的类型提升, byte、short、char 运算时直接提升为 int (与int为数值的默认值有关)。

		byte、short、char‐‐>int‐‐>long‐‐>float‐‐>double
		
注意:并非数据类型的占用内存越大,范围越大,如:long类型 占8个字节,float类型占4个字节,long数据范围没有float大。long类型要向float自动转换。

3.代码

public class DataTypeTest1 {
	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更大一些,符合从小到大的规则
		// 也发生了自动类型转换
		float num3 = 30L;
		System.out.println(num3); // 30.0
	}
}

三、强制类型转换

1.定义:将 取值范围大的类型 强制转换成 取值范围小的类型 。

2.特点:代码需要进行特殊的格式处理,不能自动完成。

3.格式:范围小的类型 范围小的变量名 = (范围小的类型) 原本范围大的数据;

4.注意:
	1.强制类型转换一般不推荐使用,因为有可能发生精度损失、数据溢出。
	
	2.byte/short/char这三种类型都可以发生数学运算,例如加法“+”.
	
	3.byte/short/char这三种类型在运算的时候,都会被首先提升成为int类型,然后再计算。
	
	4.boolean类型不能发生数据类型转换
	
5.代码:
public class DataTypeTest2 {
	public static void main(String[] args) {
		// 左边是int类型,右边是long类型,不一样
		// long --> int,不是从小到大
		// 不能发生自动类型转换!
		// 格式:范围小的类型 范围小的变量名 = (范围小的类型) 原本范围大的数据;
		int num = (int) 100L;
		System.out.println(num);
		
		// 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
		// 一旦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强制转换为short:注意必须保证逻辑上真实大小本来就没有超过short范围,否则会发生数据溢出
		short result2 = (short) (num4 + num6);
		System.out.println(result2); // 100
	}
}

	/*
	数字和字符的对照关系表(编码表):
	ASCII码表:美国信息交换标准代码。
	Unicode码表:万国码。也是数字和符号的对照关系,开头0-127部分和ASCII完全一样,但是从128开始包含有更多字符。
	48 - '0'
	65 - 'A'
	97 - 'a'
	*/
public class Demo03DataTypeChar {
	public static void main(String[] args) {
		char zifu1 = '1';
		System.out.println(zifu1 + 0); // 49
		
		char zifu2 = 'A'; // 其实底层保存的是65数字
		
		char zifu3 = 'c';
		// 左侧是int类型,右边是char类型,
		// char --> int,确实是从小到大
		// 发生了自动类型转换
		int num = zifu3;
		System.out.println(num); // 99
		
		char zifu4 = '中'; // 正确写法
		System.out.println(zifu4 + 0); // 20013
	}
}

四、规则

1.数据类型短转换为数据类型长的(比的是占用内存)

	1)原始类型为有符号类型,那么直接在原始对象的高位补该对象的符号位:
	  即:如果是负数则补1,如果是正数则0
	  
	2)原始类型为无符号类型,那么直接在原始对象高位补0
	
	3)因为:
		1.这一机制保证了,在不同类型进行转换的时候,对象转换为更长的目标对象,其数值的一致性
		2.问题会出现在有符号类型转换为更长无符号。	
			如byte->char,当byte为负数时,实际得到的数值为65535+byte
			
2.数据类型长的转换为数据类型:直接截取低位。

五、拓展知识

1.由于计算机中只有二进制,0,1,加法运算(),没有减法,用加(负数)解决,所以用补码表示数值,用补码进行运算。
2.规则:
	先将数字转换为2进制,再最左边添上符号位
	最高位为表示符号位,其他为为数值大小
	
	原码:
			正数:符号位为0
			负数:符号位为1
	反码:
			正数:等于原码
			负数:符号位不变,其他位按位求反
	补码:
			正数:等于原码
			负数:反码+1
			
	即: 
	正数转换:补码=原码
	负数转换:
		原码转化补码:符号位不变,其他位按位求反,再+1
		反码转化补码:同样符号位不变,其他位按位求反,再+1
					 或符号位不变,-1,再其他位按位求反
	
			1B=8bit
			1K=1024B
			1M=1024K
			1G=1024M
			如: byte:-128~127,byte为1个字节,8位。
				00000000~01111111 存储0~127
				注:计算机中10000000 为-128;
						   00000000 为0;
						   11111111为127
						   10000000~0存储-128~0
						   00000000~11111111 存储0~127

六、题目

1.测试b的结果,并且说明为什么会出现这样的结果?
	   byte a=(byte)129;
	   int b=a;
解析:
1)129为int,数值默认为int ,4个字节32位
2)129在int下二进制为00...0 1000 0001(补码),
3)进行强制类型转换为byte(1个字节8位),直接截取低8位1000 0001(补码),
4)补码转反码:最高位为符号位不变,其他位求反(1111 1110),再+1 (1111 1111) 得出原码,得出-127‬。

2.测试该结果,并解释为什么会出现这样的结果?
  		System.out.println((int)(char)(byte)-1);//65535
解析:
  1)-1先是int类型,二进制原码为10000....001,补码为11111...111
  2)强制类型转换为byte类型截取低位(右边)8位,得出1111 1111(补码),
  3)强制类型转换为char类型(2个字节16位),byte是有符号的,故补上8个1,得出16个1,即11111...111(补码)
  4)强制类型转换为int类型(4个字节32位),char是无符号的,故补上16个0,得出0000....0000 1111....1111(补码),16个0,16个1
  5)补码转换为原码,由于符号位为0,所以补码等于反码为65535

3.请问:Integer.MIN_VALUE-Integer.MAX_VALUE=?

System.out.println(Integer.MIN_VALUE);//-2147483648
System.out.println(Integer.MAX_VALUE);//2147483647
System.out.println(Integer.MIN_VALUE-Integer.MAX_VALUE); //1
从二进制的补码推导
1)Integer.MIN_VALUE补码表示  10000...0000,32个1
2)Integer.MAX_VALUE补码表示  01111...111,1个0,31个1
3)-Integer.MAX_VALUE补码表示 10000...001
4)相加,溢出舍去,00000...001(补码),正数补码=原码,得出1.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值