Java中,经常可以遇到类型转换的场景,从变量的定义到复制、数值变量的计算到方法的参数传递、基类与派生类间的造型等,随处可见类型转换的身影。Java中的类型转换在Java编码中具有重要的作用。

首先,来了解下数据类型的基本理解:数据是用来描述数据的种类,包括其值和基于其值基础上的可进行的操作集合。

Java中数据类型主要分为两大类:基本数据类型和引用数据类型。

基本数据类型共有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为一变量,需要到运行期间才能确定,也就是说,编译期间为以防万一,当然不可能编译通过了。此时,需要进行强制类型转换。

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

如下经典例子:

1 package com.corn.testcast;

2

3 public class TestCast {

4

5 public static void main(String[] args) {

6 int a = 233;

7 byte b = (byte) a;

8 System.out.println("b:" + b); //输出:-23

9 }

10

11 }

为什么结果是-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 }

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

二、引用类型的类型转换/造型

本部分内容将在后续时间在本文中补上。

java中类型转换的造型_Java总结篇系列:类型转换/造型相关推荐

  1. 浅析Java中对象的创建与对象的数据类型转换

    这篇文章主要介绍了Java中对象的创建与对象的数据类型转换,是Java入门学习中的基础知识,需要的朋友可以参考下 Java:对象创建和初始化过程 1.Java中的数据类型     Java中有3个数据 ...

  2. java中实参和形参_java中形参和实参的区别

    实参和形参是程序设计语言中的通用概念,并不是只有C,C++有而JAVA没有.关于他们两者的区别你知道吗?下面是学习啦小编为大家准备的java中形参和实参的区别,希望大家喜欢! java中形参和实参的区 ...

  3. java x轴 左正_Java总结篇系列:Java泛型

    一. 泛型概念的提出(为什么需要泛型)? 首先,我们看下下面这段简短的代码: 1 public class GenericTest { 2 3 public static void main(Stri ...

  4. java多态 降低代码耦合性_深度分析:理解Java中的多态机制,一篇直接帮你掌握!...

    Java中的多态 1 多态是什么 多态(Polymorphism)按字面的意思就是"多种状态".在面向对象语言中,接口的多种不同的实现方式即为多态.用白话来说,就是多个对象调用同一 ...

  5. java中break内外循环_java 中break如何跳出外部循环

    学习就是为了不断的看到自己的知识盲点,然后改正,以前知道如何使用break来跳出循环,突然学习到可以用break跳出外部的循环(以前只知道怎么调本次的循环). 上正题代码如下: break跳出本次循环 ...

  6. java中数据结构的应用_Java集合入门 (二)常用数据结构和应用场景-数组

    Java极客  |  作者  /  铿然一叶 这是Java极客的第 48 篇原创文章 一.数组的特点 1.数组大小固定 2.一个数组只能存储相同数据类型 3.随机访问性能高 4.存储空间连续,这样可以 ...

  7. java中类型的相互转化_Java中的数据类型及相互转换方法

    本文主要讲解两个部分: 一.Java中的数据类型有哪些? 二.数字类型和字符串类型相互转换的方法? 一.Java中的数据类型有哪些: Java中的数据类型有:基本数据类型和引用数据类型: 基本数据类型 ...

  8. java 中random类使用_Java中的天使和魔鬼:Unsafe类

    我们在看ConcurrentHashMap源码时经常看到Unsafe类的使用,今天我们来了解下Unsafe类. Java是一个安全的编程语言,它能最大程度的防止程序员犯一些低级的错误(大部分是和内存管 ...

  9. java中的基本数据类型_Java中的基本数据类型和引用数据类型

    数据类型用于帮助确定变量可存放的一组值,以及可对这组特定值执行的操作.Java提供了在所有平台上都普遍支持的多种数据类型. 一.基本数据类型 byte:Java中最小的数据类型,在内存中占8位(bit ...

最新文章

  1. CMB/宇宙学中相关仪器设备和术语
  2. c语言作业小学生测试题,C语言实现小学生随机出题测试计分
  3. mysql防止预约重号_mysql 防止重复插入唯一限制的数据
  4. ctfshow-WEB-web9( MD5加密漏洞绕过)
  5. alsa的动态库安装在哪里_源码编译安装MySQL8.0.20
  6. Linux设置时间和硬件时间
  7. 1-7华为HCNA认证eNSP基础B
  8. windows下端口映射(端口转发)
  9. MDK-ARM_V525新建工程 STM32使用软件仿真、RAM仿真调试及Flash下载配置详解
  10. Autodesk BIM 360 全球在线骇客马拉松
  11. opcode加密php代码,总结Opcode缓存和PHP代码的加密
  12. 电气潮流运算Matlab怎么编程,基于Matlab的电力系统潮流编程计算
  13. 【思维导图】Excel转成思维导图
  14. oracle internal_function,Oracle Internal Research内部原理研究
  15. Python调用百度AI接口
  16. 安徽大学在校生如何校外访问图书馆资源
  17. 微信小程序页面间传值方法
  18. ESP32公网对讲机
  19. 浅聊Gatsby静态站点生成器
  20. MySQL之PXC集群搭建

热门文章

  1. Apache Shiro 简介
  2. COM组件的运行机制
  3. 2013年3月16日星期六
  4. 它们是什么以及为什么我们不需要它们
  5. 软件工程方法学要素含义_日期时间数据的要素工程
  6. html中列表导航怎么和图片对齐_HTML实战篇:html仿百度首页
  7. pd种知道每个数据的类型_每个数据科学家都应该知道的5个概念
  8. Memory-Associated Differential Learning论文及代码解读
  9. leetcode 387. 字符串中的第一个唯一字符(hash)
  10. leetcode 1365. 有多少小于当前数字的数字(排序)