Fork me on GitHub

Java_day6

自用Java笔记(Ⅵ),主要记录Java常用类!奋斗ing

字符串相关的类

String

字符串是常量,用双引号引起来,他们的值在创建后就不能改变。

String对象的 字符串内容 是 存储在一个 字符数组final char[] value中。

特点

  • 实现了 Serializable 接口,表示字符串支持序列化的;

  • 实现了 Comparable 接口,表示String可以比较大小

  • String是一个final类,不可被继承,其代表不可变的字符序列。(不可变性

    体现:

    1. 当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值;
    2. 当对现有的字符串进行拼接时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值;
    3. 当调用String的replace()修改字符串,也需要重新指定内存区域赋值。
1
2
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence

通过字面量的方式(区别于new方式)给一个字符串赋值,此时的 字符串值 声明在字符串常量池中,字符串常量池 不会重复储存相同的 字符串

实例化的方式

  1. 通过字面量定义
  2. 通过new + 构造器

面试题:

1
2
3
4
5
String s = new String("abc");//此方式创建对象,在内存中创建了几个对象? 俩:一个堆空间中new结构,另一个是char[]对应的常量池中的数据“abc”
String s1 = new String("abc");
String s2 = new String("abc");
System.out.println(s1 == s2); //false
// 先造了对象在堆中,对象的value指向常量池
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public class StringTest {
@Test
public void testString(){
String s1 = "hello";
String s2 = "goodwell";

String s3 = "hellogoodwell";
String s4 = "hello" + "goodwell";
String s5 = s1 + "goodwell";
String s6 = "hello" + s2;
String s7 = s1 + s2;
//1.
System.out.println(s3 == s4);//true
//2.
System.out.println(s3 == s5);//false
System.out.println(s3 == s6);//false
System.out.println(s3 == s7);//false
System.out.println(s5 == s6);//false
System.out.println(s6 == s7);//false

String s8 = s5.intern();
//返回值得到的s8使用的常量值已经存在的“hellogoodwell”
//3.
System.out.println(s3 == s8);//true

final String s9 = "hello";
String s10 = s9 + "goodwell";
//4.
System.out.println(s3 == s10);//true
}

结论:

  • 常量与常量的 拼接结果 在常量池。且常量池中 不会存在相同内容的常量。
  • 只要拼接的其中有一个是变量,结果就在堆中(类似new)
  • 如果拼接的结果 调用intern()方法,返回值就在常量池中
  • final String(也在常量池中)和字面量连接,结果在常量池中
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class StringTest {
String str = new String("good");
char[] ch = {'t','e','s','t'};

public void change(String str, char ch[]) {
str = "test ok";
ch[0] = 'b';
}

public static void main(String[] args) {
StringTest ex = new StringTest();
ex.change(ex.str, ex.ch);
System.out.println(ex.str);//good
//不可变性
System.out.println(ex.ch);//best
}
}

常用方法

  • 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 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
  • boolean endsWith(String suffix): :测试此字符串是否以指定的后缀结束
  • boolean startsWith(String prefix): :测试此字符串是否以指定的前缀开始
  • boolean startsWith(String prefix, int toffset): :测试此字符串从指定索引开始的子字符串是否以指定前缀开始
  • 替换
  • 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 与 char[] 之间的转换:

  • String –> char[] 调用String的toCharArray()

    1
    char[] charArray = str1.toCharArray();
  • char[] –> String 调用String的构造器

    1
    String str2 = new String(arr);

String 与 byte[] 之间的转换:

  • String –> byte[] 调用String的getBytes()

    1
    byte[] bytes = str1.getBytes(); //使用默认的字符集进行编码
  • byte[] –> String 调用String的构造器

    1
    2
    String str2 = new String(bytes);//使用默认的字符集进行解码
    //说明:解码时,要求解码使用的字符集必须和编码时使用的字符集一致,否则出现乱码

StringBuffer类

java.lang.StringBuffer代表 可变的字符序列,JDK1.0中声明,可以对字符串内容进行增删,此时不会产生新的对象。很多方法与String相同。作为参数传递时,方法内部可以改变值。

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

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

StringBuffer 类的常用方法

  • StringBuffer append(xxx):提供了很多的append()方法,用于进行字符串拼接(啥都变成字符串,例如“null”)
  • 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)
  • public int length()
  • public char charAt(int n )
  • public void setCharAt(int n ,char ch)

## StringBuilder类

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

对比String 、StringBuffer 、StringBuilder

  • String(JDK1.0):不可变字符序列

  • StringBuffer(JDK1.0):可变字符序列、效率低、线程安全

  • StringBuilder(JDK 5.0):可变字符序列、效率高、线程不安全,底层都是使用char[] 存储

    注意:作为参数传递的话,方法内部String不会改变其值,StringBuffer和StringBuilder会改变其值。

1
2
3
4
5
6
7
8
String str = null;
//System.out.println(str.length());//NullPointerException
StringBuffer sb = new StringBuffer();
sb.append(str);
System.out.println(sb.length());//4
System.out.println(sb);//"null"
//StringBuffer sb1 = new StringBuffer(str);
//System.out.println(sb1);//NullPointerException

源码分析

1
2
3
4
5
6
7
8
9
10
11
12
13
String str = new String();// char[] value = new char[0];
String str1 = new String("abn");// char[] value = new char[]{'a','b','n'};

StringBuffer sb1 = new StringBuffer();// char[] value = new char[16];
sb1.append('a');// value[0] = 'a';
sb1.append('b');// value[1] = 'b';

StringBuffer sb2 = new StringBuffer("abc");// char[] value = new char["abc".length + 16]{'a','b','c's};

//问题一:
System.out.println(sb2.length());// 3
//问题二:
//扩容问题:若添加的数据底层数组装不下,那就需要扩容底层的数组;默认情况下,扩容为原来容量的2倍 + 2,同时将原有数组中的元素复制到新的数组中。

指导建议:

​ 开发中使用,StringBuffer(int capacity) 或 StringBuilder(int capacity) 指定容量的

对比三者效率:

​ StringBuilder > StringBuffer > String

总结:

  • 增: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()

时间相关的类

JDk8之前的日期和时间的API

①. java.lang.System类

System.currentTimeMillis(): 返回当前时间与1970年1月1日0时0分0秒之间 以毫秒为单位的时间差,也称为时间戳

②. java.util.Date (java.sql.Date继承前者)

构造器:

  • Date() 创建一个对应当前时间的Date对象
  • //Date(int year, int month, int day) 创建一个对应时间的Date对象
  • Date(long) 创建指定毫秒数的Date对象

方法:

  • tiString() 显示当前年月日
  • getTime() 获取当前Date对象对应的时间戳

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

  • 将java.util.Date对象转换成java.sql.Date对象

    1
    java.sql.Date dateSql = new java.sql.Date(dateUtil.getTime());
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
public class TimeTest {
//1.System类中的currentTimeMillis()
@Test
public void test11(){
long time = System.currentTimeMillis();
System.out.println(time);
//1569513273409返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差

}
//2.java.util.Date类(java.sql.Date类)
// ①两个构造器的使用
// > Date() 创建一个当前时间的Date对象
// > Date(long) 创建指定毫秒数的Date对象
// ②两个方法的使用
// > toString() 显示当前的年、月、日、时、分、秒
// > getTime() 获取当前Date对象对应的毫秒数(时间戳)
// ③java.sql.Date类对应着数据库中的日期类型的变量
// > 如何实例化
// > util.Date --getTime()-> sql.Date对象
// >< sql.Date --> util.Date对象(不需要转,学生本身就是人)
@Test
public void test22(){
//构造器一:Date() 创建一个当前时间的Date对象
Date date1 = new Date();
System.out.println(date1.toString());//Fri Sep 27 00:50:46 CST 2019
System.out.println(date1.getTime());//1569516806737
//构造器二:Date(long) 创建指定毫秒数的Date对象
Date date2 = new Date(1569516806737L);
System.out.println(date2.toString());//Fri Sep 27 00:53:26 CST 2019
//创建java.sql.Date对象
java.sql.Date date3 = new java.sql.Date(1569516806737L);
System.out.println(date3);//2019-09-27有无.toString()一样
//sql.Date --> util.Date对象
Date date8 = date3;
System.out.println(date8.toString());
//util.Date --getTime()-> sql.Date对象
//情况1
Date date4 = new java.sql.Date(1569516806737L);
java.sql.Date date5 = (java.sql.Date) date4;
//情况2
Date date6 = new Date();
java.sql.Date date7 = new java.sql.Date(date6.getTime());
}
}

③. SimpleDateFormat类

java.text.SimpleDateFormat() 不与语言环境有关的方式来对Date类的格式化和解析的具体类

构造器:

  • SimpleDateFormat() 默认的模式和语言环境创建对象
  • SimpleDateFormat(String Pattern) 用参数pattern指定的格式创建一个对象。

该对象可以

  • 格式化: 日期 -> 文本 String format(Date date)
  • 解析:文本 -> 日期 Date parse(String source)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
public class DateTimeTesr {
/*
SimpleDateFormat的使用:对日期Date类的格式化和解析
1.两个操作
- 格式化: 日期 ---> 字符串
- 解析: 字符串 ---> 日期
2.SimpleDateFormat的实例化
*/
@Test
public void testSimpleDateFormat(){
//实例化SimpleDateFormat:使用默认构造器
SimpleDateFormat sdf = new SimpleDateFormat();

//格式化: 日期 ---> 字符串
Date date = new Date();
System.out.println(date);

String format = sdf.format(date);
System.out.println(format);

//解析: 字符串 ---> 日期
String str = "19-9-27 下午8:12";
Date date1 = null;
try {
date1 = sdf.parse(str);
} catch (ParseException e) {
e.printStackTrace();
}
System.out.println(date1);

// 按照指定的方式格式化和解析,调用带参数的构造器
// SimpleDateFormat sdf1 = new SimpleDateFormat("yyyyy.MMMMM.dd GGG hh:mm aaa");
SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");
//格式化
String format1 = sdf1.format(date);
System.out.println(format1);//2019-09-27 08:22:08
//解析
Date date2 = null;
try {
date2 = sdf1.parse("2019-09-27 08:22:08");
} catch (ParseException e) {
e.printStackTrace();
}
System.out.println(date2);
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public void test12() throws ParseException {
/*
练习一: 字符串“2020-09-08”抓换成java.sql.Date

练习二: “三天打鱼两天晒网” 1990-01-01 xxxx-xx-xx 打鱼?晒网?
*/
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);
//思路:总天数%5==?
//方式一:( date2.getTime() - date1.getTime() ) / (1000 * 60 * 60 * 24) + 1
//方式二:时间分段整数年加闰年
}

④. Calendar类

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

1.实例化
方式一:创建其子类(GregorianCalendar)的对象
方式二:调用其静态方法getInstance()

2.常用方法

  • get()
  • set() 可变性
  • add()
  • getTime() 日历类 –> Date
  • setTime() Date –> 日历类

注意:

  • 获取月份时:一月是0 … 十二月是11
  • 获取星期时:周日是1 … 周六是7
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
/*
Calendar类(抽象类)的使用
*/
@Test
public void testCalendar(){
// 1.实例化
// 方式一:创建其子类(GregorianCalendar)的对象
// 方式二:调用其静态方法getInstance()
Calendar calendar = Calendar.getInstance();
System.out.println(calendar.getClass());

// 2.常用方法
// get()
int days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);//28
System.out.println(calendar.get(Calendar.DAY_OF_YEAR));//271

// set() 可变性
calendar.set(Calendar.DAY_OF_MONTH,22);
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);//22

// add()
calendar.add(Calendar.DAY_OF_MONTH,22);
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);//14

