常用类

字符串相关的类

String 的特性

  • String类:代表字符串。Java 程序中的所有字符串字面值(如“abc”)都作为此类的实例实现。
  • String是一个final类,代表不可变的字符序列(不可变性)。
  • 字符串是常量,用双引号引起来表示。它们的值在创建之后不能更改。
  • String对象的字符内容是存储在一个字符数组value[]中的。
public final class String
		implements java.io.Serializable, Comparable<String>, CharSequence {
		/** The value is used for character storage. */
		private final char value[];
		/** Cache the hash code for the string */
	private int hash; // Default to 0

String 的使用

  1. String声明为final的,不可被继承

  2. String实现了Serializable接口:表示字符串是支持序列化的。

    ​ 实现了Comparable接口:表示String可以比较大小。

  3. String 内部定义了final char[ ] value[ ] 用于存储字符串数据

  4. String 代表不可变的字符序列。简称:不可变性。
    体现:

    1. 当字符串重新赋值时,需要重写指定内存区域赋值,不能使用原有的value进行赋值。

    2. 当对现有的字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值。

      1. 当调用String的replace() 方法修改指定的字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
  5. 通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。

  6. 字符串常量池中是不会存储相同内容的字符串的。

拓展:@Test的使用 是该方法可以不用main方法调用就可以测试出运行结果,是一种测试方法
一般函数都需要有main方法调用才能执行,注意被测试的方法必须是public修饰的

练习一
String s1 = "abc";  // 字面量的定义方式
String s2 = "abc";  // 与s1的内存地址相同(字符串常量池中是不会存储相同内容的字符串的。)
System.out.println(s1 == s2);   // 比较s1和s2的地址值
s1 = "hello";
System.out.println(s1); // hello
System.out.println(s2); // abc

image-20210523104006794

练习二
//  通过字面量定义的方式:此时的s1和s2的数据abc声明在方法区中的字符串常量池中。
String s1 = "abc";
String s2 = "abc";
// 通过new + 构造器的方式:此时的s3和s4保存的地址值,是数据在最空间中开辟的空间之后的对应的地址值。
String s3 = new String("abc");
String s4 = new String("abc");
System.out.println(s1 == s2);// true
System.out.println(s1 == s3);// false
System.out.println(s1 == s4);// false
System.out.println(s3 == s4);// false

image-20210523104635998

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

image-20210523104757048

练习四
String s1 = "javaEE";
String s2 = "hadoop";
String s3 = "javaEEhadoop"; // 字面量
String s4 = "javaEE" + "hadoop";
String s5 = s1 + "hadoop";  // 存在变量,则在堆中开辟一个新空间
String s6 = "javaEE" + s2;
String s7 = s1 + s2;
System.out.println(s3 == s4);   // true
System.out.println(s3 == s5);   // false
System.out.println(s3 == s6);   // false
System.out.println(s5 == s6);   // false
System.out.println(s3 == s7);   // false
System.out.println(s5 == s7);   // false
String s8 = s6.intern();// 返回值得到的s8使用的常量值中已经存在的“javaEEhadoop”
System.out.println(s3 == s8);   // true

image-20210523131603073

拓展:

@Test
public void test4(){
    String s1 = "helloworld";
    String s2 = "hello";
    String s3 = s2 + "world";
    System.out.println(s1 == s3);   // false

    final String s4 = "hello";  // s4 是常量
    String s5 = s4 + "world";
    System.out.println(s1 == s5);   // true
}

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


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(不包含)的一个子字符串。

@Test
public void test1(){
    String s1 = "Hello World";
    String s2 = "Hello Java";
    String s3 = "hello world";
    String s4 = "   空  格   ";
    String s5 = "a"; // 97
    String s6 = "c"; // 99

    System.out.println(s1.length());    // 11
    System.out.println(s1.charAt(4));   // o
    System.out.println(s1.isEmpty());   // false

    String t1 = s1.toLowerCase();
    System.out.println(s1);   // s1本身是不可变的,仍然为原来的字符串 Hello World
    System.out.println(t1);   // t1改成小写以后的字符串 hello world
    System.out.println(s1.toUpperCase());   // HELLO WORLD

    String t2 = s4.trim();
    System.out.println("---" + s4 + "---"); // ---   空  格   ---
    System.out.println("---" + t2 + "---"); // ---空  格---

    System.out.println(s1.equals(s2));  // false
    System.out.println(s1.equalsIgnoreCase(s3));    // true
    System.out.println(s1.concat(s2));  // Hello WorldHello Java
    System.out.println(s5.compareTo(s6));   // -2
    System.out.println(s1.substring(3));    // lo World
    System.out.println(s1.substring(3,8));  // lo Wo
}
常用方法二

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
indexOf 是从左往右找;lastIndexOf 是从右往左找。
当:存在唯一的一个str或者不存在str时:indexOf(str)和lastIndexOf(str)返回值相同

@Test
public void test2(){
    String s1 = "Hello World";

    System.out.println(s1.endsWith("World"));   // true
    System.out.println(s1.startsWith("abc"));   // false
    System.out.println(s1.startsWith("llo", 2)); // true

    String s2 = "Hello Java eova";
    String s3 = "Java";

    System.out.println(s2.contains(s3)); // true
    System.out.println(s2.indexOf(s3));  // 6
    System.out.println(s2.indexOf("e", 4)); // 11
    System.out.println(s2.lastIndexOf("va"));    // 13
    System.out.println(s2.lastIndexOf("va",12)); // 8
}
常用方法三

替换:
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个,如果超过了,剩下的全部都放到最后一个元素中。

@Test
public void test3(){
    String s1 = "Hello Java java";

    System.out.println(s1.replace(\'a\', \'A\')); // Hello JAvA jAvA
    System.out.println(s1.replace("Java", "Python")); // Hello Python java

    System.out.println("***************************************");
    String str = "12hello34world5java7891mysql456";
    //把字符串中的数字替换成,,如果结果中开头和结尾有,的话去掉
    String string = str.replaceAll("\\d+", ",").replaceAll("^,|,$", "");
    System.out.println(string); // hello,world,java,mysql

    System.out.println("***************************************");
    String str1 = "12345";
    //判断str字符串中是否全部有数字组成,即有1-n个数字组成
    boolean matches = str1.matches("\\d+");
    System.out.println(matches); // true
    String tel = "0571-4534289";
    //判断这是否是一个杭州的固定电话
    boolean result = tel.matches("0571-\\d{7,8}");
    System.out.println(result);  // true

    System.out.println("***************************************");
    String str2 = "hello|world|java";
    String[] strs = str2.split("\\|");
    for (int i = 0; i < strs.length; i++) {
        System.out.println(strs[i]);
    }
    System.out.println();
    String str3 = "hello.world.java";
    String[] strs2 = str3.split("\\.");
    for (int i = 0; i < strs2.length; i++) {
        System.out.println(strs2[i]);
    }/*
        hello
        world
        java
     */
}

复习

String与基本数据类型转换

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

  • Integer包装类的public static int parseInt(String s):可以将由“数字”字 符组成的字符串转换为整型。
  • 类似地,使用java.lang包中的Byte、Short、Long、Float、Double类调相应 的类方法可以将由“数字”字符组成的字符串,转化为相应的基本数据类型。

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

  • 调用String类的public String valueOf(int n)可将int型转换为字符串。
  • 相应的valueOf(byte b)、valueOf(long l)、valueOf(float f)、valueOf(double d)、valueOf(boolean b)可由参数的相应类型到字符串的转换。
@Test    public void test1(){        String s1 = "123";        int num = Integer.parseInt(s1);        System.out.println(num);    // 123 数值型        String s2 = String.valueOf(num);        System.out.println(s2);     // 123 字符型    }}
String与字符数组转换

字符串 –> 字符数组

  • public char[ ] toCharArray():将字符串中的全部字符存放在一个字符数组 中的方法。

字符数组 –> 字符串

  • String 类的构造器:String(char[ ]) 和 String(char[ ],int offset,int length) 分别用字符数组中的全部字符和部分字符创建字符串对象。
@Testpublic void test2(){    String s1 = "abcjava";    char[] chars = s1.toCharArray();    for (int i = 0; i < chars.length; i++) {        System.out.println(chars[i]); // abcjava    }    char[] arr = {\'j\', \'a\', \'v\', \'a\'};    String s2 = new String(arr);    System.out.println(s2); // java}

String、StringBuffer & StringBuilder

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

源码分析
String str = new String(); //char[ ] value = new char[0];
String str1 = new String(“abc”); //char[] value = new char[]{\’a\’,\’b\’,\’c\’};

​ StringBuffer s1 = new StringBuffer(); //char[ ] value = new char[16]; 底层创建了一个长度是16的数组。
​ s1.append(\’a\’); //value[0] = \’a\’;
​ s1.append(\’b\’); //value[1] = \’b\’;

​ StringBuffer s2 = new StringBuffer(“abc”); //char[ ] value = new char[s2.length() + 16]

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

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

@Testpublic void test1() {    StringBuffer s1 = new StringBuffer("abc");    s1.setCharAt(0, \'m\');    System.out.println(s1); // 此时 s1 = mbc}
/*
对比String、StringBuffer、StringBuilder三者的效率:
从高到低排列:StringBuilder > StringBuffer > String
 */
@Test
public void test2() {
    long startTime = 0L;
    long endTime = 0L;
    String text = "";
    StringBuffer buffer = new StringBuffer("");
    StringBuilder builder = new StringBuilder("");
    //开始对比
    startTime = System.currentTimeMillis();
    for (int i = 0; i < 20000; i++) {
        buffer.append(String.valueOf(i));
    }
    endTime = System.currentTimeMillis();
    System.out.println("StringBuffer的执行时间:" + (endTime - startTime));

    startTime = System.currentTimeMillis();
    for (int i = 0; i < 20000; i++) {
        builder.append(String.valueOf(i));
    }
    endTime = System.currentTimeMillis();
    System.out.println("StringBuilder的执行时间:" + (endTime - startTime));
    startTime = System.currentTimeMillis();
    for (int i = 0; i < 20000; i++) {
        text = text + i;
    }

    endTime = System.currentTimeMillis();
    System.out.println("String的执行时间:" + (endTime - startTime));
}
/*
StringBuffer的执行时间:3
StringBuilder的执行时间:2
String的执行时间:189
*/


JDK8之前日期时间API

1. java.lang.System类

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

  • 此方法适于计算时间差。

计算世界时间的主要标准有:

  • UTC(Coordinated Universal Time)
  • GMT(Greenwich Mean Time)
  • CST(Central Standard Time)
@Test
public void test1(){
    long time = System.currentTimeMillis();
    // 返回当前时间与1970年1月1日0时0分0秒之间以毫秒为单位的时间差。
    System.out.println(time);	// 1622030409096
}

2. java.util.Date类

java.util.Date类
|— java.sql.Date类

  1. 两个构造器的使用

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

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

  2. 两个方法的使用

    toString(): 显示当前的年,月,日,分,秒。(默认可忽略)

    getTime(): 获取当前Date对象对应的毫秒数 (时间戳)。

    // 构造器一:Date() 创建一个对应当前时间的Date对象。Date date1 = new Date();System.out.println(date1.toString());   // Mon May 24 19:18:51 CST 2021System.out.println(date1.getTime());    // 1621855371831// 构造器二:创建指定毫秒数的Date对象。Date date2 = new Date(1421855371831L);System.out.println(date2);  // Wed Jan 21 23:49:31 CST 2015
    
  3. java.sql.Date 对应这数据库中的日期类型的变量

    ​ 如何实例化

    何将java.util.Date 对象转换为java.sql.Date 对象

    // 创建java.sql.Date 对象java.sql.Date date3 = new java.sql.Date(4652135465131L);System.out.println(date3.toString());   // 2117-06-03System.out.println(date3.getTime());    // 4652135465131// 如何将java.util.Date 对象转换为java.sql.Date 对象// 情况一://Date date4 = new java.sql.Date(4652843465131L);//java.sql.Date date5 = (java.sql.Date)date4;// 情况二:Date date6 = new Date();java.sql.Date date7 = new java.sql.Date(date6.getTime());System.out.println(date7);
    

3. java.text.SimpleDateFormat类

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

两种操作:
格式化:日期 –> 字符串
解析: 字符串 –> 日期

1. 格式化:

SimpleDateFormat() :默认的模式和语言环境创建对象

public SimpleDateFormat(String pattern):该构造方法可以用参数pattern 指定的格式创建一个对象,该对象调用:public String format(Date date):方法格式化时间对象date

2. 解析:

public Date parse(String source):从给定字符串的开始解析文本,以生成 一个Date型日期。

@Testpublic void testSimpleDateFormat() throws ParseException {    // 实例化SimpleDateFormat:使用默认构造器(基本不使用)    SimpleDateFormat sdf = new SimpleDateFormat();    Date date = new Date();    System.out.println(date);   // Wed May 26 15:08:11 CST 2021    // 格式化:日期 --> 字符串    String format = sdf.format(date);    System.out.println(format); // 2021/5/26 下午3:08    // 解析:格式化的逆过程;字符串 --> 日期(Date型对象)    String str = "19/12/18 上午4:45"; // 默认格式    Date parse = sdf.parse(str);    System.out.println(parse);    // **************************************    /*按照指定的方式格式化和解析:调用代参的构造器。*/    SimpleDateFormat sdf1 = new SimpleDateFormat("yyyy-MM-dd hh:mm:ss");    // 格式化    String format1 = sdf1.format(date);    System.out.println(format1);    // 2021-05-26 05:26:58    // 解析:(结果为Date型对象)    Date parse1 = sdf1.parse("2321-05-26 05:26:58"); // 格式必须与构造器中的一样,否则抛出异常    System.out.println(parse1);     // Thu May 26 05:26:58 CST 2321}
Letter Date or Time Component Presentation Examples
G Era designator Text AD
y Year Year 1996; 96
Y Week year Year 2009; 09
M Month in year (context sensitive) Month July; Jul; 07
L Month in year (standalone form) Month July; Jul; 07
w Week in year Number 27
W Week in month Number 2
D Day in year Number 189
d Day in month Number 10
F Day of week in month Number 2
E Day name in week Text Tuesday; Tue
u Day number of week (1 = Monday, …, 7 = Sunday) Number 1
a Am/pm marker Text PM
H Hour in day (0-23) Number 0
k Hour in day (1-24) Number 24
K Hour in am/pm (0-11) Number 0
h Hour in am/pm (1-12) Number 12
m Minute in hour Number 30
s Second in minute Number 55
S Millisecond Number 978
z Time zone General time zone Pacific Standard Time; PST; GMT-08:00
Z Time zone RFC 822 time zone -0800
X Time zone ISO 8601 time zone -08; -0800; -08:00
/*练习一:字符串“2021-5-25”转换为java.sql.Date */@Testpublic void testEser() throws ParseException {    String birth = "2021-05-25";    // 1. 实例化代参SimpleDateFormat    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");    // 2. 解析为Date型对象    Date parse = sdf.parse(birth);    // 3. 将Date型对象转化为SQL型    java.sql.Date sql1 = new java.sql.Date(parse.getTime());    System.out.println(sql1);}/*练习二:“三天打渔两天晒网"  1990-01-01  开始到今天 是在打渔还是在晒网 */@Testpublic void testFishing() throws ParseException {    String birth = "1990-01-01";    SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");    Date lowdate = sdf.parse(birth);    Date newdate = sdf.parse("2021-05-31");    // Date newdate = new Date();    long day1 = (newdate.getTime() - lowdate.getTime()) / (1000 * 60 * 60 * 24) + 1;    long day = day1 % 5;    if (day >= 1 && day <= 3) {        System.out.println("今天在打渔!!!");    } else if (day == 4 | day == 0) {        System.out.println("今天在晒网……");    }}

4. java.util.Calendar(日历)类

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

  • 获取Calendar实例的方法

    1. 使用Calendar.getInstance()方法
    2. 调用它的子类GregorianCalendar的构造器。
  • 一个Calendar的实例是系统时间的抽象表示,通过get(int field)方法来取得想 要的时间信息。比如YEAR、MONTH、DAY_OF_WEEK、HOUR_OF_DAY 、 MINUTE、SECOND

    ​ public void set(int field,int value)

    ​ public void add(int field,int amount)

    ​ public final Date getTime()

    ​ public final void setTime(Date date)

  • 注意:

    获取月份时:一月是0,二月是1,以此类推,12月是11。

    获取星期时:周日是1,周二是2 , 。。。。周六是7。

/*Calendar 日历类(抽象类)的使用    Ctrl + n */@Testpublic void testCalendar() {    // 1.实例化    // 方式一:创建其子类(GregorianCalendar)的对象。    // 方式二:调用其静态方法getInstance()。    Calendar instance = Calendar.getInstance();    // 2.使用方法    // get    int days = instance.get(Calendar.DAY_OF_YEAR);    System.out.println("今天是这一年的第" + days + "天");    //System.out.println(instance.get(Calendar.DAY_OF_MONTH));    // set    // 没有返回值,修改的是当前的instance对象本身 可变性    instance.set(Calendar.DAY_OF_YEAR, 300);    days = instance.get(Calendar.DAY_OF_YEAR);    System.out.println("今天是这一年的第" + days + "天");    // add    instance.add(Calendar.DAY_OF_YEAR, -3); // 加 -3天    days = instance.get(Calendar.DAY_OF_YEAR);    System.out.println(days);    // getTime(): 日历类 --》 Date    Date time = instance.getTime();    System.out.println(time);    // setTime(): Date --》 日历类    Date date = new Date();    instance.setTime(date);    days = instance.get(Calendar.DAY_OF_YEAR);    System.out.println(days);}


JDK8中新日期时间API

新日期时间API出现的背景

​ 如果我们可以跟别人说:“我们在1502643933071见面,别晚了!”那么就再简单不 过了。但是我们希望时间与昼夜和四季有关,于是事情就变复杂了。JDK 1.0中包含了 一个java.util.Date类,但是它的大多数方法已经在JDK 1.1引入Calendar类之后被弃用 了。而Calendar并不比Date好多少。它们面临的问题是:

  • 可变性:像日期和时间这样的类应该是不可变的。
  • 偏移性:Date中的年份是从1900开始的,而月份都从0开始。
  • 格式化:格式化只对Date有用,Calendar则不行。
  • 此外,它们也不是线程安全的;不能处理闰秒等。

总结:对日期和时间的操作一直是Java程序员最痛苦的地方之一。

LocalDate、LocalTime、LocalDateTime

​ LocalDate、LocalTime、LocalDateTime 类是其中较重要的几个类,它们的实例 是不可变的对象,分别表示使用 ISO-8601日历系统的日期、时间、日期和时间。 它们提供了简单的本地日期或时间,并不包含当前的时间信息,也不包含与时区 相关的信息。

  • LocalDate代表IOS格式(yyyy-MM-dd)的日期,可以存储 生日、纪念日等日期。
  • LocalTime表示一个时间,而不是日期。
  • LocalDateTime是用来表示日期和时间的,这是一个最常用的类之一

注:ISO-8601日历系统是国际标准化组织制定的现代公民的日期和时间的表示 法,也就是公历。

image-20210527193517977

@Test
public void TestDate(){
    // 偏移量
    Date date = new Date(2021 - 1900,5,26);
    System.out.println(date);
}

/*
LocalDate、LocalTime、LocalDateTime
 */
@Test
public void test1(){
    // now(): 获取当前的日期、时间、日期+时间
    LocalDate localDate = LocalDate.now();  // 当前日期
    LocalTime localTime = LocalTime.now();  // 当前时间
    LocalDateTime localDateTime = LocalDateTime.now();  // 当前日期和时间

    System.out.println(localDate);
    System.out.println(localTime);
    System.out.println(localDateTime);

    // of(): 设置指定的年、月、日、时、分、秒
    LocalDateTime localDateTime1 = LocalDateTime.of(2021, 5, 21, 13, 14);
    System.out.println(localDateTime1);

    System.out.println("**************");
    // getXXX(): 获取相关的属性
    System.out.println(localDateTime.getDayOfYear());
    System.out.println(localDateTime.getMonth());
    System.out.println(localDateTime.getMonthValue());
    System.out.println(localDateTime.getDayOfMonth());
    System.out.println(localDateTime.getDayOfWeek());
    System.out.println(localDateTime.getMinute());


    /*体现出不可变性*/
    // withXXX(): 设置相关的属性
    LocalDate localDate1 = localDate.withDayOfMonth(20);
    System.out.println(localDate);  // 2021-05-27
    System.out.println(localDate1); // 2021-05-20
    System.out.println(localDateTime.withHour(12));

    // plusXXX(): 在原有的基础上加上
    LocalDate localDate2 = localDate.plusMonths(5);
    System.out.println(localDate);  // 2021-05-27
    System.out.println(localDate2); // 2021-10-27

    // 在原有的基础上减去
    LocalDate localDate3 = localDate.minusDays(20);
    System.out.println(localDate);  // 2021-05-27
    System.out.println(localDate3); // 2021-05-07
}

瞬时:Instant

​ Instant:时间线上的一个瞬时点。 这可能被用来记录应用程序中的事件时间戳。它只是简单的表示自1970年1月1日0时0分0秒(UTC)开始的毫秒数。

image-20210527193741915

/*
Instant 类的使用(瞬时点)
    类似于java.util.Date类
 */
@Test
public void test2(){
    // now(): 获取本初子午线对应的标准时间
    Instant instant = Instant.now();
    System.out.println(instant); // 2021-05-27T10:05:01.837485400Z

    // 添加时间的偏移量
    OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
    System.out.println(offsetDateTime); // 2021-05-27T18:07:11.513827+08:00

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

    // ofEpochMilli(): 通过给定的毫秒数,获取Instant实例 --> Date(long millis)
    Instant instant1 = Instant.ofEpochMilli(1622110242970L);
    System.out.println(instant1);   // 2021-05-27T10:10:42.970Z
}

格式化与解析日期或时间

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

  • 预定义的标准格式。如: ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIME
  • 本地化相关的格式。如:ofLocalizedDateTime(FormatStyle.LONG)
  • 自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)

image-20210527193934256

/*DateTimeFormatter: 格式化或解析日期、时间    类似于SimpleDateFormat */@Testpublic void test3() {    //方式一:预定义的标准格式。如:ISO_LOCAL_DATE_TIME;ISO_LOCAL_DATE;ISO_LOCAL_TIME    DateTimeFormatter formatter = DateTimeFormatter.ISO_LOCAL_DATE_TIME;    // 格式化:日期 --> 字符串    LocalDateTime localDateTime = LocalDateTime.now();    String s1 = formatter.format(localDateTime);    System.out.println(localDateTime);// 2021-05-27T18:57:23.654493400    System.out.println(s1); // 2021-05-27T18:57:23.6544934    // 解析:字符串 --> 日期    TemporalAccessor parse = formatter.parse("2021-05-27T18:57:23.6544934");    System.out.println(parse);  // {},ISO resolved to 2021-05-27T18:57:23.654493400    /*方式二:    本地化相关的格式。如:ofLocalizedDateTime(FormatStyle.LONG)    FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT : 适用于LocalDateTime */    DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.SHORT);    // 格式化    String s2 = formatter1.format(localDateTime);    System.out.println(s2); // 2021/5/27 下午7:07    /*本地化相关的格式。如:ofLocalizedDate()    FormatStyle.FULL / FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT : 适用于LocalDate */    DateTimeFormatter formatter2 = DateTimeFormatter.ofLocalizedDate(FormatStyle.FULL);    String s3 = formatter2.format(LocalDate.now());    System.out.println(s3); // 2021年5月27日星期四    //重点:方式三:自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)    DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");    // 格式化    String s4 = formatter3.format(localDateTime);    System.out.println(s4); // 2021-05-27 07:21:00    // 解析    TemporalAccessor parse1 = formatter3.parse("2021-05-27 07:21:00");    System.out.println(parse1);    /* {MinuteOfHour=21, MicroOfSecond=0, SecondOfMinute=0, HourOfAmPm=7,    NanoOfSecond=0, MilliOfSecond=0},ISO resolved to 2021-05-27 */}

其他API

  • ZoneId:该类中包含了所有的时区信息,一个时区的ID,如 Europe/Paris
  • ZonedDateTime:一个在ISO-8601日历系统时区的日期时间,如 2007-12- 03T10:15:30+01:00 Europe/Paris。

    • 其中每个时区都对应着ID,地区ID都为“{区域}/{城市}”的格式,例如: Asia/Shanghai等
  • Clock:使用时区提供对当前即时、日期和时间的访问的时钟。
  • 持续时间:Duration,用于计算两个“时间”间隔
  • 日期间隔:Period,用于计算两个“日期”间隔
  • TemporalAdjuster : 时间校正器。有时我们可能需要获取例如:将日期调整 到“下一个工作日”等操作。
  • TemporalAdjusters : 该类通过静态方法 (firstDayOfXxx()/lastDayOfXxx()/nextXxx())提供了大量的常用 TemporalAdjuster 的实现。
//ZoneId:类中包含了所有的时区信息// ZoneId的getAvailableZoneIds():获取所有的ZoneIdSet<String> zoneIds = ZoneId.getAvailableZoneIds();for (String s : zoneIds) {    System.out.println(s);}// ZoneId的of():获取指定时区的时间LocalDateTime localDateTime = LocalDateTime.now(ZoneId.of("Asia/Tokyo"));System.out.println(localDateTime);//ZonedDateTime:带时区的日期时间// ZonedDateTime的now():获取本时区的ZonedDateTime对象ZonedDateTime zonedDateTime = ZonedDateTime.now();System.out.println(zonedDateTime);// ZonedDateTime的now(ZoneId id):获取指定时区的ZonedDateTime对象ZonedDateTime zonedDateTime1 = ZonedDateTime.now(ZoneId.of("Asia/Tokyo"));System.out.println(zonedDateTime1);
//Duration:用于计算两个“时间”间隔,以秒和纳秒为基准
LocalTime localTime = LocalTime.now();
LocalTime localTime1 = LocalTime.of(15, 23, 32);
//between():静态方法,返回Duration对象,表示两个时间的间隔
Duration duration = Duration.between(localTime1, localTime);
System.out.println(duration);
System.out.println(duration.getSeconds());
System.out.println(duration.getNano());
LocalDateTime localDateTime = LocalDateTime.of(2016, 6, 12, 15, 23, 32);
LocalDateTime localDateTime1 = LocalDateTime.of(2017, 6, 12, 15, 23, 32);
Duration duration1 = Duration.between(localDateTime1, localDateTime);
System.out.println(duration1.toDays());
//Period:用于计算两个“日期”间隔,以年、月、日衡量
LocalDate localDate = LocalDate.now();
LocalDate localDate1 = LocalDate.of(2028, 3, 18);
Period period = Period.between(localDate, localDate1);
System.out.println(period);
System.out.println(period.getYears());
System.out.println(period.getMonths());
System.out.println(period.getDays());
Period period1 = period.withYears(2);
System.out.println(period1);
// TemporalAdjuster:时间校正器
// 获取当前日期的下一个周日是哪天?
TemporalAdjuster temporalAdjuster = TemporalAdjusters.next(DayOfWeek.SUNDAY);
LocalDateTime localDateTime = LocalDateTime.now().with(temporalAdjuster);
System.out.println(localDateTime);
// 获取下一个工作日是哪天?
LocalDate localDate = LocalDate.now().with(new TemporalAdjuster() {
    @Override
    public Temporal adjustInto(Temporal temporal) {
        LocalDate date = (LocalDate) temporal;
        if (date.getDayOfWeek().equals(DayOfWeek.FRIDAY)) {
            return date.plusDays(3);
        } else if (date.getDayOfWeek().equals(DayOfWeek.SATURDAY)) {
            return date.plusDays(2);
        } else {
            return date.plusDays(1);
        }
    }
});
System.out.println("下一个工作日是:" + localDate);


Java 比较器

image-20210528225714770

在Java中经常会涉及到对象数组的排序问题,那么就涉及到对象之间 的比较问题。

  1. 说明:Java中的对象,正常情况下,只能进行比价: == 或 != 。不能使用 > 或 < 的
    但是在开发场景中,我们需要对多个对象进行排序,也就是比较对象的大小。
    如何实现?使用两个结构中的任何一个:Comparable 或 Comparator

  2. Comparable接口 Comparator的与使用

    ​ Comparable接口的方式一旦定义,保证Comparable接口时间线类的对象在任何位置都可以比较大小
    ​ Comparator接口属于临时性的比较。

方式一:自然排序:Comparable

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

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

  3. 重写compareTo() 的规则:

    ​ 如果当前对象this大于形参对象obj,则返回正整数,
    ​ 如果当前对象this小于形参对象obj,则返回负整数,
    ​ 如果当前对象this等于形参对象obj,则返回零。

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

@Test
public void test1() {
    String[] arr = new String[]{"AA", "RR", "BB", "GG", "CC"};
    //
    Arrays.sort(arr);   // 默认升序
    System.out.println(Arrays.toString(arr));
}
@Test
public void test2() {
    Goods[] arr = new Goods[5];
    arr[0] = new Goods("LenovoMouse", 34);
    arr[1] = new Goods("DellMouse", 43);
    arr[2] = new Goods("XiaomiMouse", 12);
    arr[3] = new Goods("HuaweiMouse", 65);
    arr[4] = new Goods("MicrosoftMouse", 43);
    Arrays.sort(arr);
    System.out.println(Arrays.toString(arr));
}

方式二:定制排序:Comparator

  1. 背景:

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

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

​ 如果方法返回正整数,则表示o1大于o2;

​ 如果返回0,表示相等;返回负整数,表示o1小于o2。

@Test
public void test3() {
    String[] arr = new String[]{"AA", "RR", "BB", "GG", "CC"};
    Arrays.sort(arr, new Comparator<String>() {
        // 按照字符串从大到小的顺序排列
        @Override
        public int compare(String o1, String 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));
}
@Test
public void test4() {
    Goods[] arr = new Goods[6];
    arr[0] = new Goods("LenovoMouse", 34);
    arr[1] = new Goods("DellMouse", 43);
    arr[2] = new Goods("XiaomiMouse", 12);
    arr[3] = new Goods("HuaweiMouse", 65);
    arr[4] = new Goods("MicrosoftMouse", 43);
    arr[5] = new Goods("LenovoMouse", 56);
    Arrays.sort(arr, new Comparator() {
        // 指明商品比较大小的方式:按照产品名称从低到高排序,在按照价格从高到低排序
        @Override
        public int compare(Object o1, Object o2) {
            if (o1 instanceof Goods && o2 instanceof Goods) {
                Goods g1 = (Goods) o1;
                Goods g2 = (Goods) o2;
                if (g1.getName().equals(g2.getName())) {
                    return -Double.compare(g1.getPrice(), g2.getPrice());
                } else {
                    return g1.getName().compareTo(g2.getName());
                }
            }
            throw new RuntimeException("输入的数据类型不一致");
        }
    });
    System.out.println(Arrays.toString(arr));
}

其他常用类

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的属性对应的值。系统中常见 的属性名以及属性的作用如下表所示:

image-20210529110725018

@Test
public void test1() {
    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);
}
Math类

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

方法 返回类型 描述 备注
abs(int a) int 取绝对值 ABSolute
sqrt(double a) double 平方根 SQuare RooT
max(int a, int b) int 取两数的最大值
min(int a, int b) int 取两数的最小值
pow(double base, double exponent) double 第一个参数的第二个参数次方 power:次方
base:底数
exponent:指数
round(float a) int 四舍五入 round off
Math.floor(x+0.5)
random() double 随机数 0.0~1.0之间的小数(包括0.0不包括1.0)
ceil(double a) double 向上取整 天花板
floor(double a) double 向下取整 地板
BigInteger与BigDecimal

java.math包的BigInteger可以表示不可变的任意精度的整数

构造器

  • 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) :返回其值为 (thisexponent) 的 BigInteger。

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

常用方法

  • public BigDecimal add(BigDecimal augend)
  • public BigDecimal subtract(BigDecimal subtrahend)
  • public BigDecimal multiply(BigDecimal multiplicand)
  • public BigDecimal divide(BigDecimal divisor, int scale, int roundingMode)
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));
}

版权声明:本文为kin02原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/kin02/p/14826038.html