9. Java常用类

一、字符串相关的类

String的特性

String:字符串,使用一对""引起来表示。

  1. String声明为final的,不可被继承
  2. String:
    • 实现了Serializable接口:表示字符串是支持序列化的。
    • 实现了Comparable接口:表示String可以比较大小。
  3. String内部定义了final char[] value用于存储字符串数据
  4. String:代表不可变的字符序列。简称:不可变性。
    • 体现
      • 当对字符串重新赋值时,需要重写指定内存区域赋值,不能使用原有的value进行赋值。
      • 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
      • 当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
  5. 通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。
  6. 字符串常量池中是不会存储相同内容的字符串的。

String的实例化方式:

方式一:通过字面量定义的方式

  • 字符串常量存储在字符串常量池,目的是共享

通过字面量定义的方式:此时的s1和s2的数据javaEE声明在方法区中的字符串常量池中。

String s1 = "javaEE";
String s2 = "javaEE";

方式二:通过new + 构造器的方式

  • 字符串非常量对象存储在堆中。

通过new + 构造器的方式:此时的s3和s4保存的地址值,是数据在堆空间中开辟空间以后对应的地址值。

String s3 = new String("javaEE");
String s4 = new String("javaEE");

注意:通过new+构造器的方式实例化,s3和s4中存储的地址指向堆空间的value[],而value又指向存放在常量池中的数据

Person p1 = new Person("Tom",12);
Person p2 = new Person("Tom",12);
System.out.println(p1.name == p2.name);//true

面试题:String s = new String("abc");方式创建对象,在内存中创建了几个对象?

  • 两个:一个是堆空间中new结构,另一个是char[]对应的常量池中的数据:"abc"
ab76e17340674ff7ca6ac8190247f066.png
String s1 = "javaEE";
String s2 = "hadoop";
String s3 = "javaEEhadoop";
String s4 = "javaEE" + "hadoop";
String s5 = s1 + "hadoop";
String s6 = "javaEE" + s2;
System.out.println(s3 == s4);//true
System.out.println(s3 == s5);//false
System.out.println(s5 == s6);//false
String s8 = s6.intern();//返回常量值中已经存在的“javaEEhadoop”
System.out.println(s3 == s8);//true
final String s9 = "javaEE";//s4:常量
String s10 = s9 + "hadoop";
System.out.println(s3 == s10);//true

结论:

  1. 常量与常量的拼接结果在常量池,且常量池中不会存在相同内容的常量。
  2. 只要其中有一个是变量,结果就在堆中。
  3. 如果拼接的结果调用intern()方法,返回值就在常量池中

注意:在代码s1 = s1 + "b"; 中实际上原来的“a”字符串对象已经丢弃了,现在在堆空间中产生了一个字符串s1+"b"(也就是"ab")。如果多次执行这些改变串内容的操作,会导致大量副本字符串对象存留在内存中,降低效率。如果这样的操作放到循环中,会极大影响程序的性能。(引入后面的StringBuffer和Stringbuild)

String常用方法

基础方法:

  • int length():返回字符串的长度: return value.length
  • char charAt(int index): 返回某索引处的字符return value[index]
  • boolean isEmpty():判断是否是空字符串:return value.length == 0
  • String toLowerCase():使用默认语言环境,将 String 中的所有字符转换为小写
  • String toUpperCase():使用默认语言环境,将 String 中的所有字符转换为大写
  • String trim():返回字符串的副本,忽略前导空白和尾部空白
  • boolean equals(Object obj):比较字符串的内容是否相同
  • boolean equalsIgnoreCase(String anotherString):与equals方法类似,忽略大小写
  • String concat(String str):将指定字符串连接到此字符串的结尾。 等价于用“+”
  • int compareTo(String anotherString):比较两个字符串的大小
  • String substring(int beginIndex):返回一个新的字符串,它是此字符串的从beginIndex开始截取到最后的一个子字符串。
  • String substring(int beginIndex, int endIndex) :返回一个新字符串,它是此字符串从beginIndex开始截取到endIndex(不包含)的一个子字符串。

对于字符串格式判断:

  • boolean endsWith(String suffix):测试此字符串是否以指定的后缀结束
  • boolean startsWith(String prefix):测试此字符串是否以指定的前缀开始
  • boolean startsWith(String prefix, int toffset):测试此字符串从指定索引开始的子字符串是否以指定前缀开始
  • boolean contains(CharSequence s):当且仅当此字符串包含指定的 char 值序列时,返回 true

