Java常用类

字符串相关类

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

对比String、StringBuffer、StringBuilder三者的效率:
从高到低排列:StringBuilder > StringBuffer > String

String

String的特性

  1. String类:代表字符串。Java 程序中的所有字符串字面值(如 "abc" )都作 为此类的实例实现。

  2. String是一个final类,也就是说字符串是常量,代表不可变的字符序列,创建之后不能更改。

    ​ 体现:

    1. 当对字符串重新赋值时,需要重写指定内存区域赋值,不能使用原有的value进行赋值。
    2. 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
    3. 当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
  3. String对象的字符内容是存储在一个字符数组value[](数组被final所修饰)中的。

【理解:】

public void test1(){
    String s1 = "abc";//字面量的定义方式
    String s2 = "abc";
    s1 = "hello";

    System.out.println(s1 == s2);//比较s1和s2的地址值
                                 //在s1被赋"hello"前结果是true  被修改后是false

    System.out.println(s1);//hello
    System.out.println(s2);//abc

    System.out.println("*****************");

    String s3 = "abc";
    s3 += "def";
    //这里是在常量池里先新造了一个"abcdef",然后再赋给了s3
    System.out.println(s3);//abcdef
    System.out.println(s2);//abc

    System.out.println("*****************");

    String s4 = "abc";
    String s5 = s4.replace('a', 'm');
    System.out.println(s4);//abc,s4并没有发生改变
    System.out.println(s5);//mbc,这个也是新创的然后再进行赋值

}
  • 解释String中的字面量:

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

String对象的创建

//本质上this.value = new char[0]; 
String  s1 = new String(); 
//本质上this.value = original.value;
String  s2 = new String(String original); 
//本质上this.value = Arrays.copyOf(value, value.length); 
String  s3 = new String(char[] a); 
String  s4 = new String(char[] a,int startIndex,int count);
  • String str1 = “abc”;与String str2 = new String(“abc”);的区别:

    ​ 字符串常量存储在 字符串常量池, 目的是共享
    ​ 字符串对象存储在堆中,其中保存的常量值的是对应在常量池当中。

  • String s = new String("abc");方式创建对象,在内存中创建了几个对象?
    两个:一个是堆空间中new结构
    一个是char[]对应的常量池中的数据:"abc"

【理解:】

public void test2(){
    //通过字面量定义的方式:此时的s1和s2的数据javaEE声明在方法区中的字符串常量池中。
    String s1 = "javaEE";
    String s2 = "javaEE";
    //通过new + 构造器的方式:此时的s3和s4保存的地址值,是数据在堆空间中开辟空间以后对应的地址值。
    String s3 = new String("javaEE");
    String s4 = new String("javaEE");

    System.out.println(s1 == s2);//true
    System.out.println(s1 == s3);//false
    System.out.println(s1 == s4);//false
    System.out.println(s3 == s4);//false

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

    System.out.println(p1.name.equals(p2.name));//true,name所存的地址比较
    System.out.println(p1.name == p2.name);//true,name对应的地址比较
                                            //因为都在这里的name是通过字面量的方式定义的,两者指向的都是常量池当中的name
    //说明不可变性
    p1.name = "Jerry";
    System.out.println(p2.name);//Tom
}

字符串拼接

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

【理解:】

public void test3(){
    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(s3 == s7);//false
    System.out.println(s5 == s6);//false
    System.out.println(s5 == s7);//false
    System.out.println(s6 == s7);//false

    String s8 = s6.intern();//堆空间的s6对象在调用intern()之后,会将常量池中已经存在的"javaEEhadoop"字符串 赋值给s8。
    System.out.println(s3 == s8);//true
}

public void test4(){
    String s1 = "javaEEhadoop";
    String s2 = "javaEE";
    String s3 = s2 + "hadoop";
    System.out.println(s1 == s3);//false

    final String s4 = "javaEE";//s4:常量
    String s5 = s4 + "hadoop";
    System.out.println(s1 == s5);//true

}

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):比较两个字符串的大小。逐个比较,返回的是ASCII码中的差值。

【查找】

  • 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字符串是否以指定前缀开始

【替换】

  • 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和其他结构的转换

  1. String和基本数据类型、包装类的转换
  • 字符串 -> 基本数据类型、包装类:
    • Integer包装类的public static int parseInt(String s):可以将由“数字”字 符组成的字符串转换为整型。
    • 类似地,使用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)可由参数的相应类型到字符串的转换
