1.Java8新特性

1)默认方法:
    默认方法就是一个在接口里面有了一个实现的方法    
2)方法引用:
    方法引用提供了非常有用的语法,可以直接引用已有Java类或对象(实例)的方法或构造器。
    与lambda联合使用,可以使语言的构造更紧凑简洁,减少冗余代码。
3)Lambda表达式:
    Lambda允许把函数作为一个方法的参数(函数作为参数传递进方法中。
4)Stream API:
    新添加的Stream API(java.util.stream) 把真正的函数式编程风格引入到Java中。
5)Date Time API:
    加强对日期与时间的处理
6)Optional类:
    Optional类已经成为Java 8 类库的一部分,用来解决空指针异常。

1.1 关于接口

1.1.1 原始接口

接口定义

接口的作用:定义一套标准、规范,一个公共的规则,所以接口中的之声明方法签名,不提供方法体,具体实现由实现类完成。

接口中的属性:public static final,即只能是常量

接口中的方法:public abstract,即方法必须为抽象方法

接口没有构造方法,所以不能实例化

接口可以通过extends继承其他接口

接口的实现必须实现接口所有方法,否则该类应该抽象类

 

接口的作用

1)多继承,java对象只能有一个父类,通过接口可以实现多继承

2)有利于代码规范,定义接口,可以让不同的开发人员按照统一规范来完成项目的开发工作,防止开发人员随意命名,导致命名不清晰,代码混乱等

3)降低项目的耦合度,提高可扩展性。一个类被多个地方调用,当其中一个地方发生业务改变时,通过定义一个新的实现类来解决即可,而避免改变所有调用的地方。

1.1.2 接口新特性(1.8)

接口的默认方法

接口中特殊的方法,被default修饰,可以包含方法体

public interface IHello {
    void sayHi();
    default void sayBye() { }
  }

作用:扩展接口功能,不破坏现有代码,让接口扩展新的功能。默认方法跟接口中其他方法不同,它是可选的。子类可以根据需求选择是否需要重写该方法。

注意:若一个类实现了多个接口,那么多个接口中不应该包含相同的默认方法,编辑会报错,除非重写该默认方法。

接口中的静态方法

等同于类中的静态方法,调用时可不需要实现接口或实例化接口的实例,来完成方法的调用。

public interface IHello {
    static void sayHello() {}
  }

1.2 Lambda表达式

1.2.1 函数式接口

a.函数式接口

有且仅包含一个抽象方法的接口(可以包含一个或多个静态或默认方法)。

1.8之前: 
   java.lang.Runnable java.util.concurrent.Callable java.util.Comparator java.io.FileFilter 
1.8之后: 
   java.util.function、Predicate<T> 、Supplier<T>

b.使用场景

Lambda表达式,使用Lambda表达式可以简化匿名内部类的实现

@FunctionalInterface注解:
    该注解来定义接口,编译器强制检查接口中是否有且仅有一个抽象方法,如果不是,则编译出错

1.2.2 Lambda表达式

Lambda 允许把函数作为一个方法的参数或返回值(函数作为参数传递进方法中),使用 Lambda 表达式可以使代码变的更加简洁紧凑。

语法: 
   (parameters) -> expression 或 (parameters) ->{ statements; } 
简化版:
   ()-> {} ()代表参数,{}代表方法体,->代表运算符 
注意: 
   1) 所有参数类型,均可以省略
   2) 当参数只有一个,小括号可以省略 
   3) 无返回值情况,当方法体只有一条语句,可以省略大括号 
   4) 有返回值情况,当方法体只有一条语句,可以省略大括号和return

示例:

1)无参无返回

public interface IHello {
    void sayHello();
}
IHello iHello = () -> {System.out.println("");};

2)单参,无返回 

public interface IHello2 {
     void sayBye(String a);
}
IHello2 iHello2 = (a) ->{System.out.println("a:" + a);};

 3)多参,无返回

public interface IHello3 {
    void sayHi(int a,int b);
}
IHello3 iHello3 = (a,b)->{ System.out.println(a + b);};