// getTime() 日历类 --> Date
Date date = calendar.getTime();
System.out.println(date);

// setTime() Date --> 日历类
Date date1 = new Date();
calendar.setTime(date1);
days = calendar.get(Calendar.DAY_OF_MONTH);
System.out.println(days);//28
}

JDK 8中新日期时间API

java.time

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
/*
LocalDate LocalTime LocalDateTime 的使用
类似于Calendar()
*/
@Test
public void testTime(){
//1. now() 获取当前时间日期
LocalDate localDate = LocalDate.now();
LocalTime localTime = LocalTime.now();
LocalDateTime localDateTime = LocalDateTime.now();

System.out.println(localDate);//2019-09-28
System.out.println(localTime);//09:19:28.307
System.out.println(localDateTime);//2019-09-28T09:19:28.307

//2. of() 设置指定的年月日时分秒,没有偏移量
LocalDateTime localDateTime1 = LocalDateTime.of(2020, 10, 10, 0, 0, 0);
System.out.println(localDateTime1);//2020-10-10T00:00

//getXxx() 获取相关属性
System.out.println(localDateTime.getDayOfMonth());//28
System.out.println(localDateTime.getDayOfWeek());//SATURDAY
System.out.println(localDateTime.getDayOfYear());//271
System.out.println(localDateTime.getHour());//9
System.out.println(localDateTime.getMonthValue());//9

//withXxx() 修改、设置 不同于Calendar() 体现不可变性
LocalDate localDate1 = localDate.withDayOfMonth(22);
System.out.println(localDate);//2019-09-28
System.out.println(localDate1);//2019-09-22

//plusXxx() 增加
LocalDateTime localDateTime2 = localDateTime.plusMonths(4);
System.out.println(localDateTime);//2019-09-28T09:29:43.638
System.out.println(localDateTime2);//2020-01-28T09:29:43.638

//minusXxx() 减去
LocalDateTime localDateTime3 = localDateTime.minusDays(3);
System.out.println(localDateTime);//2019-09-28T09:32:06.256
System.out.println(localDateTime3);//2019-09-25T09:32:06.256
}