public void test1(){
        String str1 = "123";
//        int num = (int)str1;//错误的
        int num = Integer.parseInt(str1);

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

        System.out.println(str1 == str3);
    }
  1. String和字符数组的转换
  • 字符串 -> 字符数组
    • String类的 public char[] toCharArray():将字符串中的全部字符存放在一个字符数组 中的方法。
    • public void getChars(int srcBegin, int srcEnd, char[] dst,
      int dstBegin):提供了将指定索引范围内的字符串存放到数组中的方法。
  • 字符数组 -> 字符串
    • String 类的构造器:String(char[]) 和 String(char[],int offset,int length) 分别用字符数组中的全部字符和部分字符创建字符串对象。
public void test2(){
    String str1 = "abc123";  

    char[] charArray = str1.toCharArray();
    for (int i = 0; i < charArray.length; i++) {
        System.out.println(charArray[i]);
    }

    char[] arr = new char[]{'h','e','l','l','o'};
    String str2 = new String(arr);
    System.out.println(str2);
}
  1. String和字节数组的转换
  • 字符串 -> 字节数组(编码)
    • public byte[] getBytes() :使用平台的默认字符集将此 String 编码为 byte 序列,并将结果存储到一个新的 byte 数组中。
    • public byte[] getBytes(String charsetName) :使用指定的字符集将 此 String 编码到 byte 序列,并将结果存储到新的 byte 数组。
  • 字节数组 -> 字符串(解 码)
    • String(byte[]):通过使用平台的默认字符集解码指定的 byte 数组,构 造一个新的 String。
    • String(byte[],int offset,int length) :用指定的字节数组的一部分, 即从数组起始位置offset开始取length个字节构造一个字符串对象。
/*
String 与 byte[]之间的转换
编码:String --> byte[]:调用String的getBytes()
解码:byte[] --> String:调用String的构造器

编码:字符串 -->字节  (看得懂 --->看不懂的二进制数据)
解码:编码的逆过程,字节 --> 字符串 (看不懂的二进制数据 ---> 看得懂)

说明:解码时,要求解码使用的字符集必须与编码时使用的字符集一致,否则会出现乱码。
 */
@Test
public void test3() throws UnsupportedEncodingException {
    String str1 = "abc123中国";
    byte[] bytes = str1.getBytes();//使用默认的字符集utf-8,进行编码。
    System.out.println(Arrays.toString(bytes));//这里的汉字对应的会输出3个字节

    byte[] gbks = str1.getBytes("gbk");//使用gbk字符集进行编码。
    System.out.println(Arrays.toString(gbks));//这里的汉字对应的会输出2个字节

    System.out.println("******************");

    String str2 = new String(bytes);//使用默认的字符集,进行解码。
    System.out.println(str2);

    String str3 = new String(gbks);
    System.out.println(str3);//出现乱码。原因:编码集和解码集不一致!


    String str4 = new String(gbks, "gbk");
    System.out.println(str4);//没有出现乱码。原因:编码集和解码集一致!


}

StringBuffer

概述

  • java.lang.StringBuffer代表可变的字符序列,JDK1.0中声明,可以对字符 串内容进行增删,此时不会产生新的对象。

  • 很多方法与String相同。

  • 作为参数传递时,方法内部可以改变值。

  • 常用于多线程问题,共享数据

StringBuffer类创建对象

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

  • StringBuffer():初始容量为16的字符串缓冲区
    • 每次超出时会扩容,当前的容量 ×2 +2
  • StringBuffer(int size):构造指定容量的字符串缓冲区
  • StringBuffer(String str):将内容初始化为指定字符串内容
String s = new String("我喜欢学习"); 
StringBuffer buffer = new StringBuffer("我喜欢学习"); buffer.append("数学"); 

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):将指定位置的字符进行更改

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

StringBuiilder

不建议用于多线程问题,不太安全,但是效率高。

创建对象 和 常用方法 与StringBuffer类似

JDK 8之前的日期时间类

System 静态方法

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

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

Date类

java.util.Date类

  1. 两个构造器的使用
    构造器一:空参,Date()
    创建 对应当前时间 的Date对象
    构造器二:有参,Date(155030620410L)
    创建 指定毫秒数 的Date对象

  2. 两个方法的使用
    toString():显示当前的年、月、日、时、分、秒
    getTime():获取当前Date对象对应的毫秒数。(时间戳)