增删改查:

  • int indexOf(String str):返回指定子字符串在此字符串中第一次出现处的索引
  • int indexOf(String str, int fromIndex):返回指定子字符串在此字符串中第一次出现处的索引,从指定的索引开始
  • int lastIndexOf(String str):返回指定子字符串在此字符串中最右边出现处的索引
  • int lastIndexOf(String str, int fromIndex):返回指定子字符串在此字符串中最后一次出现处的索引,从指定的索引开始反向搜索

注:indexOf和lastIndexOf方法如果未找到都是返回-1;
在只存在唯一的一个str或不存在str的情况下indexOf(str)和lastIndexOf(str)返回值相同。

  • String replace(char oldChar, char newChar):返回一个新的字符串,它是通过用 newChar 替换此字符串中出现的所有 oldChar 得到的。
  • String replace(CharSequence target, CharSequence replacement):使用指定的字面值替换序列替换此字符串所有匹配字面值目标序列的子字符串。
  • String replaceAll(String regex, String replacement) :使用给定的replacement替换此字符串所有匹配给定的正则表达式的子字符串。
  • String replaceFirst(String regex, String replacement) :使用给定的replacement 替换此字符串匹配给定的正则表达式的第一个子字符串。

替换:

  • String replace(char oldChar, char newChar):返回一个新的字符串,它是通过用 newChar 替换此字符串中出现的所有 oldChar 得到的。
  • String replace(CharSequence target, CharSequence replacement):使用指定的字面值替换序列替换此字符串所有匹配字面值目标序列的子字符串。
  • String replaceAll(String regex, String replacement):使用给定的 replacement 替换此字符串所有匹配给定的正则表达式的子字符串。
  • String replaceFirst(String regex, String replacement):使用给定的 replacement 替换此字符串匹配给定的正则表达式的第一个子字符串。

匹配:

  • boolean matches(String regex):告知此字符串是否匹配给定的正则表达式。

切片:

  • String[] split(String regex):根据给定正则表达式的匹配拆分此字符串。
  • String[] split(String regex, int limit):根据匹配给定的正则表达式来拆分此字符串,最多不超过limit个,如果超过了,剩下的全部都放到最后一个元素中。

关于String的类型转换

String与基本数据类型转换(复习)

字符串 ---> 基本数据类型、包装类

调用包装类的静态方法:parseXxx(str)

以int类型为例:
Integer包装类的public static int parseInt(String s):可以将由“数字”字符组成的字符串转换为整型。

String str1 = "123";
int num = Integer.parseInt(str1);
基本数据类型、包装类 ---> 字符串
  1. 调用String重载的valueOf(xxx)
  2. 与“”连接

以int类型为例:
调用String类的public String valueOf(int n)可将int型转换为字符串

String str2 = String.valueOf(num)
String str3 = num + "";

String与字符数组转换

字符串 ---> 字符数组
  1. public char[] toCharArray():将字符串中的全部字符存放在一个字符数组中的方法。
  2. public void getChars(int srcBegin, int srcEnd, char[] dst, int dstBegin):提供了将指定索引范围内的字符串存放到数组中的方法。

调用String的toCharArray()

char[] arr1 = str1.toCharArray();//['1','2','3']

调用String的getChars(int srcBegin, int srcEnd, char[] dst, int dstBegin)

  • srcBegin – 要复制的字符串中第一个字符的索引。
  • srcEnd – 要复制的字符串中最后一个字符后的索引。
  • dest – 目标字符数组,其中String中的字符被复制。
  • destBegin – 数组中的索引,从将被推入数组的字符开始
String a = "123456";
char[] b = new char[5];
b[0] = '0';
b[1] = '5';
a.getChars(1, 5, b, 1);
System.out.println(b);//02345
字符数组 ---> 字符串

调用String的构造器

  • String(char[]) 和 String(char[],int offset,int length) 分别用字符数组中的全部字符和部分字符创建字符串对象。
    • offset是起点,length是选取元素个数
String str2 = new String(arr2);//"123"
String str3 = new String(arr2,1,1)//"2"

String与字节数组转换

