转 Java类型转换

原文链接:http://www.cnblogs.com/lwbqqyumidi/p/3700164.html

基本数据类型共有8种,分别是:布尔型boolean, 字符型char和数值型byte/short/int/long/float/double。由于字符型char所表示的单个字符与Ascii码中相应整形对应,因此,有时也将其划分到数值型中。引用类型具体可分为:数组、类和接口。因此,本文中Java类型转换的总结也将分为基本数据类型和引用数据类型两个方面展开。

一、基本数据类型的类型转换

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

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

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

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

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

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

是不是有点奇怪?按照上面的思路去理解,将一个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
 3 public class TestCast {
 4
 5     public static void main(String[] args) {
 6         long a = 10000000000; //编译出错: The literal 10000000000 of type int is out of range
 7         long b = 10000000000L; //编译正确
 8         int c = 1000;
 9         long d = c;
10         float e = 1.5F;
11         double f = e;
12     }
13
14 }

如上:定义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
 3 public class TestCast {
 4
 5     public static void main(String[] args) {
 6         byte p = 3; // 编译正确:int到byte编译过程中发生隐式类型转换
 7         int  a = 3;
 8         byte b = a; // 编译出错:cannot convert from int to byte
 9         byte c = (byte) a; // 编译正确
10         float d = (float) 4.0;
11     }
12
13 }

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

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

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

如下经典例子:

按 Ctrl+C 复制代码

按 Ctrl+C 复制代码

为什么结果是-23?需要从最根本的二进制存储考虑。

233的二进制表示为:24位0 + 11101001,byte型只有8位,于是从高位开始舍弃,截断后剩下:11101001,由于二进制最高位1表示负数,0表示正数,其相应的负数为-23。

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

如下代码:

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

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

时间: 2024-08-24 15:20:17

转 Java类型转换的相关文章

java 类型转换

在java项目的实际开发和应用中,常常需要用到将对象转为String这一基本功能.本文将对常用的转换方法进行一个总结.常用的方法有Object.toString(),(String)要转换的对象,String.valueOf(Object)等.下面对这些方法一一进行分析.方法1:采用 Object.toString()方法请看下面的例子:      Object object = getObject();System.out.println(object.toString()); 注1在这种使用

JAVA 类型转换:隐式转换与强制转换

  这篇随笔是对java类型转换的回顾,方便忘记的时候查询,同时希望帮助有这方面需要的朋友 一. 数据类型分类: 简单数据类型:整型.实型.字符型((byte-short-char)-int-long-float-double) 二.简单类型转换 (1)低级向高级转换---自动转换:隐式转换 byte i = 0;        int a =i;        long b = i;        float c =b;        double d =c;        short j =

SQLServer类型与Java类型转换问题解决

ResultSet 接口提供用于从当前行获取列值的获取 方法(getBoolean.getLong 等).可以使用列的索引编号或列的名称获取值.一般情况下,使用列索引较为高效.列从 1 开始编号.为了获得最大的可移植性,应该按从左到右的顺序读取每行中的结果集列,每列只能读取一次. 对于获取方法,JDBC 驱动程序尝试将底层数据转换为在获取方法中指定的 Java 类型,并返回适当的 Java 值.JDBC 规范有一个表,显示允许的从 SQL 类型到 ResultSet 获取方法所使用的 Java

Java 类型转换以及Object转成其他类型

Object转int int count=(int)map.get("count") int count=Integer.parseInt((String)map.get("count")); int  count=Integer.parseInt(map.get("count").toString()); 自动数据类型转换 自动转换按从低到高的顺序转换.不同类型数据间的优先关系如下:    低--------------------------

java类型转换

java基本类型的类型转换有三种: 1.自动类型转换 2.强制类型转换 3.表达式类型的自动提升 (1)自动类型转换 由上图可以看出自动类型转换,其实是从小到大的转换. (2)强制类型转换 如果将上图,反向转换,则需要强制类型转换.并且有可能损失精度. (3)表达式类型的自动提升. 一个算术表达式中可能有多个类型数据进行运算,因此,按照自动类型转换,整个表达式都将转换成最高等级的类型,也就是最大的类型.然后进行运算.  int a=3.4/2.1;   /*  这个语句会报错,因为整个语句中最高

Java类型转换实例

例:将字符串内容转为浮点数,再还原出原字符串内容 import java.io.UnsupportedEncodingException; import java.nio.ByteBuffer; import java.nio.DoubleBuffer; import java.nio.FloatBuffer; public class test3 { public static void main(String[] args) { new test3(); } /** * 第一种方法 */ /

java——类型转换,冒泡排序,选择排序,二分查找,数组的翻转

一.类型转换 p { margin-bottom: 0.25cm; direction: ltr; color: #000000; line-height: 120%; text-align: justify; widows: 0; orphans: 0 } p.western { font-family: "Calibri", sans-serif; font-size: 10pt } p.cjk { font-family: "宋体"; font-size: 1

java类型转换详解(自动转换和强制转换)

自动转换 class Hello { public static void main(String[] args) { //自动转换 int a = 5; byte b = 6; int c = a + b; System.out.println(c); } } a是int类型,b是byte类型 当二者进行加法运算时(根据同类型相加结果还是同类型) 由于int的范围比byte的取值范围大这个时候jvm会自动将b转换成int类型 强制转换 //强制转换 int a = (int)8.8; 强制转换

java 类型转换前先做检查

1.传统的类型转换由RTTI确保正确性. 2.instanceof关键字(二元操作符) ,返回一个Boolean值,告诉我们对象是不是某个类或该类派生类的实例,他判断的是类型. if (a instanceof A) //对象a是不是属于A类型,A可能是a父类的父类,如果是这样也返回true 3.instanceof 不能比较Class对象,对于Class对象使用isAssignableFrom()判断 if (as.isAssignableFrom(cs))// Class对象cs所在类是不是