java8学习:入门

简介:

内容来自《 java8实战 》,本篇文章内容均为非盈利,旨为方便自己查询、总结备份、开源分享。如有侵权请告知,马上删除。
书籍购买地址:java8实战

  • sync的使用成本:多核CPU的每个处理器内核都有独立的告诉缓存,加锁需要这些告诉缓存同步运行,然而这有需要在内核间进行较慢的缓存一致性协议通信

流处理

  • 流是一系列数据项,一次只生成一项。程序可以从输入流中一个个读取数据项,然后以 相同的方式将数据项写入输出流
  • 拿Linux的管道命令来演示流过程

markdown_img_paste_20181105080342380

  • 如上熟悉linux都清楚cat file1 file2会创建两个文件的输出流,如果加上>那么就会生成一个合并了File1和file2的新文件,cat产生的数据流会通过管道传送给tr进行内容过滤,符合条件的又会传入下一个sort方法进行排序,最后sort排序完成的结果集交给tail -3取三行,至此流算是结束了。这些的流处理行为是高效的,因为cat的数据读一条就会传给tr过滤,tr后传入sort,sort就能在cat或tr完成前先处理几行
  • 基于上面的思想,java8引入了Stream,这样的好处是并行能力得到了极大的提升,并且站在了更高的抽象角度去编程,因为原来的编程习惯的java只能利用一个cpu,除非自己去写thread,并且thread是容易出错的
  • 用行为参数化把代码传递给方法

    • java8可以通过api传递代码。
    • 对上面的解释就是:看到上面的图的sort函数,可能它默认的只能升序或者降序排序,但是如果我们需要定制排序的话,也就只能对sort进行重新定义,那么就比如下面(java代码演示)
    @Test
    public void test() throws Exception {
        List<Integer> ints = Arrays.asList(1,3,4,5,1,5,7,7,8);
        ints.sort(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return o1.compareTo(o2);  //升序
            }
        });
        System.out.println(ints);
        ints.sort(new Comparator<Integer>() {
            @Override
            public int compare(Integer o1, Integer o2) {
                return - o1.compareTo(o2);  //降序
            }
        });
        System.out.println(ints);
    }
    • 如上的代码,其实是只有renturn...是有用的,这句代码关乎到排序规则,但是其他的代码都是无用的。那么这时候就会想,只需要写一遍匿名内部类,排序逻辑根据自己需要传入不就行了。这个思想就是:通过api传递代码

    markdown_img_paste_20181105082157197

  • 并行与共享的可变数据

    • 并发和并行的概念
    并发是两个任务可以在重叠的时间段内启动,运行和完成。并行是任务在同一时间运行,例如,在多核处理器上。 并发是独立执行过程的组合,而并行是同时执行(可能相关的)计算。 并发是一次处理很多事情,并行是同时做很多事情。 应用程序可以是并发的,但不是并行的,这意味着它可以同时处理多个任务,但是没有两个任务在同一时刻执行。 应用程序可以是并行的,但不是并发的,这意味着它同时处理多核CPU中的任务的多个子任务。 一个应用程序可以即不是并行的,也不是并发的,这意味着它一次一个地处理所有任务。 应用程序可以即是并行的也是并发的,这意味着它同时在多核CPU中同时处理多个任务。
    • 那么在以往的线程编程中,实现并行的前提就是各个副本可以独立工作,如果在多个副本出现了共享的变量或对象,这就行不通了,因为如果出现同时修改共享变量怎么办
    • java8的流实现比线程更容易,尽量保持流处理过程中,不访问共享的可变变量,这种函数就被称为纯函数,无副作用函数或无状态函数。但是依旧可以使用sync来实现一个可变数据,但是这样就与并行的概念相悖