字符串 ---> 字节数组(编码)
  1. public byte[] getBytes() :使用平台的默认字符集将此 String 编码为byte 序列,并将结果存储到一个新的 byte 数组中。
  2. public byte[] getBytes(String charsetName) :使用指定的字符集将此 String 编码到 byte 序列,并将结果存储到新的 byte 数组。

调用String的getBytes()

byte[] bytes = str1.getBytes();//使用默认的字符集,进行编码。
byte[] gbks = str1.getBytes("gbk");//使用gbk字符集进行编码。
字节数组 ---> 字符串(解码)

调用String的构造器

  1. String(byte[]):通过使用平台的默认字符集解码指定的 byte 数组,构造一个新的 String。
  2. String(byte[],int offset,int length) :用指定的字节数组的一部分,即从数组起始位置offset开始取length个字节构造一个字符串对象。
String str2 = new String(bytes);//使用默认的字符集,进行解码。
String str3 = new String(gbks, "gbk");//使用gbk字符集进行解码。

StringBuffer类和StringBuilder类

StringBuffer类

StringBuffer类不同于String,其对象必须使用构造器生成。
有三个构造器:

  • StringBuffer():初始容量为16的字符串缓冲区
  • StringBuffer(int size):构造指定容量的字符串缓冲区
  • StringBuffer(String str):将内容初始化为指定字符串内容

源码分析:

StringBuffer sb1 = new StringBuffer();//char[] value = new char[16];底层创建了一个长度是16的数组。
StringBuffer sb2 = new StringBuffer("abc");//char[] value = new char["abc".length() + 16];
问题1. System.out.println(sb2.length());//3

问题2.

  • 扩容问题:如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组。
    • 默认情况下,扩容为原来容量的2倍 + 2,同时将原有数组中的元素复制到新的数组中。

StringBuffer类的常用方法

  • StringBuffer append(xxx):提供了很多的append()方法,用于进行字符串拼接
  • StringBuffer delete(int start,int end):删除指定位置的内容
  • StringBuffer replace(int start, int end, String str):把[start,end)位置替换为str
  • StringBuffer insert(int offset, xxx):在指定位置插入xxx
  • StringBuffer reverse() :把当前字符序列逆转
  • public int indexOf(String str):返回指定子字符串在此字符串中第一次出现处的索引
  • public String substring(int start,int end):返回一个从start开始到end索引结束的左闭右开区间的子字符串
  • public int length()
  • public char charAt(int n ):返回某索引处的字符
  • public void setCharAt(int n ,char ch):把第index位的字符替换成ch

重点:
增:append(xxx)
删:delete(int start,int end)
改:setCharAt(int n ,char ch) / replace(int start, int end, String str)
查:charAt(int n )
插:insert(int offset, xxx)
长度:length();
* 遍历:for() + charAt() / toString()

StringBuilder类

StringBuilder 和 StringBuffer 非常类似,均代表可变的字符序列,而且提供相关功能的方法也一样

String、StringBuffer、StringBuilder三者的异同? 
  • String:不可变的字符序列;底层使用char[]存储
  • StringBuffer:可变的字符序列;线程安全的,效率低;底层使用char[]存储
  • StringBuilder:可变的字符序列;jdk5.0新增的,线程不安全的,效率高;底层使用char[]存储

对比String、StringBuffer、StringBuilder三者的效率:StringBuilder > StringBuffer > String

String str = null;
System.out.println(str);//"null"
StringBuffer sb = new StringBuffer();
sb.append(str);
System.out.println(sb.length());//4
System.out.println(sb);//"null"
StringBuffer sb1 = new StringBuffer(str);//NullPointerException
System.out.println(sb1);
与第五章toString()方法的使用相似,append和print中都有保护机制 

ae36808f7e029517010ad0d6094a69e2.png 272fddcd1a5cafded3a575608f9efa05.png
改变形式:

String str ="";
System.out.println(str);//无显示输出
StringBuffer sb = new StringBuffer();
sb.append(str);
System.out.println(sb.length());//0
System.out.println(sb);//
StringBuffer sb1 = new StringBuffer(str);
System.out.println(sb1);//

关于两种形式中System.out.println(sb1);的结果不相同:

  1. string str =null 是不给他分配内存空间,而 string str ="" 给它分配长度为空字符串的内存空间。
  2. string str =null 没有string对象,string str=""有一个字符串对象。

String与StringBuffer、StringBuilder之间的转换