Instant

瞬时

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
/*
Instant 的使用
类似于Date
*/
@Test
public void testInstant(){
//now() 获取本初子午线对应的标准时间
Instant instant = Instant.now();
System.out.println(instant); //2019-09-28T01:42:23.739Z

//添加时间的偏移量
OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
System.out.println(offsetDateTime); //2019-09-28T09:42:23.739+08:00

//toEpochMilli() 获取自1970年1月1日0时0分0秒(UTC)开始的毫秒数 --> Date类的getTime()
long milli = instant.toEpochMilli();
System.out.println(milli);//1569638022547

//ofEpochMilli() 通过给定的毫秒数,获取Instant实例 --> Date(long millis)
Instant instant1 = Instant.ofEpochMilli(11111111L);
System.out.println(instant1);//1970-01-01T03:05:11.111Z
}

java.time.format.DateTimeFormatter

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
/*
DateTimeFormatter 格式化或解析日期、时间
类似于SimpleDateFormat
*/
@Test
public void test3(){
//方式1:预定义的标准格式。如:ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIME
DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE_TIME;
//格式化: 日期 --> 字符串
LocalDateTime localDateTime = LocalDateTime.now();
String str = formatter.format(localDateTime);
System.out.println(localDateTime);//2019-09-28T14:49:57.682
System.out.println(str);//2019-09-28T14:49:57.682
//解析: 字符串 --> 日期
TemporalAccessor parse = formatter.parse("2019-09-28T14:49:57.682");
System.out.println(parse);//{},ISO resolved to 2019-09-28T14:49:57.682

//方式2:本地化相关的格式。如:ofLocalizedDateTime(FormatStyle.LONG)
//FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT ↑
DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG);
//格式化
String str1 = formatter1.format(localDateTime);
System.out.println(localDateTime);//2019-09-28T15:33:41.802
System.out.println(str1);//2019年9月28日 下午03时33分41秒
//ofLocalizedDate()
DateTimeFormatter formatter2 = DateTimeFormatter.ofLocalizedDate(FormatStyle.FULL);
//格式化
String str2 = formatter2.format(LocalDate.now());
System.out.println(str2);//2019年9月28日 星期六