函数

  • 方法引用

    • 编写一个目录下查询出所有隐藏文件
    @Test
    public void test() throws Exception {
        File[] files = new File(".").listFiles(new FileFilter() {
            @Override
            public boolean accept(File pathname) {
                return pathname.isHidden();
            }
        });
        for (File file : files) {
            System.out.println(file);
        }
    }
    • 如上啰里啰嗦的只是判断 pathname.isHidden(),java8引入之前只能这么写,但是现在可以这样
    @Test
    public void test() throws Exception {
        File[] files = new File(".").listFiles((File::isHidden));
        for (File file : files) {
            System.out.println("file = " + file);
        }
    }
    • 上面是将isHidden方法做为值传入listFIles方法,与用对象引用传递对象相似,在java8里写下File::isHidden的时候,就创建了一个方法引用,然后可以传递这个引用了
    • lambda匿名函数

      • 比如代码(int x ) -> x+1,代码的意思就是你传入一个2,那么他会返回3,也可以像上面定义一个方法add1,然后class::add1,但是对于简短逻辑明确的代码来说这样更简洁,如果匿名函数有很多行代码,不能一眼看出这个匿名函数是干嘛的,那么就应该把匿名函数抽出来一个方法使用
    • 实例

      • bean代码
      @AllArgsConstructor
      @NoArgsConstructor
      @Data
      public class Apple {
          private String color;
          private Integer weight;
      }
      • java8之前:苹果根据条件过滤:找出绿色苹果
      @Test
      public void test() throws Exception {
          List<Apple> apples = new ArrayList<>();
          for (Apple apple : apples) {
              if ("green".equals(apple.getColor())){
                  System.out.println(apple);
              }
          }
      }
      • java8之前:苹果根据条件过滤:找出1000克以上的大苹果
      @Test
      public void test() throws Exception {
          List<Apple> apples = new ArrayList<>();
          for (Apple apple : apples) {
              if (1000 < apple.getWeight()){           //其他代码与上面都是一直的,无奈的是只能复制黏贴改条件,复制黏贴是为限的,因为如果有一天需要改一个地方,如果忘记了另一处复制的代码,那么就会出错
                  System.out.println(apple);
              }
          }
      }
      • java8之后
      1. 改变Apple类
      @AllArgsConstructor
      @NoArgsConstructor
      @Data
      public class Apple {
          private String color;
          private Integer weight;
      
          public static boolean filterWeight(Apple apple){
              return 1000 < apple.getWeight();
          }
          public static boolean filterColor(Apple apple){
              return "green".equals(apple.getColor());
          }
      }
      1. 增加接口
      public interface FindApple<T> {
          boolean find(T t);
      }
      1. 编写方法测试
      @Test
      public void test() throws Exception {
          List<Apple> apples  = new ArrayList<>();
          filterApple(apples,Apple::filterColor);  //java8之后:苹果根据条件过滤:找出绿色苹果
          filterApple(apples,Apple::filterWeight); //java8之后:苹果根据条件过滤:找出1000克以上的大苹果
      }
      static void filterApple(List<Apple> apples,FindApple<Apple> f){
          for (Apple apple : apples) {
              if (f.find(apple)){
                  System.out.println(apple);
              }
          }
      }
      • 自己实现方法引用确实有点复杂,首先要为需要过滤的类增加自实现的static方法,然后编写带有测试方法的结果,之后就是编写方法,此方法是可以接收需要测试类内static方法的。
      • 如上方法filterApple中的f.find(apple),都是可以看出来这是在使用接口中的测试方法,但是这个方法没有实现自己的逻辑,所以到这它是不能按照要求过滤apple的,然后在上面junit测试方法中,传入的Apple::filterColor其实就是为接口中的测试方法添加逻辑,使其能按照我们的要求过滤apple
    • 从传递方法到lambda

      • 如上自己实现方法引用那真是脸上心里全都是mmp,为了实现方法引用,我们增加类Apple的方法,增加了接口等,如果java8要真是这样,那么宁可不用,不过java8是解决了这个问题的,他引入了lambda表达式,这时候上面的写法就可以写为
      1. 将增加的接口删掉~
      2. 将在Apple内增加的方法删掉~
      3. 编写代码
      @Test
      public void test() throws Exception {
          List<Apple> apples  = new ArrayList<>();
          filterApple(apples,(apple -> "green".equals(apple.getColor())));
          filterApple(apples,(apple -> (apple.getWeight() > 1000)));
      }
      static void filterApple(List<Apple> apples, Predicate<Apple> f){
          for (Apple apple : apples) {
              if (f.test(apple)){
                  System.out.println(apple);
              }
          }
      }
      • 这时候看到的变化是从原来的FindApple接口改变为了Predicate接口,其实这个接口的定义跟刚才的FIndApple接口内定义是一样一样的,如下

        @FunctionalInterface
        public interface Predicate<T> {
            boolean test(T t);
            ....
            ....
        }
      • 最大的变化就是我们并没有使用方法引用,而是直接以更直观的方法来过滤apple

  • 日常写代码的时候,在java8之前如果要遍历Map中找出map对应key的特定value值,如果更加复杂的类型就需要嵌套循环并且编写出来的代码是一大坨的

  1. void test() throws Exception {
    Map maps = new HashMap<>();
    for (Map.Entry integerStringEntry : maps.entrySet()) {

    if (integerStringEntry.getKey() > 1000) {
        if (integerStringEntry.getValue().equals("value")){
            System.out.println("integerStringEntry = " + integerStringEntry);
        }
    }

    }
    }

  • Stream api就类似SQL一样的过滤操作,如上的代码可以写为下面这样的

  1. void test() throws Exception {
    Map maps = new HashMap<>();
    maps.entrySet().stream()

        .filter(entry -> entry.getKey() > 1000)
        .filter(entry -> entry.getValue().equals("value"))
        .forEach(System.out::println);

    }

  • Stream 解决了经典的java程序只能利用一个cpu的问题,可以使程序并行执行,如图