java.sql.Date类

对应着数据库中的日期类型的变量

  1. 实例化:java.sql.Date date3 = new java.sql.Date(35235325345L);

  2. 将java.util.Date对象转换为java.sql.Date对象:

Date util_date = new Date();
        java.sql.Date sql_date = new java.sql.Date(util_date.getTime());
	public void test2(){
        //构造器一:Date():创建一个对应当前时间的Date对象
        Date date1 = new Date();
        System.out.println(date1.toString());//Sat Feb 16 16:35:31 GMT+08:00 2019

        System.out.println(date1.getTime());//1550306204104

        //构造器二:创建指定毫秒数的Date对象
        Date date2 = new Date(155030620410L);
        System.out.println(date2.toString());

        //创建java.sql.Date对象
        java.sql.Date date3 = new java.sql.Date(35235325345L);
        System.out.println(date3);//1971-02-13

        //如何将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());

SimpleDateFormat类

Date类的API不易于国际化,大部分被废弃了,java.text.SimpleDateFormat 类是一个不与语言环境有关的方式来格式化和解析日期的具体类。
它允许进行格式化:日期->文本、解析:文本日期

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

  • SimpleDateFormat() :默认的模式和语言环境创建对象
  • public SimpleDateFormat(String pattern):该构造方法可以用参数pattern 指定的格式创建一个对象
  • public String format(Date date):方法格式化时间对象date

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

  • public Date parse(String source):从给定字符串的开始解析文本,以生成 一个日期。
//这里抛出的异常是防止出现SimpleDateFormat不能够识别的格式
    public void testSimpleDateFormat() throws ParseException {
        //1.实例化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-12-18 上午11:43";
        Date date1 = sdf.parse(str);
        System.out.println(date1);

        //2.按照指定的方式格式化和解析:调用带参的构造器
//        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-02-18 11:48:27
        //解析:要求字符串必须是符合SimpleDateFormat识别的格式(通过构造器参数体现),
        //否则,抛异常
        Date date2 = sdf1.parse("2020-02-18 11:48:27");
        System.out.println(date2);
    }

Calendar类

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

  • 获取Calendar实例的方法
    • 使用其静态方法Calendar.getInstance()
    • 调用它的子类GregorianCalendar的构造器
  • 一个Calendar的实例是系统时间的抽象表示,通过get(int field)方法来取得想 要的时间信息。比如YEAR、MONTH、DAY_OF_WEEK、 HOUR_OF_DAY 、 MINUTE、SECOND
  • public void set(int field,int value)
  • public int get(int field)
  • public void add(int field,int amount)
  • public final Date getTime()
  • public final void setTime(Date date)

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

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);
        System.out.println(calendar.get(Calendar.DAY_OF_YEAR));//获得当前是这一年的第几天

        //set()
        //calendar可变性,把本身更改了
        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,-3);//在当前-3天
        days = calendar.get(Calendar.DAY_OF_MONTH);
        System.out.println(days);

        //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);

    }

JDK 8中新日期时间API

JDK 1.0中包含了 一个java.util.Date类,但是它的大多数方法已经在JDK 1.1引入Calendar类之后被弃用 了。而Calendar并不比Date好多少。它们面临的问题是:
可变性:像日期和时间这样的类应该是不可变的。
偏移性:Date中的年份是从1900开始的,而月份都从0开始。
格式化:格式化只对Date有用,Calendar则不行。
此外,它们也不是线程安全的;不能处理闰秒等。

Java 8 吸收了 Joda-Time 的精华,以一个新的开始为 Java 创建优秀的 API。 新的 java.time 中包含了所有关于本地日期(LocalDate)、本地时间(LocalTime)、本地日期时间(LocalDateTime)、时区(ZonedDateTime) 和持续时间(Duration)的类。
历史悠久的 Date 类新增了 toInstant() 方法, 用于把 Date 转换成新的表示形式。这些新增的本地化时间日期 API 大大简 化了日期时间和本地化的管理。

LocalDate、LocalTime、LocalDateTime

概述

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

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

