Java中,经常可以遇到类型转换的场景,从变量的定义到复制、数值变量的计算到方法的参数传递、基类与派生类间的造型等,随处可见类型转换的身影。Java中的类型转换在Java编码中具有重要的作用。
       首先,来了解下数据类型的基本理解:数据是用来描述数据的种类,包括其值和基于其值基础上的可进行的操作集合。

Java中数据类型主要分为两大类:基本数据类型和引用数据类型。
       基本数据类型共有8种,分别是:布尔型boolean, 字符型char和数值型byte/short/int/long/float/double。

由于字符型char所表示的单个字符与Ascii码中相应整形对应,因此,有时也将其划分到数值型中 。

引用类型具体可分为:数组、类和接口。

因此,本文中Java类型转换的总结也将分为基本数据类型和引用数据类型两个方面展开。

一、基本数据类型的类型转换【引自http://www.cnblogs.com/lwbqqyumidi/p/3700164.htm】

基本数据类型中,布尔类型boolean占有一个字节,由于其本身所代码的特殊含义,boolean类型与其他基本类型不能进行类型的转换(既不能进行自动类型的提升,也不能强制类型转换), 否则,将编译出错

1.基本数据类型中数值类型的自动类型提升

数值类型在内存中直接存储其本身的值,对于不同的数值类型,内存中会分配相应的大小去存储。如:byte类型的变量占用8位,int类型变量占用32位等。相应的,不同的数值类型会有与其存储空间相匹配的取值范围。具体如下所示:

图中依次表示了各数值类型的字节数和相应的取值范围。在Java中,整数类型(byte/short/int/long)中,对于未声明数据类型的整形,其默认类型为int型。在浮点类型(float/double)中,对于未声明数据类型的浮点型,默认为double型。

自动转换有以下规律:

小的类型自动转化为大的类型

整数类型可以自动转化为浮点类型,可能会产生舍入误差

按从低级到高级顺序:

char

Byte→short→int→long---›float→double

接下来我们看看如下一个较为经典例子:

 1 package com.corn.testcast;
2 public class TestCast {
3 public static void main(String[] args) {
4 byte a = 1000; // 编译出错 Type mismatch: cannot convert from int to byte
5 float b = 1.5; // 编译出错 Type mismatch: cannot convert from double to float
6 byte c = 3; // 编译正确
7 }
8}

是不是有点奇怪?按照上面的思路去理解,将一个int型的1000赋给一个byte型的变量a,编译出错,提示"cannot convert from int to byte"是对的,1.5默认是一个double型,将一个double类型的值赋给一个float类型,编译出错,这也是对的。但是最后一句:将一个int型的3赋给一个byte型的变量c,居然编译正确,这是为什么呢?

原因在于:jvm在编译过程中,对于默认为int类型的数值时,当赋给一个比int型数值范围小的数值类型变量(在此统一称为数值类型k,k可以是byte/char/short类型),会进行判断,如果此int型数值超过数值类型k,那么会直接编译出错。因为你将一个超过了范围的数值赋给类型为k的变量,k装不下嘛,你有没有进行强制类型转换,当然报错了。但是如果此int型数值尚在数值类型k范围内,jvm会自定进行一次隐式类型转换,将此int型数值转换成类型k。如图中的虚线箭头。

这一点有点特别,需要稍微注意下。

        在其他情况下,当将一个数值范围小的类型赋给一个数值范围大的数值型变量,jvm在编译过程中俊将此数值的类型进行了自动提升。在数值类型的自动类型提升过程中,数值精度至少不应该降低(整型保持不变,float->double精度将变高)。

 1 package com.corn.testcast;
2 public class TestCast {
3 public static void main(String[] args) {
4 long a = 10000000000; //编译出错: The literal 10000000000 of type int is out of range
5 long b = 10000000000L; //编译正确
6 int c = 1000;
7 long d = c;
8 float e = 1.5F;
9 double f = e;
10 }
11 }

如上:定义long类型的a变量时,将编译出错,原因在于10000000000默认是int类型,同时int类型的数值范围是-2^31 ~ 2^31-1,因此,10000000000已经超过此范围内的最大值,故而其自身已经编译出错,更谈不上赋值给long型变量a了。

