java8 入门到精通 - 笔记

java8 入门到精通 - 笔记,第1张

java8 入门到精通 - 笔记 Java 8 新特性
  1. 速度更快
  2. 代码更少(增加了新的lambda表达式)
  3. 强大的Stream apl
  4. 便于并行
  5. 最大化减少空指针异常 Optional
    其中最为核心的为lambda 表达式和Stream apl
一. Lambda 表达式

Lambda 是一个匿名函数,我们可以把lambda表达式理解为一段可以传递的代码(将代码像数据一样进行传递)。可以写出更简洁、更灵活的代码。作为一种更紧凑的代码风格,使java的语言表达能力得到了提升。
1、Lanbda 表达式的基础语法:
java8中引入了一个新的 *** 作符 “->” ,该 *** 作符称为箭头 *** 作符或Lambda *** 作符,箭头 *** 作符将Lambda 表达式拆分成两部分

左侧: Lambda 表达式的参数列表
右侧:Lambda 表达式中所有 需要执行的功能,即lambda体

		语法格式一:无参数 无返回值
  	   ()-> System.out.println("hello lambda")
  
  	   语法格式二:有1个参数,并且无返回值的方法。
  			(x) -> System.out.println(x);
  
	   语法格式三: 若只有一个参数,小括号可以省略不写
           x -> System.out.println(x);
       语法格式四: 有两个以上的参数,有返回值,并且lambda 体中有多条语句
 			Comparator com = (x,y)->{
			System.out.println("函数式接口");
			return Integer.compare(x, y);
		};
		语法格式五: 若lambda体中只有一条语句,return和大括号都可以不写
			  Comparator com = (x,y)-> Integer.compare(x, y);
	    语法格式六: lambda 表达式的参数列表的数据类型可以省略不写,因为JVM编译器通过上下文推断出,	数据类型,即“类型推断”   
			 (Integer x, Integer y)-> Integer.compare(x, y);

Java8 内置的四大核心的函数式接口

	Consumer : 消费型接口
  			void accept(T t)
  
  	Supplier : 供给型接口
    		T get();
   
  	Function : 函数型接口
  			R apply(T t)
  
  	Predicate : 断言行接口
  			boolean test(T t)

一、方法引用: 若Lumbda体中的内容有方法已经实现了,我们可以使用"方法引用"。(可以理解为方法引用是Lambda 表达式的另外一种表现形式)
主要有三种语法格式:

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

注意:

  • 1 Lambda 体中调用方法的参数列表与返回值类型,要与函数式接口中抽象方法的函数列表和返回值类型保持一致
  • 2 若lambda参数列表中的第一个参数是实例方法的调用者,二第二个参数是实例方法的参数时,可以使用ClassName::method

二、构造器引用

  • 格式: ClassName::new
    注意:需要调用的构造器的参数列表要与函数式接口中抽象方法的参数列表保持一致

三、数组引用
格式 type[]::new

二、Stream

Java8中有两大最为重要的改变,第一个是Lambda表达式;另外一个则是Stream API; Stream是java8中处理集合的关键抽象概念,它可以指定你希望对集合进行的 *** 作,可以执行非常复杂的查找,过滤和映射数据等 *** 作,使用Stream API对集合数据进行 *** 作,就类似于使用Sql执行数据库查询,也可以使用Stream API 来并行执行 *** 作,简而言之,StreamAPI提供了一种高效且易于使用的 处理的方式。

什么是Stream?
是数据渠道,用于 *** 作数据源(集合,数组等)所生成的元素序列。“集合讲的是数据,流讲的是计算!"

注意:

  1. Stream 自己不会存储元素。
  2. Stream 不会改变源对象,相反,他们会返回一个持有结果的新Stream,
  3. Stream *** 作是延迟执行的,这意味着他们会等到需要结果的时候才执行。

Stream 的 *** 作三个步骤

  1. 创建Strean 一个数据源(如集合、数组等),获取一个流。
  2. 中间 *** 作 一个中间 *** 作链,对数据源的数据进行处理。
  3. 终止 *** 作(终端 *** 作) 一个终止 *** 作,执行中间 *** 作链,并产生结果

创建无限流 两种方式

迭代
Stream stream4 = Stream.iterate(0, (x) -> x+2); stream4.limit(10).forEach(System.out::println);>

生成
Stream.generate(()-> Math.random()).limit(5).forEach(System.out::println);

中间 *** 作
多个中间 *** 作可以连接起来形成一个流水线,除非流水线上触发终止 *** 作,否则中间 *** 作不会去执行任何的处理,二在终止 *** 作时一次性全部处理,称为“惰性求值”

筛选与切片: filter – 接收Lambda,从流中排除某些元素 limit -截断流,使其元素不超过给的数量 skip(n)
跳过元素:返回一个扔掉了前n个元素的流,若流中元素不足n个,则返回一个空流, distinct
筛选:通过流所生成元素的hashcode()和equals(去重复元素)

映射 map: – 接收Lambda ,将元素转换成其他形式或提取信息,接收一个函数作为参数。改函数会被应用到每个 flatmap
接收一个函数作为参数,将流中每个指都换成另一个流,然后把所有流连接成一个流

排序 :sorted()
自然排序: sorted(comparator com) 定制排序