4)无参数,有返回 

public interface IHello4 {
    int sayHi();
}
IHello4 iHello4 = () -> {return 1;};

5)单参数,有返回 

public interface IHello5 {
    int sayHi(int a);
}
IHello5 iHello5 = (a) ->{return a*10;};

6)多参数,有返回值 

public interface IHello6 {
    int sayHi(int a,int b);
}
IHello6 iHello6 = (a,b) -> {return a+ b;};

1.2.3 Lambda表达式的特殊使用

a.方法引用

Lambda表达式可以直接指向一个已经实现的方法

要求:①:该方法的参数数量和参数类型必须跟接口中的方法一致 ②:该方法的返回值类型必须与接口中的一致

public interface IHello5 {
    int sayHi(int a);
}
public static int change(int a) {
    return a;
}
IHello5 iHello5 = a -> change(a);

b.双冒号(::)的使用

普通方法:(参数是某个对象,调用该对象的方法)
public interface IPerson {
    int getPerson(Person person);
}
原装版:IPerson iPerson = (person) -> {return person.getAge();};
简化版:IPerson iPerson = Person::getAge;
注意:参数对象调用的方法必需是无参
构造方法:(返回值是某个对象)
public interface IPerson {
    Person getPerson();
}
原装版:IPerson iPerson = () -> {return new Person();};
简化版:IPerson iPerson = Person::new;

注意:调用的为无参构造

1.2.4 系统内置函数式接口

Consume:参数为T,无返回
public interface Consumer<T> {
    void accept(T t);
}

Supplier:无参有返回
public interface Supplier<T> {
    T get();
}

Predicate:泛型参数T,返回值boolean
public interface Predicate<T> {
    boolean test(T t);
}

Function:参数为T,返回值为R
public interface Function<T, R> {
    R apply(T t);
}

1.3 StreamAPI

1.3.1 什么是Stream

Stream:中文成为"流",一种对集合或者数组进行处理的工具,又被称为"操作符";操作符分类两类:中间操作符、终止操作符。一般来讲,流不会修改原来的数据,它会将操作后的数据保存到另一个对象中。
Stream使用步骤:
1)创建Stream对象
2)使用中间操作符进行操作
3)使用终止操作符

1.3.2 创建stream对象

stream对象的获取,可以分为数组、list、set、map(可以得到key的set集合、value的list集合)

数组

String[] strArray = new String[] {"apple", "banana", "orange", "waltermaleon", "grape"};

Stream.of(strArray)//第一种
Arrays.stream(strArray)//第二种

list、set

List<Student> stuList = new ArrayList<Student>();
stuList.stream();//串行
stuList.parallelStream();//并行

Set<Student> set = new HashSet<>();
set.stream();
set.parallelStream();

1.3.2 中间操作符

对数据执行操作后,数据依然可以传递给下一级操作符。(一般都需要传递给终止操作符,因为中间操作符返回值为Stream对象,不传递给终止操作符返回处理后的数据没有意义)

中间操作符主要分为8种:

转换操作符:
    把原始数据转换为你所需要的数据,默认可转换成int、long、double类型 map,mapToInt,mapToLong,mapToDouble 
拍平操作符:
    即拆开,把数组拆分成单个字段,默认提供了拍平成int,long,double的操作符.
    flatmap,flatmapToInt,flatmapToLong,flatmapToDouble                                              
限流操作符:
    比如数据流中有10条数据,只取其中三条 limit 
去重操作符:
    去掉重复数据 distinct 
过滤操作符:
    筛掉不想要的数据 filter 
跳过操作符:
    跳过某些元素 skip 
挑出操作符:
    对数据进行某些操作,如读取、编辑修改等 peek 
排序操作符: 
    sorted

map转换操作符

String[] strArray = new String[] {"apple", "banana", "orange", "waltermaleon", "grape"};
Stream.of(strArray)
    .map(e -> e.length())
    .forEach(e -> System.out.println(e));

flatMap拍平操作符

