java自动转换与强制转换

前言

本文转自 http://blog.csdn.net/Mailbomb/article/details/2449261
文章中添加了一些在阅读时的感悟(阅读笔记)

正文

数据类型转换

Java语言是一种强类型的语言。强类型的语言有以下几个要求:

  • 变量或常量必须有类型

要求声明变量或常量时必须声明类型,而且只能在声明以后才能使用。

  • 赋值时类型必须一致

值的类型必须和变量或常量的类型完全一致。

  • 运算时类型必须一致

    参与运算的数据类型必须一致才能运算。

    但是在实际的使用中,经常需要在不同类型的值之间进行操作,这就需要一种新的语法来适应这种需要,这个语法就是数据类型转换。
    在数值处理这部分,计算机和现实的逻辑不太一样,对于现实来说,1和1.0没有什么区别,但是对于计算机来说,1是整数类型,而1.0是小数类型,其在内存中的存储方式以及占用的空间都不一样,所以类型转换在计算机内部是必须的。Java语言中的数据类型转换有两种:

  • 自动类型转换

编译器自动完成类型转换,不需要在程序中编写代码。

  • 强制类型转换

由于基本数据类型中boolean类型不是数字型,所以基本数据类型的转换是出了boolean类型以外的其它7种类型之间的转换。下面来具体介绍两种类型转换的规则、适用场合以及使用时需要注意的问题。

自动类型转换

自动类型转换,也称隐式类型转换,是指不需要书写代码,由系统自动完成的类型转换。由于实际开发中这样的类型转换很多,所以Java语言在设计时,没有为该操作设计语法,而是由JVM自动完成。

  • 转换规则

从存储范围小的类型到存储范围大的类型。 具体规则为:
byte→short(char)→int→long→float→double 也就是说byte类型的变量可以自动转换为short类型,示例代码:
byte b = 10;
short sh = b; 这里在赋值时,JVM首先将b的值转换为short类型,然后再赋值给sh。 在类型转换时可以跳跃。示例代码:
byte b1 = 100;
int n = b1;

  • 注意问题

在整数之间进行类型转换时,数值不发生改变,而将整数类型,特别是比较大的整数类型转换成小数类型时,由于存储方式不同,有可能存在数据精度的损失。

阅读笔记

在运算的过程中,将运算表达式的结果往范围大的类型转换,比如float ii = 333 + 3.4f + 3.4;
这里333是int型,3.4f是float类型,3.4是double类型,所以值从int - float - double
最后得到的结果也时double,所以这里double给予float引用需要强转

强制类型转换

强制类型转换,也称显式类型转换,是指必须书写代码才能完成的类型转换。该类类型转换很可能存在精度的损失,所以必须书写相应的代码,并且能够忍受该种损失时才进行该类型的转换。

  • 转换规则

从存储范围大的类型到存储范围小的类型。 具体规则为:
double→float→long→int→short(char)→byte 语法格式为:
(转换到的类型)需要转换的值 示例代码:
double d = 3.10;
int n = (int)d; 这里将double类型的变量d强制转换成int类型,然后赋值给变量n。需要说明的是小数强制转换为整数,采用的是“去1法”,也就是无条件的舍弃小数点的所有数字,则以上转换出的结果是3。整数强制转换为整数时取数字的低位,例如int类型的变量转换为byte类型时,则只去int类型的低8位(也就是最后一个字节)的值。
示例代码:
int n = 123;
byte b = (byte)n;
int m = 1234;
byte b1 = (byte)m; 则b的值还是123,而b1的值为-46。b1的计算方法如下:m的值转换为二进制是10011010010,取该数字低8位的值作为b1的值,则b1的二进制值是11010010,按照机器数的规定,最高位是符号位,1代表负数,在计算机中负数存储的是补码,则该负数的原码是10101110,该值就是十进制的-46。

  • 注意问题
    强制类型转换通常都会存储精度的损失,所以使用时需要谨慎。

阅读笔记

1、float f=3.4;是否正确?
答:不正确。3.4是双精度数,将双精度型(double)赋值给浮点型(float)属于下转型(down-casting,也称为窄化)会造成精度损失,因此需要强制类型转换float f =(float)3.4; 或者写成float f =3.4F .

注:由上面想到: byte、short 、 int 、 long 在初始化的时候给予整数,如long l = 3; byte b = 3; 虽然3默认是int类型的,但是这样是正确的。 这里需要在双精度数字后面加字母的只有float这一例了

2、short s1 = 1; s1 = s1 +1;有错吗?short s1 = 1; s1 += 1;有错吗?
答:short s1 = 1; s1 = s1 + 1是错的,s1 + 1在运算过程中1是int类型,s1是short类型,向高的转,结果为int类型,所以这里需要强转,而 s1 += 1是正确的,因为jvm自动帮我们处理好了

注:short s1 = 1 + 2 + 3;是正确的。当然除了short 比short低的byte也有同样的问题

其它

后续的复合数据类型,如类和接口等,也存在类似的转换。

阅读笔记

引用类型的强制转换与基本类型的强制转换不同,引用类型如果不想报ClassCastException的话,请务必遵循下面的步骤

public static void main(String[] args) {
    Parent pa = new ChildClazz();
    ChildClazz cc = (ChildClazz) pa;
}

解释:ChildClazz是Parent的子类,且需要强转的Parent引用必须指向ChildClazz创建的内存空间,这样执行(ChildClazz) pa的时候才不会报错。两个毫无关系的类,在强转的时候肯定是报ClassCaseException异常的

展开阅读全文

没有更多推荐了,返回首页