零基础学习之Java包装类
概述
说起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);
}
}
包装类常见的API
基本数据类型和字符串之间的转换
(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方法,因此比较的是内容而不是地址,而“= =”比较的依然是引用变量的地址,只是当包装类型和与之相对应的基本类型进行“==”比较时会先做自动拆箱处理。具体可以分类如下:
- ①如果在默认范围内的自动装箱,则相等(比较地址)
- ②如果比较的对象有自己new的,则不相等(比较地址)
- ③如果比较的对象值不在范围内部也是不想等的(比较地址)
- ④如果有基本数据类型参与对比,包装类型就会自动拆箱,就会变成了基本数据类型的对比,
这个时候比的就是值了。
常见笔试或面试问题举例
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在对比) )
}
}