String --> StringBuffer、StringBuilder

调用StringBuffer、StringBuilder的构造器

StringBuffer、StringBuilder ---> Srting
  1. 调用String的构造器
  2. StringBuffer、StringBuilder的toString()

二、JDK8之前日期时间API

java.lang.System类

System类提供的public static long currentTimeMillis()用来返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差。

此方法适于计算时间差。

@Test
public void test1(){
    long time = System.currentTimeMillis();
    //返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差。
    //称为时间戳
    System.out.println(time);
}

java.util.Date类

java.util.Date类
         |---java.sql.Date类

1.两个构造器的使用

import java.util.Date;

构造器一:Date():创建一个对应当前时间的Date对象

//构造器一:Date():创建一个对应当前时间的Date对象
Date date1 = new Date();
System.out.println(date1.toString());//Fri Sep 09 09:43:12 CST 2022
System.out.println(date1.getTime());//1662687885495

构造器二:创建指定毫秒数的Date对象

Date date2 = new Date(1662687885495L);
System.out.println(date2.toString());

2.两个方法的使用

toString():显示当前的年、月、日、时、分、秒
getTime():获取当前Date对象对应的毫秒数。(时间戳)

java.sql.Date对应着数据库中的日期类型的变量

  1. 如何实例化
java.sql.Date date3 = new java.sql.Date(35235325345L);
System.out.println(date3);//1971-02-13
  1. 如何将java.util.Date对象转换为java.sql.Date对象
//情况一:
Date date4 = new java.sql.Date(2343243242323L);
java.sql.Date date5 = (java.sql.Date) date4;
//情况二:
Date date6 = new Date();
java.sql.Date date7 = new java.sql.Date(date6.getTime());

java.text.SimpleDateFormat类

SimpleDateFormat的使用:SimpleDateFormat对日期Date类的格式化和解析

SimpleDateFormat的实例化

SimpleDateFormat sdf = new SimpleDateFormat();

格式化:日期 --->字符串

Date date = new Date();
System.out.println(date);//Fri Sep 09 10:16:25 CST 2022
String format = sdf.format(date);
System.out.println(format);//22-9-9 上午10:16

解析:格式化的逆过程,字符串 ---> 日期

String str = "19-12-18 上午11:43";
Date date1 = sdf.parse(str);
System.out.println(date1);//Wed Dec 18 11:43:00 CST 2019

按照指定的方式格式化和解析:调用带参的构造器

SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
//格式化
String format1 = sdf1.format(date);
System.out.println(format1);//2022-09-09 10:16:25
//解析
Date date2 = sdf1.parse("2020-02-18 11:48:27");

注意:按照指定的方式格式化和解析要求字符串必须是符合SimpleDateFormat识别的格式(通过构造器参数体现),否则,抛异常

SimpleDateFormat构造器中参数设置类型:

686e60766d35b700a69a479b74ebd5ac.png

练习:将字符串"2020-09-08"转换为java.sql.Date

@Test
public void testExer() throws ParseException {
    String birth = "2020-09-08";
    SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd");
    Date date = sdf1.parse(birth);
//        System.out.println(date);
    java.sql.Date birthDate = new java.sql.Date(date.getTime());
    System.out.println(birthDate);
}

java.util.Calendar(日历)类

Calendar是一个抽象基类,主要用于完成日期字段之间相互操作的功能。

1. 实例化

//方式一:创建其子类(GregorianCalendar)的对象(不常用)
//方式二:调用其静态方法getInstance()
Calendar calendar = Calendar.getInstance();
System.out.println(calendar.getClass());//class java.util.GregorianCalendar
//其实getInstance()方法中也是创建了GregorianCalendar的对象

2. 常用方法

  1. public int get(int field)

一个Calendar的实例是系统时间的抽象表示,通过get(int field)方法来取得想要的时间信息。比如YEAR、MONTH、DAY_OF_WEEK、HOUR_OF_DAY 、MINUTE、SECOND

int days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);//9
System.out.println(calendar.get(Calendar.DAY_OF_YEAR));//252
  1. public void set(int field,int value)
//calendar可变性
calendar.set(Calendar.DAY_OF_MONTH,22);
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);//22
  1. public void add(int field,int amount)
calendar.add(Calendar.DAY_OF_MONTH,-3);
System.out.println(calendar.get(Calendar.DAY_OF_MONTH));//19
  1. public final Date getTime()