markdown_img_paste_20181105101510665

- 在两个cpu上筛选数据,分割数据到两个cpu上,如1
- 按条件过滤数据,如2
- 一个cpu会将结果汇总起来,如3
  • Stream顺序处理

    maps.entrySet().stream()....
  • Stream的并行处理

    maps.entrySet().parallelStream()....

默认方法

  • 如上已经看到了,在java8中使用集合类都会有stream方法,那么在java8之前是没有这个方法,如果把java7升级到java8,那么多的这个stream方法不就的需要实现吗,那么不就不能向后兼容了吗?
  • 如上的问题全是由接口内的默认方法实现的
  • 默认方法的定义是由default开始的,比如

    public interface MyFunction {
        default boolean test(String str) {
            return str.equals("..");
        }
    }
    public class UU implements MyFunction{
        public static void main(String[] args) {
            UU uu = new UU();
            System.out.println(uu.test(".."));
        }
    }
    • 如上UU类中根本不要实现接口中的test方法就可以使用,当然也可以覆盖实现
    public class UU implements MyFunction{
    
        @Override
        public boolean test(String str) {
            return "ll".equals(str);
        }
    
        public static void main(String[] args) {
            UU uu = new UU();
            System.out.println(uu.test(".."));
        }
    }
    • 这时候实现的逻辑就是子类重写之后的了
    • 我们都知道接口是可以多重实现的,那么如果IA和IB中方法名重复了咋办
public interface MyFunction1 {
    default boolean test(String str) {
        return str.equals("..");
    }
}
public interface MyFunction2 {
    default boolean test(String str) {
        return str.equals("ll");
    }
}
public class UU implements MyFunction1,MyFunction2{   //exception
    public static void main(String[] args) {
        UU uu = new UU();
        System.out.println(uu.test(".."));
    }
}
  • 如上当两个接口中的方法名重复了,那么就会出错,解决办法就是重写重复的方法即可,当然方法重载是没有问题的
  • 这就是为什么在更新为java8之后,之前的代码还是可以使用的原因,java8 在Collection接口中加入了stream默认实现了

      default Stream<E> stream() {
          return StreamSupport.stream(spliterator(), false);
      }
