说起Java包装类就不得不说Java的特性 - 面向对象。Java是Sun公司的程序James Gosling,Bill Joe本来打算试图修改和扩展C 的功能,新开发一个语言Oak的发展。C语言是一个面向过程的语言,改进后的Java是面向对象,这是一个非常大的改进,这也是Java发展至今依旧非常流行的原因。
但是,Java又不是完全的面向对象,它沿用了C语言的基本数据类型(这是为了便于开发者的使用,使得在进行基本的数据计算时,开发者可以直接使用基础类),但当需要和Java其他对象结合使用,如存入集合中,就需要将基础数据类型实例封装为Java对象,这样操作起来又比较麻烦了,为了满足面向对象的这一特性,Java的java.lang包中设置了包装类,使得每一个基本类型都有对应的包装类,这样直接就方便了开发者的使用。
简单理解来说:包装类就是为了使得Java中的基本数据类型具有面向对象的特性。
基本数据类型对应的包装类有8个(,根据其继承关系可以分成三类:Character、Number、Boolean。继承关系图如下:
这8个包装类和基本数据类型的对应关系如下表:
基本数据类型 | 包装类(java.lang包) |
---|---|
byte | Byte |
short | Short |
int | Integer |
long | Long |
float | Float |
double | Double |
char | Character |
boolean | Boolean |
把基本类型变成包装类对象的过程称为装箱。
转为包装类的对象,是为了使用专门为对象设计的API和特性(比如泛型)
在JDK1.5之前,这个转化过程需要手动实现,即使用包装类的构造器来创建对象。(称为手动装箱);
在JDK1.15之后,这个转化过程可以自动实现,即直接赋值。(称为自动装箱)
import org.junit.Test;//导入junit包,用来进行单元测试(比main方法好用,一个类可以创建多个程序入口) public class Demo7 { /** * Integer(Byte,Double,Float,Long,Short类似) */ @Test public void test1(){ int i1 = 10; //定义基本数据类型(整型) //基本数据类型转化成包装类 //手动装箱(使用构造器) Integer i2 = new Integer(i1); Integer i3 = new Integer("11"); //也可以是字符类型,但是必须是数字,否则报错 //自动装箱(直接赋值) Integer i4 = i1; //输出结果,测试验证 System.out.println(i1); System.out.println(i2); System.out.println(i3); System.out.println(i4); } /** * Character */ @Test public void test2(){ char c1 = 'a'; //定义基本数据类型(字符型) //基本数据类型转化成包装类 //手动装箱(构造器) Character c2 = new Character(c1); //Character c3 = new Character("a"); 不能是字符,如果是字符就会分不清,本身就是了 //自动装箱(直接赋值) Character c4 = c1; //输出结果,测试验证 System.out.println(c1); System.out.println(c2); System.out.println(c4); } /** * Boolean */ @Test public void test3(){ boolean b1=true; //定义基本数据类型(布尔型) //基本数据类型转化成包装类 //手动装箱 Boolean b2=new Boolean(b1); Boolean b3=new Boolean("true");//只要参数不是true这个字符串,都返回false if(b3) System.out.println("b3=true"); else System.out.println("b3=false"); //自动装箱(直接赋值) Boolean b4=b1; //输出结果,测试验证 System.out.println(b1); System.out.println(b2); System.out.println(b4); } }
与装箱相反,拆箱就是把包装类对象转化为基本类型的过程。
转为基本数据类型,一般是因为需要运算,Java中的大多数运算符是为基本数据类型设计的。比较、算术等,这样的效率更高
在JDK1.5之前,这个转化过程需要手动实现,即通过对应的包装类调用xxxValue()方法。(称为手动拆箱);
在JDK1.15之后,这个转化过程可以自动实现,即直接赋值。(称为自动拆箱)
import org.junit.Test; public class Demo7 { /** * Integer(Byte,Double,Float,Long,Short类似) */ @Test public void test1(){ int i1 = 10;//定义基本数据类型(整型) //自动装箱,用于后面拆箱 Integer i4 = i1; //包装类转化为基本数据类型 //手动拆箱(调用xxxValue()) int i5 = i4.intValue(); //自动拆箱(直接赋值) int i6 = i4; //输出结果,测试验证 System.out.println(i1); System.out.println(i4); System.out.println(i5); System.out.println(i6); } /** * Character */ @Test public void test2(){ char c1 = 'a';//定义基本数据类型(字符型) //自动装箱,用于后面拆箱 Character c4 = c1; //包装类转化为基本数据类型 //手动拆箱 char c5 = c4.charValue(); //自动拆箱 char c6 =c4; //输出结果,测试验证 System.out.println(c1); System.out.println(c4); System.out.println(c5); System.out.println(c6); } /** * Boolean */ @Test public void test3(){ boolean b1=true; //定义基本数据类型(布尔型) //自动装箱,用于后面拆箱 Boolean b4=b1; //包装类转化为基本数据类型 //手动拆箱 boolean b5=b2.booleanValue(); //自动拆箱 boolean b6=b2; //输出结果,测试验证 System.out.println(b1); System.out.println(b4); System.out.println(b5); System.out.println(b6); } }
(1)把基本数据类型转为字符串(调用valueOf()方法)
import org.junit.Test; public class Demo8 { /** * 基本数据类型转化为字符型 */ @Test public void test(){ int a = 10; //定义基本数据类型 //String str = a;//不能直接转换 //方式一: String str = a + ""; //方式二: String str1 = String.valueOf(a); } }
(2)把字符串转为基本数据类型
String转换成对应的基本类型 ,除了Character类之外,其他所有包装类都具有parseXxx()静态方法可以将字符串参数转换为对应的基本类型:
public static byte parseByte(String s)
:将字符串参数转换为对应的byte基本类型。public static short parseShort(String s)
:将字符串参数转换为对应的short基本类型。public static int parseInt(String s)
:将字符串参数转换为对应的int基本类型。public static long parseLong(String s)
:将字符串参数转换为对应的long基本类型。public static float parseFloat(String s)
:将字符串参数转换为对应的float基本类型。public static double parseDouble(String s)
:将字符串参数转换为对应的double基本类型。public static boolean parseBoolean(String s)
:将字符串参数转换为对应的boolean基本类型。int a = Integer.parseInt("整数的字符串"); double a = Double.parseDouble("小数的字符串"); boolean b = Boolean.parseBoolean("true或false");
注意: 如果字符串参数的内容无法正确转换为对应的基本类型,
则会抛出java.lang.NumberFormatException
异常。
求某个数据类型的最大值和最小值,直接调用MAX_VALUE和MIN_VALUE方法
import org.junit.Test; public class Demo8 { /** * Integer类型的最大值和最小值(其他类似) */ @Test public void test1(){ //最大值 int maxValue = Integer.MAX_VALUE; //最小值 int minValue = Integer.MIN_VALUE; //输出结果,验证测试 System.out.println(maxValue); System.out.println(minValue); } }
如果转包装类型的大小写,则需要调用toUpperCase()方法和toLowerCase()方法
import org.junit.Test; public class Demo8 { /** * Character转大小写 */ @Test public void test2(){ char x1 = Character.toUpperCase('x'); char x2 = Character.toLowerCase('X'); System.out.println(x1); System.out.println(x2); } }
如果转包装类型的进制,则需要调用对应方法(二进制toBinaryString(),十六进制toHexString(),八进制toOctalString())
package com.atguigu.demo; import org.junit.Test; public class Demo8 { /** * 转进制 */ @Test public void test3(){ int i = 6; //十进制的6 String s1 = Integer.toBinaryString(i); //二进制 String s2 = Integer.toHexString(i); //十六进制 String s3 = Integer.toOctalString(i); //八进制 System.out.println(s1); System.out.println(s2); System.out.println(s3); } }
有一部分包装类提供了对象的缓存。对于频繁使用的包装类对象,其类在初始化时就会提前创建好对象,当需要使用该包装类的对象时,如果该对象包装的值在缓存的范围内(这个范围是在Java源码中设置的),就返回缓存的对象;当该对象的值大于缓存的范围,就会创建新的对象并返回。
下面以Integer的源码进行说明
Integer i = 10; //自动装箱,默认会调用valueOf方法 /** * Integer的自动装箱源码 */ public static Integer valueOf(int i) { //判断i的值,是否在一个范围 -128 ~ 127 从数组中取值(地址) if (i >= IntegerCache.low && i <= IntegerCache.high) //如果值在这个范围,返回一个数组中的值 return IntegerCache.cache[i + (-IntegerCache.low)]; //如果不在这个范围,new一个(和自己new的对象没区别) return new Integer(i); }
各个包装类对应的缓存对象大小如下表:
包装类 | 缓存对象 |
---|---|
Byte | -128~127 |
Short | -128~127 |
Integer | -128~127 |
Long | -128~127 |
Float | 没有 |
Double | 没有 |
Character | 0~127 |
Boolean | true和false |
包装类中“== ”与equals的用法比较,可以算是非常重要的考点了。
包装类中的equals方法和String类一样,都是重写了Object类中的equals方法,因此比较的是内容而不是地址,而“= =”比较的依然是引用变量的地址,只是当包装类型和与之相对应的基本类型进行“==”比较时会先做自动拆箱处理。具体可以分类如下:
import org.junit.Test; public class Demo9 { @Test public void test(){ Integer i1 = 10; //1. Integer这个类会加载 2. 在创建i2这个局部变量 Integer i2 = 10; System.out.println(i1 == i2); //true 对比地址 Integer i3 = new Integer(10); System.out.println(i1 == i3); //false 对比地址 Integer i4 =300; Integer i5 =300; System.out.println(i4 == i5);//false 对比地址 Integer i6 = 300; int i7 = 300; System.out.println(i6 == i7); //true 对比值(因为有基本数据类型(i6会转为int在对比) ) } }