//日历类---> Date
System.out.println(calendar.getTime());//Mon Sep 19 11:10:24 CST 2022
  1. public final void setTime(Date date)
//Date ---> 日历类
Date date = new Date();
calendar.setTime(date);
System.out.println(calendar.get(Calendar.DAY_OF_MONTH));//9

注意

  1. 获取月份时:一月是0,二月是1,以此类推,12月是11
  2. 获取星期时:周日是1,周二是2 , 。。。。周六是7

总结:jdk 8之前的日期时间的API测试

  1. System类中currentTimeMillis();
  2. java.util.Date和子类java.sql.Date;
  3. SimpleDateFormat;
  4. Calendar;

三、JDK8中新日期时间API

背景:JDK 1.0中包含了一个java.util.Date类,但是它的大多数方法已经在JDK 1.1引入Calendar类之后被弃用
了。而Calendar并不比Date好多少。它们面临的问题是:

  • 可变性:像日期和时间这样的类应该是不可变的。
  • 偏移性:Date中的年份是从1900开始的,而月份都从0开始。
//偏移量
System.out.println(new Date(2020,9,8));//Fri Oct 08 00:00:00 CST 3920
System.out.println(new Date(2020 - 1900,9 - 1,8));//Tue Sep 08 00:00:00 CST 2020
  • 格式化:格式化只对Date有用,Calendar则不行。
  • 此外,它们也不是线程安全的;不能处理闰秒等。

新时间日期API:
java.time – 包含值对象的基础包(常用)
java.time.chrono – 提供对不同的日历系统的访问
java.time.format – 格式化和解析时间和日期(常用)
java.time.temporal – 包括底层框架和扩展特性
java.time.zone – 包含时区支持的类

java.time 中包含了所有关于本地日期(LocalDate)、本地时间(LocalTime)、本地日期时间(LocalDateTime)、时区(ZonedDateTime)和持续时间(Duration)的类

LocalDate、LocalTime、LocalDateTime 的使用

说明:

  1. LocalDateTime相较于LocalDate、LocalTime,使用频率要高
  2. 类似于Calendar
141d260580a75bba4d24294a0d022529.png

now():获取当前的日期、时间、日期+时间

LocalDate localDate = LocalDate.now();
System.out.println(localDate);//2022-09-09
LocalTime localTime = LocalTime.now();
System.out.println(localTime);//14:25:20.552
LocalDateTime localDateTime = LocalDateTime.now();
System.out.println(localDateTime);//2022-09-09T14:25:20.552

of():设置指定的年、月、日、时、分、秒。没有偏移量

LocalDateTime localDateTime1 = LocalDateTime.of(2020, 10, 6, 13, 23, 43);
System.out.println(localDateTime1);//2020-10-06T13:23:43

getXxx():获取相关的属性

System.out.println(localDateTime.getDayOfMonth());

不可变性的体现

withXxx():设置相关的属性

LocalDate localDate1 = localDate.withDayOfMonth(22);
System.out.println(localDate);//2022-09-09
System.out.println(localDate1);//2022-09-22

plusXxx():相关属性的增加

LocalDateTime localDateTime3 = localDateTime.plusMonths(3);
System.out.println(localDateTime);//2022-09-09T14:25:20.552
System.out.println(localDateTime3);//2022-12-09T14:25:20.552

minusXxx():相关属性的减少

LocalDateTime localDateTime4 = localDateTime.minusDays(6);
System.out.println(localDateTime);//2022-09-09T14:25:20.552
System.out.println(localDateTime4);//2022-09-03T14:25:20.552

瞬时:Instant

java.time包通过值类型Instant提供机器视图,不提供处理人类意义上的时间单位。类似于 java.util.Date类。

0e8be93e5bb2020ee71301abfa97692e.png

now():获取本初子午线对应的标准时间

Instant instant = Instant.now();
System.out.println(instant);//2022-09-09T06:46:09.546Z

atOffset():添加时间的偏移量

OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
System.out.println(offsetDateTime);//2022-09-09T14:46:09.546+08:00

toEpochMilli():获取自1970年1月1日0时0分0秒(UTC)开始的毫秒数 ---> Date类的getTime()

long milli = instant.toEpochMilli();
System.out.println(milli);//1662705969546

