• JDK8新特性


    转自:jdk8新特性

    一、Lambda表达式

    1、语法:完整的Lambda表达式由三部分组成:参数列表、箭头、声明语句

    (Type1 param1, Type2 param2, ..., TypeN paramN) -> { statment1; statment2; //............. return statmentM;}

    2、绝大多数情况,编译器都可以从上下文环境中推断出lambda表达式的参数类型,所以参数可以省略

    (param1,param2, ..., paramN) -> { statment1; statment2; //............. return statmentM;}

    3、当lambda表达式的参数个数只有一个,可以省略小括号

    param1 -> { statment1; statment2; //............. return statmentM;}

    4、当lambda表达式只包含一条语句时,可以省略大括号、return和语句结尾的分号

    param1 -> statment

    展示例子:

    // bean类
    import lombok.AllArgsConstructor;
    import lombok.Data;
    import lombok.NoArgsConstructor;
    import lombok.ToString;
    
    @Data
    @ToString
    @NoArgsConstructor
    @AllArgsConstructor
    public class Student {
        //名字
        private String name;
        //性别
        private String sex;
        //薪水
        private int salary;
        //年龄
        private int age;
        //星座
        private String star;
    
        public static void printStuMethod(Student stu) {
            System.out.println("get student, name is: " + stu.getName());
        }
    }
    
    // 构造bean list对象
    List<Student> list = Arrays.asList(
        new Student("张三","男",5000,18,"天秤座"),
        new Student("李四","男",4000,16,"双鱼座"),
        new Student("王五","男",3000,24,"水瓶座")
    );
    • list的遍历
      // 标准写法
      list.stream().forEach((Student x) -> {System.out.println(x);});
      // 简略写法
      list.stream().forEach(x -> System.out.println(x));
      // 双冒号简略写法
      list.stream().forEach(System.out::println);
      // 双冒号调用类内部静态方法
      list.stream().forEach(Student::printStuMethod);
      // 结合stream的其他方法
      list.stream().filter(e -> e.getStar().equals("天秤座")).forEach(System.out::println);
    • map的遍历
      // 构造map对象
      Map<String, Student> map = Maps.uniqueIndex(list, new Function<Student, String>() {
          @Override
          public String apply(Student person) {
              return person.getName();
          }
      });
      // 输出方法
      public static void printMap(String key, Student value){
          System.out.println(key + " " + value);
      }
      
      // 标准写法输出
      map.forEach( (k,v) -> System.out.println(k+" "+v) );
      // 调用内部方法输出
      map.forEach( (k,v) -> printMap(k, v) );
    • set的遍历
      Set<String> sets = Sets.newHashSet("1", "2", "3");
      // 输出方法
      public static void printSet(String set){
          System.out.println(set);
      }
      //  写法一
      sets.stream().forEach(System.out::println);
      // 写法二,调用内部方法
      sets.stream().forEach(x -> printSet(x));
    • 带有返回值的 lambda 用法
      标准实例
      // 定义接口
      public interface Calculator {
          //定义一个计算两个int整数和的方法并返回结果
          public abstract int calc(int a,int b);
      }
      // 调用
      public class DemoCalculator {
          public static void main(String[] args) {
              //调用invokeCalc方法,方法的参数是一个接口可以使用匿名内部类
              invokeCalc(5, 10, new Calculator() {
                  @Override
                  public int calc(int a, int b) {
                      return a + b;
                  }
              });
       
              //使用Lambda表达式简化匿名内部类的书写
              invokeCalc(10,20,(int a,int b)->{
                  return a + b;
              });
       
              //优化省略Lambda表达式
              invokeCalc(10,20,(a,b)-> a + b);
          }
       
          public static void invokeCalc(int a,int b,Calculator c){
              int sum = c.calc(a,b);
              System.out.println(sum);
          }
      }

      二、方法调用(双冒号)的用法

      方法调用实例

      方法调用的五种形式

      • 对象实例::实例方法名
      • 对象实例名::实例方法名
      • 类名::静态方法名
      • 类名::实例方法名
      • 类名::new

      JDK8中增加了一个新的包:java.util.function,它里面包含了常用的函数式接口,例如:

      • Predicate<T>:接收 T 并返回 boolean
      • Consumer<T>:接收 T,不返回值
      • Function<T, R>:接收 T,返回 R
      • Supplier<T>:提供 T 对象(例如工厂),不接收值
      • UnaryOperator<T>:接收 T 对象,返回 T
      • BinaryOperator<T>:接收两个 T,返回 T
        import java.util.function.BiPredicate;
        import java.util.function.Consumer;
        import java.util.function.Supplier;
        
        public class MethodReference {
        
            public MethodReference(String s) {
                System.out.println("print ==> " + s);
            }
        
            public static void main(String[] args) {
                method1();
                method2();
                method3();
                method4();
                method5();
            }
        
            // 1、实例对象::实例方法名
            public static void method1(){
                Student student = new Student("ZhangSan",23);
        
                Supplier<String> supplier1 = () -> student.getName();
                System.out.println("Lambda形式: "+supplier1.get());
        
                Supplier<String> supplier2 = student::getName ;
                System.out.println("方法引用形式: "+supplier2.get());
            }
        
            // 2、实例对象名::实例方法名
            public static void method2(){
                //传统的lambda表达式
                Consumer<String> consumer = (x) -> System.out.println(x);
                consumer.accept("Hi: 我是Lambda表达式实现的!"); //打印:Hi: 我是Lambda表达式实现的!
        
                //方法引用实现
                consumer = System.out::println;
                consumer.accept("Hello : ZhangSan,我是使用方法引用实现的 ");
            }
        
            // 3、类名::静态方法名
            public static void method3(){
                //传统的lambda表达式
                Consumer<String> consumer = (str) -> MethodReference.sayName(str);
                consumer.accept("Hello : XiangYang");
        
                //方法引用实现
                consumer = MethodReference::sayName;
                consumer.accept("Hello : XiangYang");
            }
        
            public static void sayName(String name){
                System.out.println(name);
            }
        
            // 4、类名::实例方法名
            // 参数列表的第一个参数是实例方法的调用者, 第n>1个参数是实例方法的参数时进行使用
            public static void method4(){
                BiPredicate<String,String> biPredicate = (x , y) -> x.equals(y);
                boolean test = biPredicate.test("hello","hi");
                System.out.println(test);
        
                biPredicate = String::equals;
                test = biPredicate.test("hello","hello");
                System.out.println(test);
            }
        
            // 5、类名::new
            public static void method5(){
        
                Consumer<String> consumer = MethodReference::new;
                consumer.accept("hello");
            }
        }

        三、Stream函数式操作流元素集合

        父类:BasicStream
        子类:Stream、IntStream、LongStream、DoubleStream
        包含两个类型,中间操作(intermediate operations)和结束操作(terminal operations)
        下面是所有方法的属于那一端操作的方法:

        下面测试一下map、flatmap、reduce三个方法,其它方法的用法请参考:
        https://my.oschina.net/mdxlcj/blog/1622718

        先准备一个list

        public static List<Student> list = Arrays.asList(
                    new Student("张三", 12),
                    new Student("李四", 15),
                    new Student("王五", 21)
        );

        map:将List<Student> 转换为List<String>, collect是将结果转换为List

        List<String> names = list.stream().map(Student::getName).collect(Collectors.toList());
        names.stream().forEach(System.out::println);

        mapToInt:转换数值流,等同mapToLong、mapToDouble,如下这个是取最大值

        IntStream intStream = list.stream().mapToInt(Student::getAge);
        Stream<Integer> integerStream = intStream.boxed();
        Optional<Integer> max = integerStream.max(Integer::compareTo);
        System.out.println(max.get());

        flatMap:将流中的每一个元素 T 映射为一个流,再把每一个流连接成为一个流

        List<String> list2 = new ArrayList<>();
        list2.add("aaa bbb ccc");
        list2.add("ddd eee fff");
        list2.add("ggg hhh iii");
        list2 = list2.stream().map(s -> s.split(" ")).flatMap(Arrays::stream).collect(Collectors.toList());
        System.out.println(list2);
        reduce reduce(accumulator) :参数是一个执行双目运算的 Functional Interface,假如这个参数表示的操作为 op,stream 中的元素为 x, y, z, …,则 reduce() 执行的就是 x op y op z ...,所以要求 op 这个操作具有结合性(associative),即满足: (x op y) op z = x op (y op z),满足这个要求的操作主要有:求和、求积、求最大值、求最小值、字符串连接、集合并集和交集等。另外,该函数的返回值是 Optional 的:
        Optional <integer>sum1 = numStream.reduce((x, y) -> x + y);
        reduce(identity, accumulator) :可以认为第一个参数为默认值,但需要满足 identity op x = x,所以对于求和操作,identity 的值为 0,对于求积操作,identity 的值为 1。返回值类型是 stream 元素的类型:Integer sum2 = numStream.reduce(0, Integer::sum);
        reduce 如果不加参数identity则返回的是 optional 类型的,reduce 在进行双目运算时,其中一个场景是与identity做比较操作,因此我们应该满足identity op x = x

        Stream<Integer> stream = Arrays.stream(new Integer[]{1, 2, 3, 4, 5, 6, 7, 8});
        
        //求集合元素之和
        Integer result = stream.reduce(0, Integer::sum);
        System.out.println(result);
        
        //求和
        stream.reduce((i, j) -> i + j).ifPresent(System.out::println);
        
        //求最大值
        stream.reduce(Integer::max).ifPresent(System.out::println);
        
        //求最小值
        stream.reduce(Integer::min).ifPresent(System.out::println);
        
        //做逻辑
        stream.reduce((i, j) -> i > j ? j : i).ifPresent(System.out::println);
        
        //求逻辑求乘积
        int result2 = stream.filter(i -> i % 2 == 0).reduce(1, (i, j) -> i * j);
        
        Optional.of(result2).ifPresent(System.out::println);

        四 、接口新增:默认方法与静态方法

        在JDK8之前,Interface之中是可以定义变量和方法的,变量必须是public static final 的 , 方法必须是public abstract的 , 由于这些修饰符是默认的,所以在JDK8之前,下面写法是等价的

        public interface JDK8BeforeInterface {
        
            public static final int field1 = 0;
        
            int field2 = 0;
        
            public abstract void m1(int a) throws Exception ;
        
            void m2(int a) throws Exception;
        
        }

        JDK8及以后,允许我们在接口中定义static方法和default方法。

        public interface  JDK8Interface {
        
            // static修饰符定义静态方法
            static void staticMethod() {
                System.out.println("接口中定义静态方法");
            }
            // default修饰符定义默认方法
            default void defaultMethod() {
                System.out.println("接口中的默认方法");
            }
        }

        再定义一个接口的实现类:

        public class JDK8InterfaceImpl implements JDK8Interface ,JDK8Interface1 {
            //实现接口后,因为默认方法不是抽象方法,所以可以不重写,但是如果开发需要,也可以重写
        }

        静态方法,只能通过接口名调用,不可以通过实现类的类名或者实现类的对象调用。default方法,只能通过接口实现类的对象来调用。

        public class Main {
            public static void main(String[] args) {
                // static方法必须通过接口类调用
                JDK8Interface.staticMethod();
                //default方法必须通过实现类的对象调用
                new JDK8InterfaceImpl().defaultMethod();
            }
        }

        当然如果接口中的默认方法不能满足某个实现类需要,那么实现类可以覆盖默认方法。

        public class AnotherJDK8InterfaceImpl implements JDK8Interface {
            @Override
            public void defaultMethod() {
                System.out.println("接口实现类覆盖了接口中的default");
            }
        }

        五、最新的Date/Time API

        最新日期api用法参考

        新的时间及日期API位于java.time中,它们都是不可变且线程安全的

        属性含义
        Instant 代表的是时间戳
        LocalDate 代表日期,比如2020-01-14
        LocalTime 代表时刻,比如12:59:59
        LocalDateTime 代表具体时间 2020-01-12 12:22:26
        ZonedDateTime 代表一个包含时区的完整的日期时间,偏移量是以UTC/ 格林威治时间为基准的
        Period 代表时间段
        ZoneOffset 代表时区偏移量,比如:+8:00
        Clock 代表时钟,比如获取目前美国纽约的时间

        1、获取当前时间

        Instant instant = Instant.now(); //获取当前时间戳
        System.out.println("instant ==> " + instant);
        //instant ==> 2020-02-01T12:09:28.772Z
        
        LocalDate localDate = LocalDate.now();  //获取当前日期
        System.out.println("localDate ==> " + localDate);
        //localDate ==> 2020-02-01
        
        LocalTime localTime = LocalTime.now();  //获取当前时刻
        System.out.println("localTime ==> " + localTime);
        //localTime ==> 20:09:28.914
        
        LocalDateTime localDateTime = LocalDateTime.now();  //获取当前具体时间
        System.out.println("localDateTime ==> " + localDateTime);
        //localDateTime ==> 2020-02-01T20:09:28.914
        
        ZonedDateTime zonedDateTime = ZonedDateTime.now();   //获取带有时区的时间
        System.out.println("zonedDateTime ==> " + zonedDateTime);
        //zonedDateTime ==> 2020-02-01T20:09:28.916+08:00[Asia/Shanghai]

        2、字符串转日期

        String date = "2020-01-10";
        DateTimeFormatter formatter1 = DateTimeFormatter.ofPattern("yyyy-MM-dd");
        LocalDate localDate = LocalDate.parse(date, formatter1);
        System.out.println("localDate ==> " + localDate);

        3、日期转字符串

        Date dd = new Date();
        LocalDateTime ldt = dd.toInstant().atZone(ZoneId.systemDefault()).toLocalDateTime();
        System.out.println(ldt);
        DateTimeFormatter dtf = DateTimeFormatter.ofPattern("yyyy年MM月dd日 HH:mm:ss");
        String format = ldt.format(dtf);
        System.out.println(format);

        4、Date转换LocalDate

        Date date = new Date();
        Instant instant = date.toInstant();
        ZoneId zoneId = ZoneId.systemDefault();
        
        // atZone()方法返回在指定时区从此Instant生成的ZonedDateTime。
        LocalDate localDate = instant.atZone(zoneId).toLocalDate();
        System.out.println("Date = " + date);
        System.out.println("LocalDate = " + localDate);

        5、LocalDate转Date

        ZoneId zoneId = ZoneId.systemDefault();
        LocalDate localDate = LocalDate.now();
        ZonedDateTime zdt = localDate.atStartOfDay(zoneId);
        
        Date date = Date.from(zdt.toInstant());
        
        System.out.println("LocalDate = " + localDate);
        System.out.println("Date = " + date);

        6、时间戳转LocalDateTime

        long timestamp = System.currentTimeMillis();
        
        Instant instant = Instant.ofEpochMilli(timestamp);
        
        LocalDateTime.ofInstant(instant, ZoneId.systemDefault());

        7、LocalDateTime转时间戳

        LocalDateTime dateTime = LocalDateTime.now();
        
        dateTime.toInstant(ZoneOffset.ofHours(8)).toEpochMilli();
        
        dateTime.toInstant(ZoneOffset.of("+08:00")).toEpochMilli();
        
        dateTime.atZone(ZoneId.systemDefault()).toInstant().toEpochMilli();

        8、LocalDate方法总结

        getYear()                         int        获取当前日期的年份
        getMonth()                        Month      获取当前日期的月份对象
        getMonthValue()                   int        获取当前日期是第几月
        getDayOfWeek()                    DayOfWeek  表示该对象表示的日期是星期几
        getDayOfMonth()                   int        表示该对象表示的日期是这个月第几天
        getDayOfYear()                    int        表示该对象表示的日期是今年第几天
        withYear(int year)                LocalDate  修改当前对象的年份
        withMonth(int month)              LocalDate  修改当前对象的月份
        withDayOfMonth(intdayOfMonth)     LocalDate  修改当前对象在当月的日期
        isLeapYear()                      boolean    是否是闰年
        lengthOfMonth()                   int        这个月有多少天
        lengthOfYear()                    int        该对象表示的年份有多少天(365或者366)
        plusYears(longyearsToAdd)         LocalDate  当前对象增加指定的年份数
        plusMonths(longmonthsToAdd)       LocalDate  当前对象增加指定的月份数
        plusWeeks(longweeksToAdd)         LocalDate  当前对象增加指定的周数
        plusDays(longdaysToAdd)           LocalDate  当前对象增加指定的天数
        minusYears(longyearsToSubtract)   LocalDate  当前对象减去指定的年数
        minusMonths(longmonthsToSubtract) LocalDate  当前对象减去注定的月数
        minusWeeks(longweeksToSubtract)   LocalDate  当前对象减去指定的周数
        minusDays(longdaysToSubtract)     LocalDate  当前对象减去指定的天数
        compareTo(ChronoLocalDateother)   int        比较当前对象和other对象在时间上的大小,返回值如果为正,则当前对象时间较晚,
        isBefore(ChronoLocalDateother)    boolean    比较当前对象日期是否在other对象日期之前
        isAfter(ChronoLocalDateother)     boolean    比较当前对象日期是否在other对象日期之后
        isEqual(ChronoLocalDateother)     boolean    比较两个日期对象是否相等
  • 相关阅读:
    【HDU】4092 Nice boat(多校第四场1006) ——线段树 懒惰标记 Prime
    【POJ】2528 Mayor's posters ——离散化+线段树 Prime
    【HDU】1754 I hate it ——线段树 单点更新 区间最值 Prime
    C语言 linux环境基于socket的简易即时通信程序 Prime
    Java异常处理
    重载和重写的区别与联系
    SQL Server的优点与缺点
    Servlet基础
    C语言图形编程
    socket通讯,TCP,UDP,HTTP的区别
  • 原文地址:https://www.cnblogs.com/zouhong/p/16502853.html
Copyright © 2020-2023  润新知