常用方法

  • 创建对象
    • now() / * now(ZoneId zone)
      静态方法,根据当前时间创建对象/指定时区的对象
    • of()
      静态方法,根据指定日期/时间创建对象
    • getDayOfMonth()/getDayOfYear()
      获得月份天数(1-31) /获得年份天数(1-366)
    • getDayOfWeek()
      获得星期几(返回一个 DayOfWeek 枚举值)
    • getMonth()
      获得月份, 返回一个 Month 枚举值
    • getMonthValue() / getYear()
      获得月份(1-12) /获得年份
    • getHour()/getMinute()/getSecond()
      获得当前对象对应的小时、分钟、秒
    • withDayOfMonth()/withDayOfYear()/withMonth()/withYear()
      将月份天数、年份天数、月份、年份修改为指定的值并返回新的对象
    • plusDays(), plusWeeks(), plusMonths(), plusYears(),plusHours()
      向当前对象添加几天、几周 、几个月、几年、几小时
    • minusMonths() / minusWeeks()/minusDays()/minusYears()/minusHours()
      从当前对象减去几月、几周、几天、几年、几小时
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(2020, 10, 6, 13, 23, 43);
    System.out.println(localDateTime1);


    //getXxx():获取相关的属性
    System.out.println(localDateTime.getDayOfMonth());  //月的第几天
    System.out.println(localDateTime.getDayOfWeek());   //周的第几天
    System.out.println(localDateTime.getMonth());   //哪个月,英文输出
    System.out.println(localDateTime.getMonthValue());  //那个月,数字输出
    System.out.println(localDateTime.getMinute());  //多少分钟

    //体现不可变性
    //withXxx():设置相关的属性
    LocalDate localDate1 = localDate.withDayOfMonth(22);
    System.out.println(localDate);      //本身没改
    System.out.println(localDate1);     //更改的传给了返回值


    LocalDateTime localDateTime2 = localDateTime.withHour(4);
    System.out.println(localDateTime);
    System.out.println(localDateTime2);

    //不可变性
    LocalDateTime localDateTime3 = localDateTime.plusMonths(3);
    System.out.println(localDateTime);  //本身没改
    System.out.println(localDateTime3); //更改的传给了返回值

    LocalDateTime localDateTime4 = localDateTime.minusDays(6);
    System.out.println(localDateTime);
    System.out.println(localDateTime4);
}

Instant

Instant:时间线上的一个瞬时点。
这可能被用来记录应用程序中的事件时间戳。

Instant的精度可以达到纳秒级。
(1 ns = 10-9 s) 1秒 = 1000毫秒 =106微秒=109纳秒

时间戳:
格林威治时间1970年01月01日00时00分00秒(北京时间1970年01月01 日08时00分00秒)起至现在的总秒数。

方法:

  • now()
    静态方法,返回默认UTC时区的Instant类的对象
  • ofEpochMilli(long epochMilli)
    静态方法,返回在1970-01-01 00:00:00基础上加上指定毫秒 数之后的Instant类的对象
  • atOffset(ZoneOffset offset)
    结合即时的偏移来创建一个 OffsetDateTime
  • toEpochMilli()
    返回1970-01-01 00:00:00到当前时间的毫秒数,即为时间戳
