深入理解Java之装箱与拆箱
一、Java数据类型
1、在说装箱与拆箱之前,先说一下Java的基本数据类型,Java从数据类型上可以划分为值类型
与引用类型
,值类型是四类八种
,分别是:
- 整数型:byte̵,short̵,int̵,long
- 浮点型:float,double
- 字符型:char
- 布尔型:boolean
数据类型 | 内存 | 默认值 | 包装类 |
---|---|---|---|
byte | 8位 | 0 | Byte |
short | 16位 | 0 | short |
int | 32位 | 0 | Integer |
long | 64位 | 0L或0l | Long |
float | 32位 | 0.0F或0.0f | Float |
double | 64位 | 0.0D或0.0d | Double |
char | 16位 | \u0000 | Character |
boolean | 8位 | flase | Boolean |
2、引用类型:
- 数组
- 类(class)
- 接口(Interface)
- 枚举(enum)
3、值类型与引用类型的区别
- 从概念方面上来说:
- 值类型:变量名指向具体的值
- 引用类型:变量名指向数据对象的内存地址
- 从内存构建方面上来说:
- 值类型:变量在声明之后,Java就会立刻分配给它内存空间
- 引用类型:它以特殊的方式(类似C指针)指向对象实体,这类变量声明时不会分配内存,只是存储
- 从使用方面上来说:
- 值类型:使用时需要赋具体值,判断时用 ” == “号
- 引用类型:使用时可以赋null,判断时使用 equals 方法
二、Java数据类型转换
1、自动转换
定义:程序在执行过程中“悄然”进行的转换,不需要用户提前声明,一般是从位数低的类型向位数高的类型转换
优先关系:按从低到高的顺序转换。不同类型数据间的优先
关系如下:
- 低--------------------------------------------->高
- byte,short,char-> int -> long -> float -> double
转换规则:
运算中,不同类型的数据先转化为同一类型,然后进行运算
操作数1类型 操作数2类型 转换后的类型 byte、short、char int int byte、short、char、int long long byte、short、char、int、long float float byte、short、char、int、long、float double double
2、强制转换
定义:强制类型转换则必须在代码中声明,转换顺序不受限制
格式:在需要转型的数据前加上“( )”,然后在括号内加入需要转化的数据类型
结果:精度可能会丢失,也可能更加精确
int x;
double y; x = (int)3.14 + (int)5.20 //精度丢失
y = (double)x + (double)8 //精度提升 输出:x = 8;y = 16.0
三、Java之装箱与拆箱
1、包装类
- Java是面向对象语言,号称万事万物皆对象,因此,8种基本数据类型有了对应的类,这就是包装类
2、什么是装箱与拆箱
装箱:将值类型装换成引用类型的过程
拆箱:将引用类型转换成值类型的过程
自动装箱:
int x = 3;
Integer y = x; //int --> Integer,Integer y = x <==> Integer y = Integer.valueOf(x)
自动拆箱:
Integer x = new Integer(5);
int y = x; //Integer --> int,int y = x <==> int y = x.intValue()
3、装箱和拆箱是如何实现的
- 装箱过程是通过调用包装器的valueOf方法实现的
- 拆箱过程是通过调用包装器的 xxxValue方法实现的。(xxx代表对应的基本数据类型)
4、注意点:
大量使用自动拆装箱会使性能降低,还会造成大量的内存消耗
在重载方法中,可能出现问题
List<Integer> list = new ArrayList<>();
Integer x,y,z;
x = 1;y = 2;z = 4;
list.add(x);list.add(y);list.add(z); list.remove(2);
在上面这段代码中ArrayList.remove方法有两个重载方法,那么list.remove(2)是调用了哪个方法,remove掉的是值为2的对象,还是remove了index为2,值为4的那个对象呢?
在这种情况下,编译器不会进行自动拆装箱,所以调用的是remove(int index),index为2值为4的这个Integer对象会被remove.
如果要调用 remove(Object o)的方法,应该这么写 list.remove(y)
缓存值问题
案例解析:
Integer i1 = 100;
Integer i2 = 100;
Integer i3 = 200;
Integer i4 = 200;
System.out.println(i1==i2);
System.out.println(i3==i4); Output: true false
观察源码:
Intteger.valueOf方法
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
IntegerCache类
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[]; static {
// high value may be configured by property
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h; cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++); // range [-128, 127] must be interned (JLS7 5.1.7)
assert IntegerCache.high >= 127;
} private IntegerCache() {}
}
从源码可以看出,在通过valueOf方法创建Integer对象的时候,如果数值在[-128,127]之间,便返回指向IntegerCache.cache中已经存在的对象的引用;否则创建一个新的Integer对象
Byte、Short、Integer、Long四种包装类默认创建了数值为[-128,127]的相应类型的缓存数据,但是超出此范围仍会创建新的对象。
Character默认会创建[0,127]的响应类型的缓存数据
两种浮点型没有实现常量池技术,在某个范围内的整型数值的个数是有限的,而浮点数却不是
包装类 常量池 常量池范围 Byte 存在 [-128,127] Short 存在 [-128,127] Integer 存在 [-128,127] Long 存在 [-128,127] Character 存在 [0,127] Float 不存在 无 Double 不存在 无 注意点:
- 当 "=="运算符的两个操作数都是 包装器类型的引用,则是比较指向的是否是同一个对象,而如果其中有一个操作数是表达式(即包含算术运算)则比较的是数值(即会触发自动拆箱的过程)
- 对于包装器类型,equals方法并不会进行类型转换
- 算术运算会触发装箱与拆箱过程
文章为原创,转载请声明出处
深入理解Java之装箱与拆箱的更多相关文章
- 全面理解java自动装箱和拆箱(转)
自动装箱和拆箱从Java 1.5开始引入,目的是将原始类型值转自动地转换成对应的对象.自动装箱与拆箱的机制可以让我们在Java的变量赋值或者是方法调用等情况下使用原始类型或者对象类型更加简单直接. 如 ...
- 如何理解Java中的自动拆箱和自动装箱?
小伟刚毕业时面的第一家公司就被面试官给问住了... 如何理解Java中的自动拆箱和自动装箱? 自动拆箱?自动装箱?什么鬼,听都没听过啊,这...这..知识盲区... 回到家后小伟赶紧查资料,我透,这不 ...
- 【转】java 自动装箱与拆箱
java 自动装箱与拆箱 这个是jdk1.5以后才引入的新的内容,作为秉承发表是最好的记忆,毅然决定还是用一篇博客来代替我的记忆: java语言规范中说道:在许多情况下包装与解包装是由编译器自行完成的 ...
- Java进阶(三十七)java 自动装箱与拆箱
Java进阶(三十七)java 自动装箱与拆箱 前言 这个是jdk1.5以后才引入的新的内容.java语言规范中说道:在许多情况下包装与解包装是由编译器自行完成的(在这种情况下包装称为装箱,解包装称为 ...
- java 自动装箱自动拆箱
1.Java数据类型 在介绍Java的自动装箱和拆箱之前,我们先来了解一下Java的基本数据类型. 在Java中,数据类型可以分为两大种,Primitive Type(基本类型)和Reference ...
- Java 自动装箱与拆箱
Java 自动装箱与拆箱(Autoboxing and unboxing) 什么是自动装箱拆箱 基本数据类型的自动装箱(autoboxing).拆箱(unboxing)是自J2SE 5.0开始提供 ...
- [转]JAVA自动装箱和拆箱
http://www.cnblogs.com/dolphin0520/p/3780005.html 1.Java数据类型 装箱和拆箱之前,我们先来了解一下Java的基本数据类型. 在Java中,数据类 ...
- JAVA基础之——三大特征、接口和抽象类区别、重载和重写区别、==和equals区别、JAVA自动装箱和拆箱
1 java三大特征 1)封装:即class,把一类实体定义成类,该类有变量和方法. 2)继承:从已有的父类中派生出子类,子类实现父类的抽象方法. 3)多态:通过父类对象可以引用不同的子类,从而实现不 ...
- [转]java 自动装箱与拆箱
转自:http://www.cnblogs.com/shenliang123/archive/2012/04/16/2451996.html 这个是jdk1.5以后才引入的新的内容,作为秉承发表是最好 ...
随机推荐
- C#LeetCode刷题-拒绝采样
拒绝采样篇 # 题名 通过率 难度 470 用 Rand7() 实现 Rand10() 34.4% 中等 478 在圆内随机生成点 22.8% 中等
- C#LeetCode刷题之#225-用队列实现栈(Implement Stack using Queues)
问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/4106 访问. 使用队列实现栈的下列操作: push(x) -- ...
- python使用zipfile递归压缩和解压缩文件
import shutil,zipfile,os class ToolModel(object): def dfs_get_zip_file(self,input_path, result, igno ...
- MySQL执行文件中的SQL
连接进入MySQL服务, 使用source ${文件名}执行. 末尾不能带分号.
- MacOS抓包工具Charles
抓包工具有wireshark, tcpdump, 还有就是Charles. 今天分享的是最后一个Charles.抓包分2个, 一个是移动端的,一个是macOS自带的应用. 安装Charles http ...
- 【期外】(三)Linux下集成开发环境Geany
今天小编发现了一个很好的软件,它的名字就叫做Geany. 这是Linux系统中的开发工具,相当的好用. Linux与windows最大的不同正是不是集成开发环境,所以写起代码来总是用文档写好后,然后再 ...
- 初入Shell
shell 第1章 Shell概述 大数据程序员为什么要学习Shell呢? 1)需要看懂运维人员编写的Shell程序. 2)偶尔会编写一些简单Shell程序来管理集群.提高开发效率. 第2章 Shel ...
- 构造vlan报文
#!/usr/bin/python from scapy.all import * packet = Ether(src='dc:99:14:01:a3:5e', dst='ff:ff:ff:ff:f ...
- Open MPI 4.0 编译安装
电脑上目前使用的mpi环境是2.1.1版本的openmpi,是我之前直接使用系统的包管理工具安装的.但是系统包版本一般都比较老旧,现在openmpi最新版已经出到了4.0,即将出4.1了,所以我打算升 ...
- 第3篇 Scrum 冲刺博客(专✌️团队)
目录 一.站立式会议 1.1 会议照片 1.2 成员完成情况 二.项目燃尽图 三.成员代码/文档签入记录 3.1 代码签入 3.2 Issue链接 3.3 CodeReview代码规范文档 四.最新项 ...