Java中的类型转换

数据类型的转换

数据类型的转换是赋值的数据类型和变量接收的数据类型不一致时发生的一种情况。这里我们需要考虑的是自动类型转换和强制类型转换,也可以称做隐式转换和显式转换。
Java 语言是一种强类型的语言。强类型的语言有以下几个要求:

1.变量或常量必须有类型:要求声明变量或常量时必须声明类型,而且只能在声明以后才能使用。
2.赋值时类型必须一致:值的类型必须和变量或常量的类型完全一致。
3.运算时类型必须一致:参与运算的数据类型必须一致才能运算。

自动类型转换

自动类型转换也称隐式类型转换,指从低级类型向高级类型的转换,由jvm负责自行执行转换
在这里插入图片描述
实心箭头表示无无信息丢失类型转换,虚线箭头表示可能有信息丢失类型转换。

例:

int x=50;            //声明x为int型
float y=x;           //将x赋值给float型的y

此时输出的y=50.0

强制类型转化

强制类型转化也称显示类型转换,当把高精度的变量值赋值给低精度变量,必须使用显示类型转换(强制类型转换)。
语法:(类型名)要转换的值;
例:

int a=(int)45.555;    //此时输出a的值为45

注意:以显式类型转换会导致精度损失

精度损失分类:
一:
数值大小超过小类型取值范围,位数足够。源数据数据位变为目标数据的符号位。

        int i=129;
        byte b=(byte)i;
        System.out.println(b);

输出结果:
在这里插入图片描述
这里我们需要知道在计算机系统中数值一律用二进制补码来存储
二进制最高位是符号位,0代表正数,1代表负数。
正数的值是其本身,负数是除最高位(符号位)外,其它位逐位取反,在加1。
(129)十进制 =(10000001)二进制
转化后为8位,但是比byte允许的最大数值大(-128~127),所将最高位视为符号位。所以逐位取反加1。所以输出为-127。

二:
现在我们进行另外一种尝试,当我们给i赋值为257时,输出结果又是什么呢?

        int i=257;
        byte b=(byte)i;
        System.out.println(b);

在这里插入图片描述
输出结果为1,这是为什么?我们将257转化为二进制
257=100000001
发现9位,而我们知道byte类型存储位数最多为8位,所以此时计算机将最高位抛弃成为00000001,所以输出就是1。

对象类型转换

将一个类型强制转换成另一个类型的过程被称为类型转换。这里所说的对象类型转换,是**指存在继承关系的对象,不是任意类型的对象。**当对不存在继承关系的对象进行强制类型转换时,会抛出 Java 强制类型转换(java.lang.ClassCastException)异常。
Java 语言允许某个类型的引用变量引用子类的实例,而且可以对这个引用变量进行类型转换。Java 中引用类型之间的类型转换(前提是两个类是父子关系)主要有两种,分别是向上转型(upcasting)和向下转型(downcasting)。
对于父类对象指向子类对象,这里体现了Java面向对象的三大特征中的多态。Java面向对象的三大特征

向上转型

父类引用指向子类对象为向上转型,语法格式如下:

fatherClass obj = new sonClass();

其中,fatherClass 是父类名称或接口名称,obj 是创建的对象,sonClass 是子类名称。

向上转型就是把子类对象直接赋给父类引用,不用强制转换。使用向上转型可以调用父类类型中的所有成员,不能调用子类类型中特有成员,最终运行效果看子类的具体实现。

向下转型

向上转型相反,子类对象指向父类引用为向下转型,语法格式如下:

sonClass obj = (sonClass) fatherClass;

其中,fatherClass 是父类名称,obj 是创建的对象,sonClass 是子类名称。

向下转型可以调用子类类型中所有的成员,不过需要注意的是如果父类引用对象指向的是子类对象,那么在向下转型的过程中是安全的,也就是编译是不会出错误。但是如果父类引用对象是父类本身,那么在向下转型的过程中是不安全的,编译不会出错,但是运行时会出现我们开始提到的 Java 强制类型转换异常,一般使用 instanceof 运算符来避免出此类错误。

例如,Animal 类表示动物类,该类对应的子类有 Dog 类,使用对象类型表示如下:

Animal animal = new Dog();    // 向上转型,把Dog类型转换为Animal类型
Dog dog = (Dog) animal; // 向下转型,把Animal类型转换为Dog类型

