包装类
Java 针对八种基本数据类型定义了相应的引用类型:包装类(封装类)。有了包装类的特点,就可以调用类中的方法,Java 才是真正的面向对象。
把基本类型进行包装,提供更加完善的功能。
基本类型是没有任何功能的,只是一个变量,记录值,而包装类可以有更加丰富的功能
与基本类型的对应关系
概述
自动装箱:把 基本类型 包装成对应的 包装类型 的过程
Integer obj1 = new Integer(4);//使用构造函数函数
Float f = new Float(“4.56”);
Long l = new Long(“asdf”); //NumberFormatException
Integer obj2 = Integer.valueOf(4);
//使用包装类中的 valueOf 方法Integer a = 5;
//a是引用类型,引用了包装对象的地址。
编译器会完成对象的自动装箱:Integer a = Integer.valueOf(5);
自动拆箱:从包装类型的值,自动变成 基本类型的值
int i = a;//a现在是包装类型,没法给变量赋值,需要把5取出来。
编译器会完成自动拆箱:int i = a.intValue();
练习: 自动装箱与自动拆箱测试
创建包: cn.tedu.api
创建类: TestBox.java
package cn.tedu.api;
/*本类用于测试自动装箱和自动拆箱*/
public class TestBox {
public static void main(String[] args) {
//1.定义包装类型的数据
//回顾:以前创建包装类型的两种方式
Integer i1 = new Integer(127);
Integer i2 = Integer.valueOf(127);
//2.现在的方式:
/*1.自动装箱:编译器会自动把基本类型int 5,包装成包装类型Integer
* 然后交给i3来保存,自动装箱底层发生的代码Integer.valueOf(5);
* valueOf()的方向: int --> Integer*/
Integer i3 = 5;//不会报错,这个现象就是自动装箱
/*2.自动拆箱:编译器会自动把包装类型的i1拆掉"箱子",变回基本类型数据127
* 然后交给i4来保存,自动拆箱底层发生的代码:i1.intValue();
* intValue()的方向:Integer -> int
* */
int i4 = i1;//不会报错,这个现象就是自动拆箱
}
}
BigDecimal
BigDecimal:常用来解决精确的浮点数运算不精确的问题
创建对象
方式一 :
BigDecimal(double val)
将double转换为BigDecimal,后者是double的二进制浮点值十进制表示形式,有坑!
方式二 :
BigDecimal(String val)
将String类型字符串的形式转换为BigDecimal
常用方法
Add(BigDecimal bd) : 做加法运算
Subtract(BigDecimal bd) : 做减法运算
Multiply(BigDecimal bd) : 做乘法运算
Divide(BigDecimal bd) : 做除法运算,除不尽时会抛异常
Divide(BigDecimal bd,保留位数,舍入方式) : 除不尽时使用
setScale(保留位数,舍入方式) : 同上
pow(int n) : 求数据的几次幂
练习:测试常用方法
创建包: cn.tedu.bigdecimal
创建类: TestBigDecimal.java
需求: 接收用户输入的两个小数,做运算
package cn.tedu.api;
import java.math.BigDecimal;
import java.util.Scanner;
public class TestBigDecimal {
public static void main(String[] args) {
//f1();//使用普通的 +-*/ 四则运算,暴露出浮点数运算不精确的问题
f2();//使用BigDecimal来解决浮点数运算不精确的问题
}
private static void f2() {
//1.提示并接收用户输入的两个小数
System.out.println("请输入您要计算的两个小数:");
double a = new Scanner(System.in).nextDouble();
double b = new Scanner(System.in).nextDouble();
//2.创建工具类对象,把基本类型a和b交给工具类对象BigDecimal来保存
/*1.最好不要用double作为构造函数的参数,不然还会有不精确的现象,有坑!!!*/
/*2.最好使用重载的,参数类型是String的构造函数
* double转String,直接拼个空串就可以*/
BigDecimal bd1 = new BigDecimal(a+"");
BigDecimal bd2 = new BigDecimal(b+"");
//3.通过BigDecimal的对象来调用其方法,实现精确运算
//3.1 定义BigDecimal类型的引用类型变量来保存结果
BigDecimal bd3;
//3.2 Add(BigDecimal bd) : 做加法运算
bd3 = bd1.add(bd2);
System.out.println(bd3);
//3.3 Subtract(BigDecimal bd) : 做减法运算
bd3 = bd1.subtract(bd2);
System.out.println(bd3);
//3.4 Multiply(BigDecimal bd) : 做乘法运算
bd3 = bd1.multiply(bd2);
System.out.println(bd3);
//3.5 Divide(BigDecimal bd) : 做除法运算,除不尽时会抛异常
/*3.除法运算,除不尽时会抛出异常ArithmeticException*/
//方案一:(除不尽时有问题)
//bd3 = bd1.divide(bd2);
/*divide(m,n,o)
m是要除以哪个对象,n指要保留几位,o指舍入方式(比如四舍五入)*/
//方案二:
bd3 = bd1.divide(bd2,3,BigDecimal.ROUND_HALF_UP);
System.out.println(bd3);
}
private static void f1() {
//1.提示并接收用户输入的两个小数
System.out.println("请输入您要计算的两个小数:");
double a = new Scanner(System.in).nextDouble();
double b = new Scanner(System.in).nextDouble();
//2.做运算
System.out.println(a + b);//不精确
System.out.println(a - b);//不精确
System.out.println(a * b);//不精确
System.out.println(a / b);//不精确
}
}
** 拓展**
舍入方式解析
ROUND_HALF_UP 四舍五入,五入 如:4.4结果是4; 4.5结果是5
ROUND_HALF_EVEN 公平舍入(银行常用):比如:在5和6之间,靠近5就舍弃成5,靠近6就进位成6,如果是5.5,就找偶数,变成6
ROUND_UP 直接进位,不管0.1还是0.9,都进位
ROUND_DOWN 直接舍弃,不算0.1还是0.9,都舍弃
ROUND_CEILING(天花板) 向上取整,取实际值的大值
ROUND_FLOOR(地板) 向下取整,取实际值的小值