终止 *** 作

allMatch 检测是否匹配所有元素
anyMatch 检测是否至少匹配一个元素
nonematch 检测是否没有匹配所有元素
findfirst 返回第一个元素
findAny 返回当前流中的任意元素
count 返回流中元素的总个数
max 返回流中最大值
min 返回流中最小值

归约

reduce(T identity,Binaryoprator)/ reduce(BinaryOpertor)
可以将流中元素反复结合起来,得到一个值,返回T/Optional
注:map和reduce 的连接通常称为map-reduce模式,因Google用它来进行网络搜索而出名 收集 Collect(Collector c)
将流转换为其他形式,接收一个Collector 接口的实现,用于给Stream中元素做汇总的方法
注:Collector 接口中方法的实现决定了如何对流执行收集 *** 作(如收集到List,Set,Map)。但是Collectors 实现类提供了很多静态方法,可以方便的创建常见收集器实例。
具体方法与实例如下:
并行流与顺序流
并行流就是把一个内容分成多个数据块,并用不同的线程分别处理每个数据块的流;
Java8中将并行进行了优化,我们可以很容易的对数据进行并行 *** 作,String Api 可以声明性的通过parallel() 与 sequential()在并行流与顺序流之间进行切换

了解Fork/Join框架

Fork/Join 框架:就是在必要的情况下,将一个大任务,进行拆分(rork)成若干个小任务(拆到不可在拆时),在将一个个小任务运算的结果进行join汇总

Fork/join 框架与传统线程池的区别

采用“工作窃取”模式(work-stealing)
当执行新的任务时它可以将其拆分分成更小的任务执行,并将小任务加到线程队列中,然后在从一个随机线程队列中偷一个并把它放到自己的队列中,相对于一般的线程池实现,fork/join框架的优势体现在对其中包含的任务的处理方式上,在一般的线程池中,如果一个线程正在执行的任务由于某些原因无法继续运行,那么该线程会处于等待状态,二在fork/join框架实现中,如果某个子问题由于等待另外一个子问题完成而无法继续运行,那么处理该子问题的线程会主动寻找其他尚未运行的子问题来执行,这种方式减少了线程的等待时间,提高了性能。

Optional 类

Optional类: 是一个容器类,代表一个值存在或不存在,原来用null表示一个值不存在,现在optional可以更好的表达这个概念,并且可以避免空指针异常。
常用方法: Optional.of(T t) 创建一个Optional 实例 Optional.empty():
创建一个空的Optional实例 Optional.ofNullable(T t):
若T不为null,创建Optional实例,否则创建空实例 isPresent():
判断是否包含值 orElse(T t):如果调用对象包含值,返回该值,否则返回torElseGet(Supplier s): 如果调用对象包含值,返回该值,否则返回s> 获取的值。
Map(Function f)如果有值对其处理,并返回处理后的Optional,否则返回Optional.empty()
flatMap(Function mapper): 与map类似,要求返回值必须是Optional

接口中的默认方法和静态方法

接口中的默认方法
接口默认方法的“类优先”原则
若一个接口中定义了一个默认方法,而另外一个父类或接口中又定义了一个同名的方法时
A.	选择父类中的方法。如果一个父类提供了具体的实现,那么接口中具有相同名称和参数的默认方法会被忽略
B.	接口冲突。如果一个父接口提供一个默认方法,而另一个接口也提供了一个具有相同名称和参数列表的方法,(不管方法是否是默认方法),那么必须覆盖该方法来解决冲突
public interface MyInteface {
	default String getName(){
		return "接口2中的默认方法";
	}
	
	public static void show() {
		System.out.print("接口2中的静态方法");
	}
}
Java8 新日期对象

使用LocalDate、LocalTime、LocalDateTime 类的实例是不可变的对象,分别表示使用ISO-8601日历系统的日期、时间、日期和时间。他们提供了简单的日期或时间,并不包含的时间信息,也不包含于时区相关的信息。
注:iso-8601日历系统是国籍标准化组织制定的先飞公民的如期和时间表示法。

日期的 *** 纵

1.	TemporalAdjuster:时间校正器。有时我们可能需要获取例如:将日期调整到下个周日等 *** 做。
2.	TemporalAdjusters:该类通过静态方法提供了大量的常用TemporalAdjuster的实现
例如获取下个周日:
LocalDate nestSunday = LocalDate.new().with(
		TemporalAdjusters.next(DayOfWeek.SUNDAY)
);

时区的处理

Java8中加入了对时区的支持,带时区的时间为分别为ZonedDate, ZinedTime ,ZonedDateTime,
其中每个时区都对应着ID,地区ID都为“{区域}/{城市}”的格式 例如: Asia/Shanghai等 Zoneld:
该类中包含了所有时区信息 getAvailableZonelds():可以获取所有时区信息
of(id):用指定时区信息获取Zoneld对象

重复注解与类型注解

Java8 对注解处理提供了两点改进,可重复的注解及可用于类型的注解
//重复注解
	@MyAnnotation("hello")
	@MyAnnotation("word")
	public void show() {
		
	}
//类型注解
	public void show(@MyAnnotation("abc") String str) {
		
	}

欢迎分享,转载请注明来源:内存溢出

原文地址: https://outofmemory.cn/zaji/5692740.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存