ofEpochMilli():通过给定的毫秒数,获取Instant实例 -->Date(long millis)

Instant instant1 = Instant.ofEpochMilli(1550475314878L);
System.out.println(instant1);//2019-02-18T07:35:14.878Z

DateTimeFormatter类

格式化或解析日期、时间,类似于SimpleDateFormat

29ca49b1526d2fae34755c1f0cbfa568.png

java.time.format.DateTimeFormatter 类:该类提供了三种格式化方法

方式一:预定义的标准格式。如:ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIME

DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE_TIME;
//格式化:日期-->字符串
LocalDateTime localDateTime = LocalDateTime.now();
String str1 = formatter.format(localDateTime);
System.out.println(localDateTime);//2022-09-09T20:05:33.687
System.out.println(str1);//2022-09-09T20:05:33.687
//解析:字符串 -->日期
TemporalAccessor parse = formatter.parse("2019-02-18T15:42:18.797");
System.out.println(parse);//{},ISO resolved to 2019-02-18T15:42:18.797

方式二:

  • 本地化相关的格式。如:ofLocalizedDateTime()
    FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT :适用于LocalDateTime
DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG);
System.out.println(formatter1.format(localDateTime));//2022年9月9日 下午08时13分44秒
  • 本地化相关的格式。如:ofLocalizedDate()
    FormatStyle.FULL / FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT : 适用于LocalDate
DateTimeFormatter formatter2 = DateTimeFormatter.ofLocalizedDate(FormatStyle.MEDIUM);
System.out.println(formatter2.format(LocalDate.now()));//2022-9-9

重点: 方式三:自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)

DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");
System.out.println(formatter3.format(LocalDateTime.now()));//2022-09-09 08:13:44
System.out.println(formatter3.parse("2019-02-18 03:52:09"));
//{SecondOfMinute=9, MicroOfSecond=0, MinuteOfHour=52, HourOfAmPm=3, NanoOfSecond=0, MilliOfSecond=0},ISO resolved to 2019-02-18

补充

其它API

2db1042a6fe2401d14b143d1123f730e.png

日期处理的转换

0d20c143c5ee3db28977a8a12aca96b1.png

四、Java比较器

Java实现对象排序的方式有两种:

  1. 自然排序:java.lang.Comparable
  2. 定制排序:java.util.Comparator

Comparable接口

  1. 像String、包装类等实现了Comparable接口,重写了compareTo(obj)方法,给出了比较两个对象大小的方式。

  2. 像String、包装类重写compareTo()方法以后,进行了从小到大的排列

    1. 重写compareTo(obj)的规则:
    • 如果当前对象this大于形参对象obj,则返回正整数,
    • 如果当前对象this小于形参对象obj,则返回负整数,
    • 如果当前对象this等于形参对象obj,则返回零。
  3. 对于自定义类来说,如果需要排序,我们可以让自定义类实现Comparable接口,重写compareTo(obj)方法。

    • 在compareTo(obj)方法中指明如何排序

举例:

class A implements Comparable {
    private int price;
    private String name;
    public A(int price, String name) {
        this.price = price;
        this.name = name;
    }
    public String toString() {
        return "A{" +
                "price=" + price +
                ", name='" + name + '\'' +
                '}';
    }
    public int compareTo(Object o) {
        if (o instanceof A) {
            A a = (A) o;
            if (this.price > a.price) {
                return 1;
            }else if(this.price==a.price){
                return this.name.compareTo(a.name);//价格相同比较名字
            }else
                return -1;
        } else
            throw new RuntimeException("不属于商品类型");
    }
}
@Test
public void Test() {
    A[] a=new A[3];
    a[0]=new A(12,"d");
    a[1]=new A(25,"c");
    a[2]=new A(12,"a");
    Arrays.sort(a);
    System.out.println(Arrays.toString(a));
    //[A{price=12, name='a'}, A{price=12, name='d'}, A{price=25, name='c'}]
}

对于Arrays.sort()和Arrays.toString的理解:

  1. sort():在方法排序的内部调用了该对象的类中的compareTo()方法,比较当前项与后一项的大小,当返回值为1时,使用swap()调换两个元素
  2. 在类的compareTo()方法中,可以进行嵌套比较,例如相同价格比较名字
  3. toString():调用了该对象的类中的toString()方法,如果没有重写,返回地址值

Comparator接口