此时,若想正确赋值,改变10000000000自身默认的类型即可,直接改成10000000000L即可将其自身类型定义为long型。此时再赋值编译正确。

将值为1000的int型变量c赋值给long型变量d,按照上文所述,此时直接发生了自动类型提升, 编译正确。同理,将e赋给f编译正确。

接下来,还有一个地方需要注意的是:char型其本身是unsigned型,同时具有两个字节,其数值范围是0 ~ 2^16-1,因为,这直接导致byte型不能自动类型提升到char,char和short直接也不会发生自动类型提升(因为负数的问题),同时,byte当然可以直接提升到short型。

2.基本数据类型中的数值类型强制转换

当我们需要将数值范围较大的数值类型赋给数值范围较小的数值类型变量时,由于此时可能会丢失精度(1讲到的从int到k型的隐式转换除外),因此,需要人为进行转换。我们称之为强制类型转换。首先我们看一下如下的例子:

 1 package com.corn.testcast;
2 public class TestCast {
4 public static void main(String[] args) {
5 byte p = 3; // 编译正确:int到byte编译过程中发生隐式类型转换
6 int a = 3;
7 byte b = a; // 编译出错:cannot convert from int to byte
8 byte c = (byte) a; // 编译正确
9 float d = (float) 4.0;
10 }
11 }

byte p =3;编译正确在1中已经进行了解释。接下来将一个值为3的int型变量a赋值给byte型变量b,发生编译错误。这两种写法之间有什么区别呢?

区别在于前者3是直接量,编译期间可以直接进行判定,后者a为一变量,需要到运行期间才能确定,也就是说,编译期间为以防万一,当然不可能编译通过了。此时,需要进行强制类型转换。

强制类型转换所带来的结果是可能会丢失精度,如果此数值尚在范围较小的类型数值范围内,对于整型变量精度不变,但如果超出范围较小的类型数值范围内,很可能出现一些意外情况。

如下经典例子:

 1 package com.corn.testcast;
2 public class TestCast {
3 public static void main(String[] args) {
4 int a = 233;
5 byte b = (byte) a;
6 System.out.println("b:" + b); // 输出:-23
7 }
8 }

为什么结果是-23?需要从最根本的二进制存储考虑。233的二进制表示为:24位0 + 11101001,byte型只有8位,于是从高位开始舍弃,截断后剩下:11101001,由于二进制最高位1表示负数,0表示正数,其相应的负数为-23。

3.进行数学运算时的数据类型自动提升与可能需要的强制类型转换

如下代码:

 1 package com.corn.testcast;
2 public class TestCast {
3 public static void main(String[] args) {
4 byte a = 3 + 5; // 编译正常 编译成 3+5直接变为8
5 int b = 3, c = 5;
6 byte d = b + c; // 编译错误:cannot convert from int to byte
7 byte e = 10, f = 11;
8 byte g = e + f; // 编译错误 +直接将10和11类型提升为了int
9 byte h = (byte) (e + f); //编译正确
10 }
11 }

当进行数学运算时,数据类型会自动发生提升到运算符左右之较大者,以此类推。当将最后的运算结果赋值给指定的数值类型时,可能需要进行强制类型转换。

二、引用数据类型的类型转换

在Java中由于继承和向上转型,子类可以非常自然地转换成父类,但是父类转换成子类则需要强制转换。因为子类拥有比父类更多的属性、更强的功能,所以父类转换为子类需要强制。那么,是不是只要是父类转换为子类就会成功呢?

其实不然,他们之间的强制类型转换是有条件的。

当我们用一个类型的构造器构造出一个对象时,这个对象的类型就已经确定的,也就说它的本质是不会再发生变化了。在Java中我们可以通过继承、向上转型的关系使用父类类型来引用它,这个时候我们是使用功能较弱的类型引用功能较强的对象,这是可行的。但是将功能较弱的类型强制转功能较强的对象时,就不一定可以行了。