例 1
下面通过具体的示例演示对象类型的转换。例如,父类 Animal 和子类 Cat 中都定义了实例变量 name、静态变量 staticName、实例方法 eat() 和静态方法 staticEat()。此外,子类 Cat 中还定义了实例变量 str 和实例方法 eatMethod()。

父类 Animal 的代码如下:

public class Animal {
    public String name = "Animal:动物";
    public static String staticName = "Animal:可爱的动物";
    public void eat() {
        System.out.println("Animal:吃饭");
    }
    public static void staticEat() {
        System.out.println("Animal:动物在吃饭");
    }
}

子类 Cat 的代码如下:

public class Cat extends Animal {
    public String name = "Cat:猫";
    public String str = "Cat:可爱的小猫";
    public static String staticName = "Dog:我是喵星人";
    public void eat() {
        System.out.println("Cat:吃饭");
    }
    public static void staticEat() {
        System.out.println("Cat:猫在吃饭");
    }
    public void eatMethod() {
        System.out.println("Cat:猫喜欢吃鱼");
    }
    public static void main(String[] args) {
        Animal animal = new Cat();
        Cat cat = (Cat) animal; // 向下转型
        System.out.println(animal.name); // 输出Animal类的name变量
        System.out.println(animal.staticName); // 输出Animal类的staticName变量
        animal.eat(); // 输出Cat类的eat()方法
        animal.staticEat(); // 输出Animal类的staticEat()方法
        System.out.println(cat.str); // 调用Cat类的str变量
        cat.eatMethod(); // 调用Cat类的eatMethod()方法
    }
}

通过引用类型变量来访问所引用对象的属性和方法时,Java 虚拟机将采用以下绑定规则:
实例方法与引用变量实际引用的对象的方法进行绑定,这种绑定属于动态绑定,因为是在运行时由 Java 虚拟机动态决定的。例如,animal.eat() 是将 eat() 方法与 Cat 类绑定。
静态方法与引用变量所声明的类型的方法绑定,这种绑定属于静态绑定,因为是在编译阶段已经做了绑定。例如,animal.staticEat() 是将 staticEat() 方法与 Animal 类进行绑定。
成员变量(包括静态变量和实例变量)与引用变量所声明的类型的成员变量绑定,这种绑定属于静态绑定,因为在编译阶段已经做了绑定。例如,animal.name 和 animal.staticName 都是与 Animal 类进行绑定。

对于 Cat 类,运行时将会输出如下结果:

Animal:动物
Animal:可爱的动物
Cat:吃饭
Animal:动物在吃饭
Cat:可爱的小猫
Cat:猫喜欢吃鱼

强制对象类型转换

Java 编译器允许在具有直接或间接继承关系的类之间进行类型转换。对于向下转型,必须进行强制类型转换;对于向上转型,不必使用强制类型转换。

例如,对于一个引用类型的变量,Java 编译器按照它声明的类型来处理。如果使用 animal 调用 str 和 eatMethod() 方法将会出错,如下:

animal.str = "";    // 编译出错,提示Animal类中没有str属性
animal.eatMethod();    // 编译出错,提示Animal类中没有eatMethod()方法

如果要访问 Cat 类的成员,必须通过强制类型转换,如下:

((Cat)animal).str = "";    // 编译成功
((Cat)animal).eatMethod();    // 编译成功

把 Animal 对象类型强制转换为 Cat 对象类型,这时上面两句编译成功。对于如下语句,由于使用了强制类型转换,所以也会编译成功,例如:
Cat cat = (Cat)animal; // 编译成功,将Animal对象类型强制转换为Cat对象类型
类型强制转换时想运行成功就必须保证父类引用指向的对象一定是该子类对象,最好使用 instanceof 运算符判断后,再强转,例如:
Animal animal = new Cat();
if (animal instanceof Cat) {
Cat cat = (Cat) animal; // 向下转型

}

子类的对象可以转换成父类类型,而父类的对象实际上无法转换为子类类型。因为通俗地讲,父类拥有的成员子类肯定也有,而子类拥有的成员,父类不一定有。因此,对于向上转型,不必使用强制类型转换。例如:
Cat cat = new Cat();
Animal animal = cat; // 向上转型,不必使用强制类型转换

如果两种类型之间没有继承关系,那么将不允许进行类型转换。例如:
纯文本复制
Dog dog = new Dog();
Cat cat = (Cat)dog; // 编译出错,不允许把Dog对象类型转换为Cat对象类型

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值