Stream.of("a-b-c-d","e-f-i-g-h")
     .flatMap(e->Stream.of(e.split("-")))
     .forEach(e->System.out.println(e));

limit限流操作符

Stream.of(1,2,3,4,5,6)
    .limit(3) //限制三个
    .forEach(e->System.out.println(e));

distinct去重操作符

Stream.of(1,2,3,1,2,5,6,7,8,0,0,1,2,3,1)
    .distinct() //去重
    .forEach(e->System.out.println(e));

filter过滤操作符

Stream.of(1,2,3,1,2,5,6,7,8,0,0,1,2,3,1)
     .filter(e->e>=5) //过滤小于5的
     .forEach(e->System.out.println(e));

skip跳过操作符

Stream.of(1,2,3,4,5,6,7,8,9)
    .skip(4) //跳过前四个
    .forEach(e->System.out.println(e));

peek挑出操作符

User w = new User("w",10);
User x = new User("x",11);
User y = new User("y",12);
Stream.of(w,x,y)
    .peek(e->{e.setName(e.getAge()+e.getName());}) //重新设置名字 变成 年龄+名字
    .forEach(e->System.out.println(e.toString()));

sorted排序操作符

Stream.of(2,1,3,6,4,9,6,8,0) 
    .sorted()  //默认排序
    .forEach(e->System.out.println(e));

User x = new User("x",11);
User y = new User("y",12);
User w = new User("w",10);
Stream.of(w,x,y)
    .sorted((e1,e2)->e1.age>e2.age?1:e1.age==e2.age?0:-1)
    .forEach(e->System.out.println(e.toString()));
    
注意:多重三元运算符(三目运算符)嵌套
String str =  a.equals("123") ? "123" : ( b.equals("456") ? "456" : "789");
//如果a等于123,就给str赋值123;否则,如果b等于456,就给str赋值123,前面两个如果都不成立就赋值789

 

1.3.3 终止操作符

中间操作符处理数据后,须有终止操作符进行收集、消费操作。终止操作符只能用一次。

收集操作符:将数据收集起来
    collect
统计操作符:
    count
查找操作符:
    findFirst,findAny
匹配操作符:流中是否存在符合条件的元素,返回boolean值
    noneMatch、allMatch、anyMatch
最值操作符:
    max,min
遍历操作符:
    forEach
数组操作符:将数据流元素变为数组
    toArray
规约操作符:将整个数据流规约为一个值,count/min/max都属于规约
    reduce

collect收集操作符

Set<String> stringSet = 
    Stream.of("apple", "banana", "orange", "waltermaleon", "grape")
    .collect(Collectors.toSet()) //set 容器

count统计操作符

long count = 
    Stream.of("apple", "banana", "orange", "waltermaleon", "grape")
    .count();

查找操作符

findFirst:取流中的第一个元素
    Optional<String> stringOptional = 
        Stream.of("apple", "banana", "orange", "waltermaleon", "grape")
        .findFirst();
    stringOptional.get();//得到第一个元素


findAny:获取流中任意一个元素
    Optional<String> findFirst = 
        Stream.of("apple", "banana", "orange", "waltermaleon", "grape")
        .findAny();
     stringOptional.get();//得到任意一个元素,实际还是第一个

匹配操作符

boolean result = Stream.of("aa","bb","cc","aa")
                .noneMatch(e->e.equals("aa"));
noneMatch:没有一个匹配返回true
allMatch:全部匹配返回true
anyMatch:任意一个匹配返回true

最值操作符:

Optional<Integer> integerOptional = 
    Stream.of(0,9,8,4,5,6,-1)
    .max((e1,e2)->e1.compareTo(e2));//max元素中最大的
    .min((e1,e2)->e1.compareTo(e2));//min元素中最大的

reduce规约操作符

所有的元素归约成一个,比如对所有元素求和,乘等
int sum = Stream.of(0,9,8,4,5,6,-1)
    .reduce(0,(e1,e2)->e1+e2);

toArray 数组操作符

转成数组
Object[] objects=Stream.of(0,2,6,5,4,9,8,-1)
                .toArray();

 

1.4 日期时间API