//方式3:自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)
DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");
//格式化
String str3 = formatter3.format(localDateTime);
System.out.println(localDateTime);//2019-09-28T15:38:38.889
System.out.println(str3);//2019-09-28 03:38:38

//解析
TemporalAccessor parse1 = formatter3.parse("2019-09-28 03:38:38");
System.out.println(parse1);//{SecondOfMinute=38, HourOfAmPm=3, MicroOfSecond=0, NanoOfSecond=0, MinuteOfHour=38
}

Java比较器

对象数组的排序问题,涉及对象之间的比较。Java对象正常情况下,只能进行比较: == 或 != ,不能使用 > 或 < 。Java实现对象排序的方式有两种:

自然排序:java.lang.Comparable

Comparable接口的使用举例 自然排序

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

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

  3. 重写compareTo(obj)的规则:

    如果当前对象this大于形参对象obj,则返回正整数;

    如果当前对象this小于形参对象obj,则返回负整数;

    如果当前对象this等于形参对象obj,则返回零。

  4. 对于自定义类,若需要排序,可让自定义类实现Comparable接口,重写compareTo(obj)方法指明如何排序

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@Test
public void test1(){
String[] arr = new String[]{"AA","CC","KK","MM","GG","JJ","DD"};
Arrays.sort(arr);
System.out.println(Arrays.toString(arr));
}