public void test2(){
    //now():获取本初子午线对应的标准时间
    Instant instant = Instant.now();
    System.out.println(instant);//2019-02-18T07:29:41.719Z

    //添加时间的偏移量
    OffsetDateTime offsetDateTime = instant.atOffset(ZoneOffset.ofHours(8));
    System.out.println(offsetDateTime);//2019-02-18T15:32:50.611+08:00

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

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

DateTimeFormatter

DateTimeFormatter:格式化或解析日期、时间

三种格式化方法:

  • 预定义的标准格式。如:
    ISO_LOCAL_DATE_TIME;
  • 本地化相关的格式。如:
    ofLocalizedDateTime(FormatStyle.LONG)
  • 自定义的格式。如:
    ofPattern(“yyyy-MM-dd hh:mm:ss”)
public void test3(){
//        方式一:预定义的标准格式。如: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);
        System.out.println(str1);//2019-02-18T15:42:18.797

        //解析:字符串 -->日期
        TemporalAccessor parse = formatter.parse("2019-02-18T15:42:18.797");
        System.out.println(parse);

//        方式二:
//        本地化相关的格式。如:ofLocalizedDateTime()
//        FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT :适用于LocalDateTime
        DateTimeFormatter formatter1 = DateTimeFormatter.ofLocalizedDateTime(FormatStyle.LONG);
        //格式化
        String str2 = formatter1.format(localDateTime);
        System.out.println(str2);//2019年2月18日 下午03时47分16秒


//      本地化相关的格式。如:ofLocalizedDate()
//      FormatStyle.FULL / FormatStyle.LONG / FormatStyle.MEDIUM / FormatStyle.SHORT : 适用于LocalDate
        DateTimeFormatter formatter2 = DateTimeFormatter.ofLocalizedDate(FormatStyle.MEDIUM);
        //格式化
        String str3 = formatter2.format(LocalDate.now());
        System.out.println(str3);//2019-2-18


//       重点: 方式三:自定义的格式。如:ofPattern(“yyyy-MM-dd hh:mm:ss”)
        DateTimeFormatter formatter3 = DateTimeFormatter.ofPattern("yyyy-MM-dd hh:mm:ss");
        //格式化
        String str4 = formatter3.format(LocalDateTime.now());
        System.out.println(str4);//2019-02-18 03:52:09

        //解析
        TemporalAccessor accessor = formatter3.parse("2019-02-18 03:52:09");
        System.out.println(accessor);

    }

其它类

  • 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 的实现。

Java比较器

Java中的对象,正常情况下,只能进行比较:== 或 != ,不能使用 > 或 <的
但是在开发场景中,我们需要对多个对象进行排序,需要比较对象的大小。

Java实现对象排序的方式有两种:
自然排序:java.lang.Comparable
定制排序:java.util.Comparator

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

  • Comparable接口的方式一旦一定,能够保证Comparable接口实现类的对象在任何位置都可以比较大小。
  • Comparator接口属于临时性(匿名内部类)的比较,需要的时候定义。

Comparable接口

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

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

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

  3. 重写compareTo(obj)的规则:
    如果当前对象this大于形参对象obj,则返回正整数,
    如果当前对象this小于形参对象obj,则返回负整数,
    如果当前对象this等于形参对象obj,则返回零。

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

public void test1(){
    String[] arr = new String[]{"AA","CC","KK","MM","GG","JJ","DD"};
    //
    Arrays.sort(arr);

    System.out.println(Arrays.toString(arr));

}

//对几个商品进行排序(自定义的类要实现接口并重写方法)
//Goods implements  Comparable

//@Override	public int compareTo(Object o)

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接口

Comparator接口的使用:定制排序

  1. 背景:
    当元素的类型没有实现java.lang.Comparable接口 而又 不方便修改代码
    或者实现了java.lang.Comparable接口的排序规则不适合当前的操作,
    那么可以考虑使用 Comparator 的对象来排序
  2. 重写compare(Object o1,Object o2)方法,比较o1和o2的大小:
    如果方法返回正整数,则表示o1大于o2;
    如果返回0,表示相等;
    返回负整数,表示o1小于o2。
    public void test3(){
        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));
    }

    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("huaweiMouse",224);
        arr[5] = new Goods("microsoftMouse",43);

        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的,所以无法创建该类的对象
    不,所以也可以很方便的进行调用。

  • 成员变量

    • 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-20210219100816064

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) 角度—>弧度

包装类

针对八种基本数据类型定义相应的引用类型—包装类(封装类)
有了类的特点,就可以调用类中的方法,Java才是真正的面向对象

image-20210219103152894

image-20210219113545941

基本类型 -> 包装类

public void test1(){
      
   
      int num1 = 10;
//    System.out.println(num1.toString());
      Integer in1 = new Integer(num1);
      System.out.println(in1.toString());
      
      Integer in2 = new Integer("123");
      System.out.println(in2.toString());
      
      //报异常
//    Integer in3 = new Integer("123abc"); 不是纯粹的数
//    System.out.println(in3.toString());
      
   
   
      Float f1 = new Float(12.3f);
      Float f2 = new Float("12.3");
      System.out.println(f1);
      System.out.println(f2);
      
   
   
      Boolean b1 = new Boolean(true);
      Boolean b2 = new Boolean("TrUe");
      System.out.println(b2);
      Boolean b3 = new Boolean("true123");
      System.out.println(b3);//false
      
      
      Order order = new Order();
      System.out.println(order.isMale);//基本数据类型中的默认false
      System.out.println(order.isFemale);//是引用数据类型了,默认为null
   }