1.4.1 新旧版本的区别

旧版本:
    1)线程不安全
    2)设计较差,java.util、java.sql都有日期类,而格式化类却在java.text包下,设计不合理
    3)时区处理麻烦:日期类不提供国际化,没有时区支持
新版本:(java.time包)
    1)Local(本地):简化了日期时间的处理,没有时区问题
    2)Zoned(时区):通过制定时区处理日期时间问题

1.4.2 本地化日期时间API

主要涉及的类为:LocalDate、LocalTime、LocalDateTime

方法

描述

now()

静态方法,根据当前时间创建对象

of()

静态方法,根据指定日期/时间创建 对象

plusDays, plusWeeks, plusMonths, plusYears

向当前 LocalDate 对象添加几天、 几周、几个月、几年

minusDays, minusWeeks, minusMonths, minusYears

从当前 LocalDate 对象减去几天、 几周、几个月、几年

plus, minus

添加或减少一个 Duration或 Period

withDayOfMonth, withDayOfYear, withMonth, withYear

将月份天数、年份天数、月份、年 份修改为指定的值并返回新的 LocalDate对象

getDayOfMonth

获得月份天数(1-31)

getDayOfYear

获得年份天数(1-366)

getDayOfWeek

获得星期几(返回一个 DayOfWeek 枚举值)

getMonth

获得月份, 返回一个 Month枚举值

getMonthValue

获得月份(1-12)

getYear

获得年份

until

获得两个日期之间的 Period 对象, 或者指定 ChronoUnits的数字

isBefore, isAfter

比较两个 LocalDate

isLeapYear

判断是否是闰年

 

******对象创建******
LocalDate localDate = LocalDate.now();//获取当前日期
LocalDate localDate = LocalDate.of(year, month, dayOfMonth);//获取特定日期

******获取年、月、日信息******
int year = localDate.getYear();//年
int monthValue = localDate.getMonthValue();//月
int dayOfMonth = localDate.getDayOfMonth();//日
int dayOfYear  = localDate.getDayOfYear();//一年中的第几天
int dayOfWeek = localDate.getDayOfWeek();//周几

******枚举对象ChronoField获取年、月、日信息******
int year = localDate.get(ChronoField.YEAR);
int month = localDate.get(ChronoField.MONTH_OF_YEAR);
int day = localDate.get(ChronoField.DAY_OF_MONTH);

******日期修改******
LocalDate date = localDate.withYear(year);//变为指定年
LocalDate date = localDate.withMonth(month);//变为指定月
LocalDate date = localDate.withDayOfMonth(dayOfMonth);//变为指定日
//日期添加
LocalDate date = localDate.plus(amountToAdd, unit);
LocalDate date = localDate.plusYears(yearsToAdd);
LocalDate date = localDate.plusMonths(monthsToAdd);
LocalDate date = localDate.plusWeeks(weeksToAdd);
LocalDate date = localDate.plusDays(daysToAdd)
//日期减少
LocalDate date = localDate.minnus(amountToAdd, unit);
LocalDate date = localDate.minnusYears(yearsToAdd);
LocalDate date = localDate.minnusMonths(monthsToAdd);
LocalDate date = localDate.minnusWeeks(weeksToAdd);
LocalDate date = localDate.minnusDays(daysToAdd)

******日期比较******
localDate.isAfter(otherDate)
localDate.isBefore(otherDate)

******其他方法******
int len = localDate.lengthOfMonth();//31(这个月有多少天)
boolean leap = localDate.isLeapYear();//false(是不是闰年)
localDate.toEpochDay() - specialDay.toEpochDay();//日期间隔天数

DateTimeFormatter

//解析日期
String dateStr= "2018年12月18日";
DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy年MM月dd日");
LocalDate date= LocalDate.parse(dateStr, formatter);

//日期转换为字符串
LocalDateTime now = LocalDateTime.now();
DateTimeFormatter format = DateTimeFormatter.ofPattern("yyyy年MM月dd日 hh:mm a");
String nowStr = now .format(format);
System.out.println(nowStr);