//Eg:商品价格从低到高:按价格排序
class Goods implements Comparable {
}
@Override
public int compareTo(Objact o) {
if(o instanceod Goods) {
Goods goods = (Goods)o;
return Double.compare(this.price,goods.price)
}
throw new RuntimeException("传入的数据类型不一致");
}

定制排序:java.util.Comparator

  1. 背景:

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

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

    若方法返回正整数,表示o1大于o2;

    返回0,表示相等;

    返回负数,表示o1小于o2。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
@Test
public void test2(){
String[] arr = new String[]{"AA","CC","KK","MM","GG","JJ","DD"};
Arrays.sort(arr,new Comparator(){
//按照字符串大到小
@Override
public int compare(Object o1, Object o2) {
if(o1 instanceof String && o2 instanceof String){
String s1 = (String) o1;
String s2 = (String) o2;
return -s1.compareTo(s2);
}
// return 0;
throw new RuntimeException("输入的数据类型不一致");
}
});
System.out.println(Arrays.toString(arr));
}

Comparable接口与Comparator的使用的对比:

  • 前者一旦指定,保证Comparable接口实现类的对象在任何位置都可以比较大小

  • 后者属于临时性的比较。Arrays.sort(arr,new Comparator(){…..}

其他类

System类

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

  • 由于该类的构造器是private的,所以无法创建该类的对象,也就是无法实例化该类。其内部的成员变量和成员方法都是static的,所以也可以很方便的进行调用。

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

  • 成员方法

    • native long currentTimeMillis():
      该方法的作用是返回当前的计算机时间,时间的表达格式为当前计算机时间和GMT时间(格林威治时间)1970年1月1号0时0分0秒所差的毫秒数。

    • void exit(int status):
      该方法的作用是退出程序。其中status的值为0代表正常退出,非零代表异常退出。 使用该方法可以在图形界面编程中实现程序的退出功能等。

    • void gc():
      该方法的作用是请求系统进行垃圾回收。至于系统是否立刻回收,则取决于系统中垃圾回收算法的实现以及系统执行时的情况。

    • String getProperty(String key):
      该方法的作用是获得系统中属性名为key的属性对应的值。系统中常见的属性名以及属性的作用如下表所示:

      | 属性名 | 属性说明 |
      | ————- | ————————– |
      | java. version | Java运行时环境版本 |
      | java. home | java安装目录操作系统的名称 |
      | os.version | 操作系统的版本 |
      | user.nane | 用户的账户名称 |
      | user.home | 用户的主目录 |
      | user.dir | 用户的当前工作目录 |

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
@Test
public void test3(){
String javaVersion = System.getProperty("java.version");
System.out.println("java的version:" + javaVersion);
String javaHome = System.getProperty("java.home");
System.out.println("java的home:" + javaHome);
String osName = System.getProperty("os.name");
System.out.println("os的name:" + osName);
String osVersion = System.getProperty("os.version");
System.out.println("os的version:" + osVersion);
String userName = System.getProperty("user.name");
System.out.println("user的name:" + userName);
String userHome = System.getProperty("user.home");
System.out.println("user的home:" + userHome);
String userDir = System.getProperty("user.dir");
System.out.println("user的dir:" + userDir);
}
out:
java的version:1.8.0_191
java的home:C:\Program Files\Java\jdk1.8.0_191\jre
os的name:Windows 10
os的version:10.0
user的name:goodwell
user的home:C:\Users\goodwell
user的dir:D:\Codes\IdeaProjects\JavaSenior\Day

Math类

java.lang.Math 提供了一系列静态方法用于 科学 计算。其 方法的参数和返回值类型一般为double 型

  • abs 绝对值
  • acos,asin,atan,cos,sin,tan 三角函数
  • sqrt 平方根
  • pow(double a,doble b) a 的b 次幂
  • log 自然对数
  • exp e 为底指数
  • max(double a,double b)
  • min(double a,double b)
  • random() 返回0.0 到1.0 的随机数
  • long round(double a) double 型数据a 转换为long 型(四舍五入)
  • toDegrees(double angrad) 弧度—> 角度
  • toRadians(double angdeg) 角度—>弧度

BigInteger与BigDecimal

BigInteger类

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

  • 构造器
     BigInteger(String val):根据字符串构建BigInteger对象

  • 常用 方法
     public BigInteger abs()

    返回此 BigInteger 的绝对值的 BigInteger。
     BigInteger add(BigInteger val) :

    返回其值为 (this + val) 的 BigInteger
     BigInteger subtract(BigInteger val) :

    返回其值为 (this - val) 的 BigInteger
     BigInteger multiply(BigInteger val) :

    返回其值为 (this * val) 的 BigInteger
     BigInteger divide(BigInteger val) :

    返回其值为 (this / val) 的 BigInteger。整数相除只保留整数部分。
     BigInteger remainder(BigInteger val) :

    返回其值为 (this % val) 的 BigInteger。
     BigInteger[] divideAndRemainder(BigInteger val):返回包含 (this / val) 后跟(this % val) 的两个BigInteger 的数组。
     BigInteger pow(int exponent) :

    返回其值为 (this exponent ) 的 BigInteger。

### BigDecimal类

一般的Float类和Double类可以用来做科学计算或工程计算,但在 商业计算中,到 要求数字精度比较高,故用到java.math.BigDecimal类 。
BigDecimal类支持不可变的、任意精度的有符号十进制定点数。

  • 构造器
     public BigDecimal(double val)
     public BigDecimal(String val)
  • 常用方法
     public BigDecimal add(BigDecimal augend)
     public BigDecimal subtract(BigDecimal subtrahend)
     public BigDecimal multiply(BigDecimal multiplicand)
     public BigDecimal divide(BigDecimal divisor, int scale, int roundingMode)
1
2
3
4
5
6
7
8
9
public void testBigInteger() {
BigInteger bi = new BigInteger("12433241123");
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));
System.out.println(bd.divide(bd2, 15, BigDecimal.ROUND_HALF_UP));
}

##

-------------本文结束goodwell感谢您的阅读-------------
小二,上酒~
undefined