class Order{
	
	boolean isMale;
	Boolean isFemale;
}

包装类 -> 基本类型

类的对象无法做一些基本运算,但是转换为基本类型之后就可以进行运算了

调用包装类Xxx的xxxValue()

@Test
public void test2(){
   Integer in1 = new Integer(12);
   
   int i1 = in1.intValue();
   System.out.println(i1 + 1);
   
   
   Float f1 = new Float(12.3);
   float f2 = f1.floatValue();
   System.out.println(f2 + 1);
}

自动装箱和自动拆箱

自动装箱:
基本数据类型 --> 包装类

int num2 = 10;
Integer in1 = num2;//自动装箱

boolean b1 = true;
Boolean b2 = b1;//自动装箱

自动拆箱:
包装类 --> 基本数据类型

int num3 = in1;//自动拆箱

基本数据类型、包装类--->String类型

​ 调用String重载的valueOf(Xxx xxx)

@Test
public void test4(){
   
   int num1 = 10;
   //方式1:连接运算
   String str1 = num1 + "";
   
   //方式2:调用String的valueOf(Xxx xxx)
   	//基本数据类型
   float f1 = 12.3f;
   String str2 = String.valueOf(f1);//"12.3"
   	//包装类
   Double d1 = new Double(12.4);
   String str3 = String.valueOf(d1);//"12.4"
   
}

String类型--->基本数据类型、包装类

调用包装类的parseXxx(String s)

public void test5(){
      String str1 = "123";
      //错误的情况:
//    int num1 = (int)str1;
//    Integer in1 = (Integer)str1;  String 和 Integer不存在继承关系,不能强转
   
      //如果字符串中含有非数字的字符,会报NumberFormatException,
      int num2 = Integer.parseInt(str1);
      System.out.println(num2 + 1);
      
      String str2 = "true1";
      boolean b1 = Boolean.parseBoolean(str2);
      System.out.println(b1);
   }

练习

【题1:】

   public void test1() {
      Object o1 = true ? new Integer(1) : new Double(2.0); //编译的时候将new的对象的类型进行统一成Double
      System.out.println(o1);// 1.0

   }

【题2:】

   public void test2() {
      Object o2;
      if (true)
         o2 = new Integer(1);
      else
         o2 = new Double(2.0);
      //这里并没有要求同一类型
      System.out.println(o2);// 1

   }

【题3:】

   public void test3() {
      Integer i = new Integer(1);
      Integer j = new Integer(1);
      System.out.println(i == j);//false,不同对象,地址不同
      
      //Integer内部定义了IntegerCache结构,IntegerCache中定义了Integer[],
      //其中保存了从-128~127范围的整数。如果我们使用自动装箱的方式,给Integer赋值的范围在
      //-128~127范围内时,可以直接使用数组中的元素,不用再去new了。目的:提高效率,节省缓存
      
      Integer m = 1;
      Integer n = 1;
      System.out.println(m == n);//true,用的是数组中的同一个值

      Integer x = 128;//相当于new了一个Integer对象
      Integer y = 128;//相当于new了一个Integer对象
      System.out.println(x == y);//false
   }

BigInteger与BigDecimal

BigInteger

  • 对应整型

  • Integer类作为int的包装类,能存储的最大整型值为231-1
    Long类也是有限的, 最大为263-1。
    如果要表示再大的整数,不管是基本数据类型还是他们的包装类 都无能为力,更不用说进行运算了。

  • java.math包的BigInteger可以表示不可变的任意精度的整数。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) :返回其值为 (thisexponent) 的 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)

枚举类

当类的对象只有有限个,确定的。
当需要定义一组常量时,强烈建议使用枚举类

  • 枚举类的实现

    • JDK5.0之前需要自定义枚举类
    • JDK5.0新增的 enum 关键字用于定义枚举类
  • 若枚举只有一个对象, 则可以作为一种单例模式的实现方 式。

  • 枚举类的属性

    • 枚举类对象的属性不应允许被改动, 所以应该使用 private final 修饰
    • 枚举类的使用 private final 修饰的属性应该在构造器中为其赋值
    • 若枚举类显式的定义了带参数的构造器, 则在列出枚举值时也必须对应的 传入参数

自定义枚举类

  1. 私有化类的构造器,保证不能在类的外部创建其对象
  2. 在类的内部创建枚举类的实例。声明为:public static final
  3. 对象如果有实例变量,应该声明为private final,并在构造器中初始化