背景:当元素的类型没有实现java.lang.Comparable接口而又不方便修改代码,或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,那么可以考虑使用 Comparator 的对象来排序。

重写compare(Object o1,Object o2)方法,比较o1和o2的大小:

  • 如果方法返回正整数,则表示o1大于o2;
  • 如果返回0,表示相等;
  • 返回负整数,表示o1小于o2。
@Test
public void Test1() {
    A[] a = new A[3];
    a[0] = new A(12, "d");
    a[1] = new A(25, "c");
    a[2] = new A(12, "a");
    Arrays.sort(a, new Comparator() {
        public int compare(Object o1, Object o2) {//按照价格从高到低排序
            if(o1 instanceof A && o2 instanceof A){
                A a= (A) o1;
                A b= (A) o2;
                if(a.getPrice()>b.getPrice()){
                    return -1;
                }else if(a.getPrice()==b.getPrice()){//按照名字从高到低排序
                    return -a.getName().compareTo(b.getName());
                }else
                    return 1;
            }
            throw new RuntimeException("不属于商品类型");
        }
    });
    System.out.println(Arrays.toString(a));
    //[A{price=25, name='c'}, A{price=12, name='d'}, A{price=12, name='a'}]
}

五、System类

System类代表系统,系统级的很多属性和控制方法都放置在该类的内部。该类位于java.lang包

该类的构造器是private的
其内部的成员变量和成员方法都是static的

成员变量

  • System类内部包含in、out和err三个成员变量,分别代表标准输入流(键盘输入),标准输出流(显示器)和标准错误输出流(显示器)。

成员方法

  1. native long currentTimeMillis():
    • 该方法的作用是返回当前的计算机时间,时间的表达格式为当前计算机时间和GMT时间(格林威治时间)1970年1月1号0时0分0秒所差的毫秒数。
  2. void exit(int status):
    • 该方法的作用是退出程序。其中status的值为0代表正常退出,非零代表异常退出。使用该方法可以在图形界面编程中实现程序的退出功能等。
  3. void gc():
    • 该方法的作用是请求系统进行垃圾回收。至于系统是否立刻回收,则取决于系统中垃圾回收算法的实现以及系统执行时的情况。
  4. String getProperty(String key):
    • 该方法的作用是获得系统中属性名为key的属性对应的值。系统中常见的属性名以及属性的作用如下表所示:
21cef4448f63c551d63e79f04e50667b.png
String osName = System.getProperty("os.name");
System.out.println("os的name:" + osName);//os的name:Windows 10

六、Math类

d28e804f5947af6ba8feb5a15633240a.png

七、BigInteger类与BigDecimal类

BigInteger类

java.math包的BigInteger可以表示不可变的任意精度的整数。BigInteger 提供所有 Java 的基本整数操作符的对应物,并提供 java.lang.Math 的所有相关方法。另外,BigInteger 还提供以下运算:模算术、GCD 计算、质数测试、素数生成、位操作以及一些其他操作。

构造器:

BigInteger(String val):根据字符串构建BigInteger对象
4790a25fcc4eae9c9f1e040032821125.png

BigDecimal类

一般的Float类和Double类可以用来做科学计算或工程计算,但在商业计算中,
要求数字精度比较高,故用到java.math.BigDecimal类。

  • BigDecimal类支持不可变的、任意精度的有符号十进制定点数。

构造器:

public BigDecimal(double val)
public BigDecimal(String val)

常用方法

  1. public BigDecimal add(BigDecimal augend)
  2. public BigDecimal subtract(BigDecimal subtrahend)
  3. public BigDecimal multiply(BigDecimal multiplicand)
  4. public BigDecimal divide(BigDecimal divisor, int scale, int roundingMode)
BigDecimal bd = new BigDecimal("12435.351");
BigDecimal bd2 = new BigDecimal("11");
System.out.println(bi);
//System.out.println(bd.divide(bd2));//没有指定小数规格
System.out.println(bd.divide(bd2, BigDecimal.ROUND_HALF_UP));//1130.486
System.out.println(bd.divide(bd2, 5, BigDecimal.ROUND_HALF_UP));//1130.48645

BigDecimal.ROUND_HALF_UP表示达到指定位数后的第一个数四舍五入

posted @ 2023-09-30 16:16  LemonPuer  阅读(19)  评论(0)    收藏  举报