目录
相关文章
|
3月前
|
存储 安全 Java
从入门到精通:Java Map全攻略,一篇文章就够了!
【10月更文挑战第17天】本文详细介绍了Java编程中Map的使用,涵盖Map的基本概念、创建、访问与修改、遍历方法、常用实现类(如HashMap、TreeMap、LinkedHashMap)及其特点,以及Map在多线程环境下的并发处理和性能优化技巧,适合初学者和进阶者学习。
80 3
|
5天前
|
自然语言处理 Java
Java中的字符集编码入门-增补字符(转载)
本文探讨Java对Unicode的支持及其发展历程。文章详细解析了Unicode字符集的结构,包括基本多语言面(BMP)和增补字符的表示方法,以及UTF-16编码中surrogate pair的使用。同时介绍了代码点和代码单元的概念,并解释了UTF-8的编码规则及其兼容性。
78 60
|
1月前
|
Java 开发者 微服务
Spring Boot 入门:简化 Java Web 开发的强大工具
Spring Boot 是一个开源的 Java 基础框架,用于创建独立、生产级别的基于Spring框架的应用程序。它旨在简化Spring应用的初始搭建以及开发过程。
60 6
Spring Boot 入门:简化 Java Web 开发的强大工具
|
3月前
|
XML Java 编译器
Java学习十六—掌握注解:让编程更简单
Java 注解(Annotation)是一种特殊的语法结构,可以在代码中嵌入元数据。它们不直接影响代码的运行,但可以通过工具和框架提供额外的信息,帮助在编译、部署或运行时进行处理。
108 43
Java学习十六—掌握注解:让编程更简单
|
1月前
|
监控 架构师 Java
Java虚拟机调优的艺术:从入门到精通####
本文作为一篇深入浅出的技术指南,旨在为Java开发者揭示JVM调优的神秘面纱,通过剖析其背后的原理、分享实战经验与最佳实践,引领读者踏上从调优新手到高手的进阶之路。不同于传统的摘要概述,本文将以一场虚拟的对话形式,模拟一位经验丰富的架构师向初学者传授JVM调优的心法,激发学习兴趣,同时概括性地介绍文章将探讨的核心议题——性能监控、垃圾回收优化、内存管理及常见问题解决策略。 ####
|
2月前
|
监控 安全 Java
Java中的多线程编程:从入门到实践####
本文将深入浅出地探讨Java多线程编程的核心概念、应用场景及实践技巧。不同于传统的摘要形式,本文将以一个简短的代码示例作为开篇,直接展示多线程的魅力,随后再详细解析其背后的原理与实现方式,旨在帮助读者快速理解并掌握Java多线程编程的基本技能。 ```java // 简单的多线程示例:创建两个线程,分别打印不同的消息 public class SimpleMultithreading { public static void main(String[] args) { Thread thread1 = new Thread(() -> System.out.prin
|
2月前
|
Java 大数据 API
14天Java基础学习——第1天:Java入门和环境搭建
本文介绍了Java的基础知识,包括Java的简介、历史和应用领域。详细讲解了如何安装JDK并配置环境变量,以及如何使用IntelliJ IDEA创建和运行Java项目。通过示例代码“HelloWorld.java”,展示了从编写到运行的全过程。适合初学者快速入门Java编程。
|
2月前
|
存储 安全 Java
🌟Java零基础-反序列化:从入门到精通
【10月更文挑战第21天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
88 5
|
2月前
|
安全 Java 调度
Java中的多线程编程入门
【10月更文挑战第29天】在Java的世界中,多线程就像是一场精心编排的交响乐。每个线程都是乐团中的一个乐手,他们各自演奏着自己的部分,却又和谐地共同完成整场演出。本文将带你走进Java多线程的世界,让你从零基础到能够编写基本的多线程程序。
38 1
|
2月前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。