/* 三、Enum类中的常用方法:
 *    values()方法:返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值。
 *    valueOf(String str):可以把一个字符串转为对应的枚举类对象。要求字符串必须是枚举类对象的“名字”。如不是,会有运行时异常:IllegalArgumentException。
 *    toString():返回当前枚举类对象常量的名称
 *
 * 四、使用enum关键字定义的枚举类实现接口的情况
 *   情况一:实现接口,在enum类中实现抽象方法
 *   情况二:让枚举类的对象分别实现接口中的抽象方法
 */
public class SeasonTest {

    public static void main(String[] args) {
        Season spring = Season.SPRING;
        System.out.println(spring);

    }

}

//自定义枚举类
class Season{
    //1.声明Season对象的属性:private final修饰
    private final String seasonName;
    private final String seasonDesc;

    //2.私有化类的构造器,并给对象属性赋值
    private Season(String seasonName,String seasonDesc){
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }

    //3.提供当前枚举类的多个对象:public static final的
    public static final Season SPRING = new Season("春天","春暖花开");
    public static final Season SUMMER = new Season("夏天","夏日炎炎");
    public static final Season AUTUMN = new Season("秋天","秋高气爽");
    public static final Season WINTER = new Season("冬天","冰天雪地");

    //4.其他诉求1:获取枚举类对象的属性
    public String getSeasonName() {
        return seasonName;
    }

    public String getSeasonDesc() {
        return seasonDesc;
    }
    //4.其他诉求2:提供toString()
    @Override
    public String toString() {
        return "Season{" +
                "seasonName='" + seasonName + '\'' +
                ", seasonDesc='" + seasonDesc + '\'' +
                '}';
    }
}

用enum关键字定义枚举类

  • 使用说明
    • 使用 enum 定义的枚举类默认继承了 java.lang.Enum类,因此不能再 继承其他类
    • 枚举类的构造器只能使用 private 权限修饰符
    • 枚举类的所有实例必须在枚举类中显式列出(, 分隔 ; 结尾)。
      列出的 实例系统 会自动添加 public static final 修饰
    • 必须在枚举类的第一行声明枚举类对象
  • JDK 1.5 中可以在 switch 表达式中使用Enum定义的枚举类的对象 作为表达式, case 子句可以直接使用枚举值的名字, 无需添加枚举 类作为限定。
public class SeasonTest1 {
    public static void main(String[] args) {
        SeasonEnum summer = SeasonEnum.SUMMER;
        //toString():返回枚举类对象的名称
        System.out.println(summer); //直接输出的话是常量名SUMMER

    }
}

//使用enum关键字枚举类
enum SeasonEnum {
  
    //1.提供当前枚举类的对象,多个对象之间用","隔开,末尾对象";"结束
    SPRING("春天","春暖花开")
    SUMMER("夏天","夏日炎炎")
    AUTUMN("秋天","秋高气爽")
    WINTER("冬天","冰天雪地")

    //2.声明Season对象的属性:private final修饰
    private final String seasonName;
    private final String seasonDesc;

    //3.私有化类的构造器,并给对象属性赋值
    private SeasonEnum(String seasonName,String seasonDesc){
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }

枚举类常用方法

image-20210219132303230
//values():返回所有的枚举类对象构成的数组
        SeasonEnum[] values = SeasonEnum.values();
        for(int i = 0;i < values.length;i++){
            System.out.println(values[i]);
        }

//valueOf(String objName):返回枚举类中对象名是objName的对象,相当于 字符串 转成了 枚举类对象。
        SeasonEnum winter = SeasonEnum.valueOf("WINTER");
	//如果没有objName的枚举类对象,则抛异常: IllegalArgumentException
//        Season1 winter = Season1.valueOf("WINTER1");
        System.out.println(winter);
        winter.show();

实现接口的枚举类

  • 和普通 Java 类一样,枚举类可以实现一个或多个接口
  • 若每个枚举值在调用实现的接口方法呈现相同的行为方式,则只 要统一实现该方法即可。
  • 若需要每个枚举值在调用实现的接口方法呈现出不同的行为方式, 则可以让每个枚举值分别来实现该方法
posted @ 2021-02-19 13:32  feifan666  阅读(43)  评论(0)    收藏  举报