举个例子来说明。比如系统中存在Father、Son两个对象。首先我们先构造一个Son对象,然后用一个Father类型变量引用它:

Father father = new Son();在这里Son 对象实例被向上转型为father了,但是请注意这个Son对象实例在内存中的本质还是Son类型的,只不过它的能力临时被消弱了而已,如果我们想变强怎么办?将其对象类型还原!

Son son = (Son)father;这条语句是可行的,其实father引用仍然是Father类型的,只不过是将它的能力加强了,将其加强后转交给son引用了,Son对象实例在son的变量的引用下,恢复真身,可以使用全部功能了。

前面提到父类强制转换成子类并不是总是成功,那么在什么情况下它会失效呢?

当引用类型的真实身份是父类本身的类型时,强制类型转换就会产生错误。例如:

Father father = new Father();

Son son = (Son) father;

这个系统会抛出ClassCastException异常信息。

    所以编译器在编译时只会检查类型之间是否存在继承关系,有则通过;而在运行时就会检查它的真实类型,是则通过,否则抛出ClassCastException异常。所以在继承中,子类可以自动转型为父类,但是父类强制转换为子类时只有当引用类型真正的身份为子类时才会强制转换成功,否则失败。

综上:简单总结:(引自https://blog.csdn.net/qq_20289209/article/details/52796655)

1.向上造型:即父类引用指向子类对象,可以自动转换。如:

Father father = new Son();

这里的引用father指向内存中对象实质还是Son类型,不过对象的功能被临时削弱为father。

2.向下转型:即强制类型转换。

被向上造型的对象进行强制类型转换是没有问题的:

Son son =(Son)father;

但是强转没有向上造型的对象会出现错误的:

Father father = new Father();

Son son = (Son)father;

以下将进行代码测试:

  1. public class Father {
  2.  
    public void test(){
  3.  
    System.out.println("Father的test方法。。。。");
  4.  
    }
  5.  
     
  6.  
    public void test1(){
  7.  
    System.out.println("Father的test1方法。。。。");
  8.  
    }
  9.  
     
  10.  
    public static void main(String[] args) {
  11.  
    //测试向上造型
  12.  
    Father father = new Son();
  13.  
    father.test(); //Father的test方法。。。。
  14.  
    father.test1(); //Son的test1方法。。。。
  15.  
    //father.test2(); //编译出错
  16.  
     
  17.  
    //测试强制类型转化
  18.  
    Son son = (Son)father;
  19.  
    son.test(); //Father的test方法。。。。
  20.  
    son.test2(); //Son的test2方法。。。。
  21.  
     
  22.  
    //测试错误的强转
  23.  
    Father father2 = new Father();
  24.  
    son = (Son)father2;
  25.  
    /*
  26.  
    * 上面这条编译不报错,但是运行异常,下面是错误报告
  27.  
    * Exception in thread "main" java.lang.ClassCastException:
  28.  
    * test.Father cannot be cast to test.Son
  29.  
    */
  30.  
    }
  31.  
    }
  32.  
     
  33.  
    class Son extends Father{
  34.  
    public Son(){
  35.  
    }
  36.  
     
  37.  
    public void test1(){
  38.  
    System.out.println("Son的test1方法。。。。");
  39.  
    }
  40.  
     
  41.  
    public void test2(){
  42.  
    System.out.println("Son的test2方法。。。。");
  43.  
    }
  44.  
    }
             

Java基础之类型转换总结篇的更多相关文章

  1. Java基础加强之多线程篇(线程创建与终止、互斥、通信、本地变量)

    线程创建与终止 线程创建 Thread类与Runnable接口的关系 public interface Runnable { public abstract void run(); } public ...

  2. Java基础-MySQL数据库扫盲篇

    Java基础-MySQL数据库扫盲篇 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.数据库概述 1>.什么是数据库 数据库就是存储数据的仓库,其本质是一个文件系统,数据按 ...

  3. Java基础-数据类型转换

     1).简单类型数据间的转换,有两种方式:自动转换和强制转换,通常发生在表达式中或方法的参数传递时.  自动转换 当一个较"小"数据与一个较"大"的数据一起运算 ...

  4. Java基础12 类型转换与多态

    链接地址:http://www.cnblogs.com/vamei/archive/2013/04/01/2992662.html 作者:Vamei 出处:http://www.cnblogs.com ...

  5. java基础-- 之类型转换 和 一些运算符

    ---   首先   看一个 Scanner  类 import java.util.Scanner; public class TestScanner { public static void ma ...

  6. Java基础12 类型转换与多态(转载)

    类型检查 Java的任意变量和引用经过类型声明(type declaration),才能使用.我们之前见过对象数据.类数据.方法参数.方法返回值以及方法内部的自动变量,它们都需要声明其类型.Java是 ...

  7. java基础第十二篇之集合、增强for循环、迭代器和泛型

    Collection接口中的常用方法: * 所有的子类子接口都是具有的 * 集合的方法:增删改查 * * public boolean add(E e);//添加元素 返回值表示是否添加成功 * pu ...

  8. Java基础语法(总结篇)

    关键字&标识符 关键字的概念与特征 概念:Java关键字是事先定义好的对Java的编译器有特殊的意义,他们用来表示一种数据类型,或者表示程序的结构等,关键字不能用作变量名.方法名.类名.包名和 ...

  9. java基础 小知识点汇总篇

    1.&和&& |和|| 的区别? a&b,会执行a,执行b ,得出结论 a&&b,会执行a,如果a false,则不执行b,直接返回false 2.Ma ...

随机推荐

  1. 如何解决Python下 pip install module 下载慢解决方法?

    对于Python来编程的用户最大的一个痛点就是,下载模块是下载速度特别慢,那么有没有解决方法呢? 换Python的pip下载源 1.首先安装一个模块 pqi,在cmd下 pip install pqi ...

  2. python爬取千库网

    url:https://i588ku.com/beijing/0-0-default-0-8-0-0-0-0-1/ 有水印 但是点进去就没了 这里先来测试是否有反爬虫 import requests ...

  3. 最全总结 | 聊聊 Python 数据处理全家桶(Redis篇)

    1. 前言 前面两篇文章聊到了 Python 处理 Mysql.Sqlite 数据库常用方式,本篇文章继续说另外一种比较常用的数据存储方式:Redis Redis:Remote Dictionary ...

  4. 测试工具-XPath使用

    XML有两种MIME类型,即application/xml和text/xml,在HTTP中,MIME Type类型被定义在Content-Type header中.我们经常也会看到接口返回数据类型为X ...

  5. Neo4j---性能优化

    不会项目管理的研发不是好loder(^_^ ^_^),开个玩笑,目的是想说项目管理很重要,研发同胞们需要重视.重视.重视(重要的事情说三遍).随着项目业务扩展,不再是停留在基本某一业务范围,海量数据接 ...

  6. Centos-系统任务队列信息-uptime

    uptime 显示系统的当前时间.系统从启动到当前运行时间.当前总共在线用户.系统1.5.15分钟负载情况

  7. python3 读取写入excel操作-win32com

    前面有写一篇是用xlrd操作excel的,这一篇是使用win32com来进行操作excel,个人推荐使用win32com. 要使用win32com组件,也需要先导入win32com包. # -*- c ...

  8. 排序算法:冒泡排序(Bubble Sort)

    冒泡排序 算法原理 冒泡排序的原理是每次从头开始依次比较相邻的两个元素,如果后面一个元素比前一个要大,说明顺序不对,则将它们交换,本次循环完毕之后再次从头开始扫描,直到某次扫描中没有元素交换,说明每个 ...

  9. Java知识系统回顾整理01基础01第一个程序02命令行格式编译和执行Java程序

    一.先看运行效果 在控制台下运行第一个Java程序,可以看到输出了字符串 hello world 二.准备项目目录 通常都会在e: 创建一个project目录 在这个例子里,我们用的是e:/proje ...

  10. MATLAB中conv2的详细用法 (以及【matlab知识补充】conv2、filter2、imfilter函数原理)

    转载: 1.https://blog.csdn.net/jinv5/article/details/52874880 2.https://blog.csdn.net/majinlei121/artic ...