jdk1.8新特性
jdk1.8新特性主要知识点:
1.特殊的接口
2.Lambda表达式
3.数据流编程
4.并行数据排序
5.optional中避免null检查
6.新的时间和日期APL
7.可重复注释
方法的调用:可以使用::调用方法
特殊的接口
1.类的两个要素:属性和方法
*属性的三要素:类型,名字,值
*属性的五要素:修饰符,返回值,方法名,形参,方法体
方法引用
三种表现形式:
- 对象::实例方法名
- 类::静态方法名
- 类::实例方法名 (lambda参数列表中第一个参数是实例方法的调用 者,第二个参数是实例方法的参数时可用)
public void test() {
/**
*注意:
* 1.lambda体中调用方法的参数列表与返回值类型,要与函数式接口中抽象方法的函数列表和返回值类型保持一致!
* 2.若lambda参数列表中的第一个参数是实例方法的调用者,而第二个参数是实例方法的参数时,可以使用ClassName::method
*
*/
Consumer<Integer> con = (x) -> System.out.println(x);
con.accept(100);
// 方法引用-对象::实例方法
Consumer<Integer> con2 = System.out::println;
con2.accept(200);
// 方法引用-类名::静态方法名
BiFunction<Integer, Integer, Integer> biFun = (x, y) -> Integer.compare(x, y);
BiFunction<Integer, Integer, Integer> biFun2 = Integer::compare;
Integer result = biFun2.apply(100, 200);
// 方法引用-类名::实例方法名
BiFunction<String, String, Boolean> fun1 = (str1, str2) -> str1.equals(str2);
BiFunction<String, String, Boolean> fun2 = String::equals;
Boolean result2 = fun2.apply("hello", "world");
System.out.println(result2);
}
*在接口中的方法可以有方法体
Lambda表达式
lambda表达式本质上是一段匿名内部类,也可以是一段可以传递的代码
.是被python,scala逼出来的,语法简洁
2.匿名类:
//匿名内部类
Comparator<Integer> cpt = new Comparator<Integer>() {
@Override
public int compare(Integer o1, Integer o2) {
return Integer.compare(o1,o2);
}
};
TreeSet<Integer> set = new TreeSet<>(cpt);
System.out.println("=========================");
//使用lambda表达式
Comparator<Integer> cpt2 = (x,y) -> Integer.compare(x,y);
TreeSet<Integer> set2 = new TreeSet<>(cpt2);
*目标方法的参数得是一个接口
*方法的参数里面直接new后面跟上接口名
*必须马上写类体,重写方法体
3.Lambda就是一个匿名函数,我们只需将要执行的代码放到Lambda表达式中
Lambda表达式的好处:可以简化匿名内部类,让代码更加精简
示例
package com.jinghangzz.jdk_new.lambda;
import java.util.Arrays;
import java.util.List;
import java.util.function.Consumer;
/**
* lambda的测试类 最终要实现的效果和匿名类很像
* 如果看到一个方法的参数类型后面跟了一个...
* 在方法定义的时候,把这三个点的参数,当成数组使用(String ... str)和(String[] str)效果一样
* 在方法使用的时候,这三个点要传入实参,实参的个数随便填写
* ...必须是方法形参的最后一个
* ~示例:方法调用时(String[] str)
* String[] str = {"1","2","3"} ;
* 目标方法(str)
* ~方法调用时String ... str
* 目标方法("1","2","3","3","3","3","3","3")
* @author TeaBig
*/
public class LambdaTest
{
public static void main(String[] args)
{
System.out.println("====");
/* 调用方法 */
String[] str = {"1","a","b","c","d"};
//method1Arr(str,"2");
/* 调用方法 */
//method2Point(str);
method2Point(1,"1","a","b","c","d","e");
/* 集合抽取lambda */
lambdaList();
}
lambda表达式,测试类
/**
* lambda表达式,测试类
*/
private static void lambdaList()
{
/* 准备一个List */
List<String> list = Arrays.asList("1","a","b","c","d");
/* 如何循环
* void accept(T t);
* (类型忽略,直接写名字,形参名,名字随便写)
* */
// list.forEach( (t) ->
// {
// System.out.println("==lambda循环==" + t);
// } );
/* 超级简单的写法 */
list.forEach( t -> System.out.println("==lambda循环==" + t));
/* 匿名类
* 泛型是和容器的泛型一致
* */
list.forEach(new Consumer<String>()
{
@Override
public void accept(String t)
{
System.out.println("==Consumer循环==" + t) ;
}
});
}
方法的形参是数组
/**
* 方法的形参是数组
* @param str
*/
public static void method1Arr(String[] str)
{
/* 查看数组里面的内容 */
for (int i = 0; i < str.length; i++)
{
String temp = str[i];
System.out.println(i + "==method1Arr==" + temp);
}
}
方法的形参是三个点
/**
* 方法的形参是三个点
* @param str
*/
public static void method2Point(int j , String... str)
{
/* 查看数组里面的内容 */
for (int i = 0; i < str.length; i++)
{
String temp = str[i];
System.out.println(i + "==method2Point==" + temp);
}
}
}
Lambda的标准格式
(参数列表)->{
}
(参数列表):参数列表
{}:方法体
->:没有实际含义,起到连接的作用
Lmabda表达式的语法总结:** () -> ();**
无参数无返回值 :
() -> System.out.println(“Hello WOrld”)
有一个参数无返回值
(x) -> System.out.println(x)
有且只有一个参数无返回值
x -> System.out.println(x)
有多个参数,有返回值,有多条lambda体语句
(x,y) -> {System.out.println(“xxx”);return xxxx;};
有多个参数,有返回值,只有一条lambda体语句
(x,y) -> xxxx
数据流编程
1.Spark特性和方法相像
2.数据:容器;
*放的是什么(泛型)
*大小
*Crud
3.需求List里面存储的数据超级大;
*找10个及格的分数
*循环之中满足需求,break;
*容器的大小:无上限
4.重要的接口:Stream
/*Stream接口的方法可以归为两类(算子就是方法)
*Stream接口的方法可以归为两类(算子就是方法)
*方法的返回值还是Stream:(Transformaction算子)
方法的返回值不是Stream:(action算子)`
transformaction算子
/**
* 数据流编程
* @author TeaBig
*/
public class DataStream
{
/**
* 测试一个方法(框架)
*/
@Test
public void base()
{
/* 准备一个容器 */
List<Integer> intList = Arrays.asList(100,2,38,12,50,87);
/* 准备一个Stream */
Stream<Integer> souStream = null;
try
{
souStream = intList.stream();
/* 并行:多线程 */
//souStream = intList.parallelStream();
System.out.println("=isParallel==>" + souStream.isParallel());
/* 循环数据
* java.util.function.Consumer
* void accept(T t);
*
* forEach:action算子
* */
souStream.forEach( t -> System.out.println("==Stream循环==" + t));
} catch (Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
} finally
{
if(souStream != null)
{
souStream.close();
souStream = null ;
}
}
}
过滤
/**
* 过滤
* 按照指定的条件,对数据流里面的数据进行过滤
*/
@Test
public void transformaction_Filter()
{
/* 准备一个容器 */
List<Integer> intList = Arrays.asList(100,2,38,12,50,87);
/* 准备一个Stream */
Stream<Integer> souStream = intList.stream() ;
/* 过滤
* java.util.function.Predicate<T>
* boolean test(T t);
*
* 只有一行代码,return可以省略
* */
Stream<Integer> filterStream = souStream.filter( (t) -> t > 50);
/* 循环数据
* java.util.function.Consumer
* void accept(T t);
*
* forEach:action算子
* */
filterStream.forEach( t -> System.out.println("==Stream循环==" + t));
/* 关闭 */
souStream.close();
}
/**
* 过滤
*/
@Test
public void transformaction_Filter_Base()
{
/* 准备一个容器 */
List<Integer> intList = Arrays.asList(100,2,38,12,50,87);
/* 准备一个Stream */
Stream<Integer> souStream = intList.stream() ;
/* 过滤算子 */
Stream<Integer> filterStream = souStream.filter( new Predicate<Integer>()
{
@Override
public boolean test(Integer t)
{
System.out.println("=filter==test=" + t);
return t > 50;
}
} );
filterStream.forEach( new Consumer<Integer>()
{
@Override
public void accept(Integer t)
{
System.out.println("==循环==" + t);
}
} );
/* 关闭 */
souStream.close();
}
map算子
public void transformaction_Map()
{
/* 准备一个容器 */
List<Integer> intList = Arrays.asList(100,2,38,12,50,87);
/* 准备一个Stream */
Stream<Integer> souStream = null;
try
{
souStream = intList.stream();
/* map算子;要把输入参数,经过一个方法变成另外一个参数;(处理参数)
* java.util.function.Function<T, R>
* R apply(T t);
*
* 需求:要把这里面的整数,都加一个字符串
* */
Stream<String> mapStream = souStream.map( (t) -> t + "==我变化了");
/* 循环数据
* java.util.function.Consumer
* void accept(T t);
*
* forEach:action算子
* */
mapStream.forEach( t -> System.out.println("==Stream循环==" + t));
} catch (Exception e)
{
// TODO Auto-generated catch block
e.printStackTrace();
} finally
{
if(souStream != null)
{
souStream.close();
souStream = null ;
}
}
}
FlatMap算子
public void transformaction_other()
{
/* 准备一个容器 */
List<Integer> intList = Arrays.asList(100,2,50,50,50,87);
/* 准备一个Stream */
Stream<Integer> souStream = null;
try
{
souStream = intList.stream();
/*
* distinct:去重
* */
//Stream<Integer> stream = souStream.distinct();
/* 取前几条记录 */
//Stream<Integer> stream = souStream.limit(2);
/* 跳过前n条数据 */
//Stream<Integer> stream = souStream.skip(2);
/* 排序:默认升序 */
//Stream<Integer> stream = souStream.sorted();
/* java.util.Comparator<T>:
* int compare(T o1, T o2);
* */
//Stream<Integer> stream = souStream.sorted( (t1,t2) -> -(t1 - t2) );
Stream<Integer> stream = souStream.unordered();
/* 循环数据
* java.util.function.Consumer
* void accept(T t);
*
* forEach:action算子
* */
stream.forEach( t -> System.out.println("==Stream循环==" + t));
} catch (Exception e)
{
e.printStackTrace();
} finally
{
if(souStream != null)
{
souStream.close();
souStream = null ;
}
}
}
}
action算子
5.数据流
*一根管子,一个数据会把所有的管子走完以后,再走第二个数据
*而不是所有的数据走完第一个管子以后,再一块走第二个管子
重点:
所有的transfofrmaction算子都是懒惰的,必须碰到action算子,才会一起执行
函数式接口
package com.jinghangzz.jdk_new.inter;
import java.util.function.Supplier;
/**
* 接口:
* 为什么接口中的方法要有方法体:
* 接口就是为了定标准,这个标准要求所有的实现类,实现抽象方法;
* 其实有些方法,所有的实现类代码都是一样的;干脆就把这些代码挪到接口中;
*
* @author TeaBig
*/
public interface IAnimal
{
/* 属性 */
int HEAD_NUM = 1 ;
/* 下面的写法等同于上面的
* 被static修饰的,名字都是大写,
* 但是驼峰标识:单词之前使用_隔开
* */
//public static int HEADNUM = 1 ;
*/
抽象的方法:
/* 抽象的方法 */
/**
* 抽象的方法,跑
*/
void run();
/* 丢人的写法 */
//public abstract void run1();
/* ====新特性开始 ==== */
/**
* 有方法体的方法(普通的方法)
* 吃
* 要求:
* 必须被default修饰
* 此default和修饰符里面的default不一样
* 它是被public修饰
*/
default void eat()
{
System.out.println("==IAnimal==eat");
}
/**
* 睡的方法
*/
default String sleep()
{
System.out.println("==IAnimal==sleep");
return "===" ;
}
接口中可以定义默认实现方法和静态方法
静态方法:
/* 静态的方法 */
static void staticMethod()
{
System.out.println("==IAnimal==staticMethod");
}
/**
* 参数是一个接口;
* Supplier<IAnimal> supplier(供应商)
* @return
*/
static IAnimal createObj(Supplier<IAnimal> supplier)
{
/* 让供应商去提供供货 */
return supplier.get() ;
}
/* ====新特性结束 ==== */
}
*当一个类继承父类又实现接口时,若后两者方法名相同,则优先继承父类中的同名方法,即“类优先”,如果实现两个同名方法的接口,则要求实现类必须手动声明默认实现哪个接口中的方法。
/**
* 人,是一个实现类
* 如果一个类实现一个接口,就要把接口中所有的抽象方法实现一遍
* 也可以实现接口中有方法体的方法(不能是静态方法)
* @author TeaBig
*/
public class Person implements IAnimal
{
@Override
public void run()
{
System.out.println("==Person==run");
}
@Override
public void eat()
{
/* 调用父类的方法 */
//IAnimal.super.eat();
System.out.println("==Person==eat");
}
}
/**
* 狗
* @author TeaBig
*/
public class Dog implements IAnimal
{
@Override
public void run()
{
System.out.println("==Dog==run");
}
}
测试类
/**
* 所有的程序的入口函数
* @author TeaBig
*/
public class ClientMain
{
public static void main(String[] args)
{
System.out.println("==南宋中兴四将==");
/* 测试第一个
* 父类引用指向子类对象
* */
IAnimal person = new Person();
/* 调用属性 */
int headNum = person.HEAD_NUM ;
System.out.println("==person.HEAD_NUM==" + headNum);
/* 接口名.属性 */
System.out.println("==IAnimal.HEAD_NUM==" + IAnimal.HEAD_NUM);
/* 调用方法 */
/* 调用抽象方法 */
person.run();
/*=====新知识======*/
/* 调用的是接口中有方法体的方法 */
person.eat();
person.sleep();
/* 调用的是接口中的静态方法 */
IAnimal.staticMethod();
/*==new对象==*/
// IAnimal dog = new Dog();
/* Dog::new 调用的是Dog中的new方法
* ::调用方法的简写,左边是类名或者接口名,右边是方法名
* 返回值必须是functional Interface;(等讲到lambda表达式的时候再说)
* 必须有返回值
*/
IAnimal dog = IAnimal.createObj(Dog::new);
System.out.println("==createObj==" + dog);
IAnimal dog1 = Dog::new;
/* 调用方法 */
//String res = dog::sleep;
}
}
action算子
数据流只能碰到一个action算子
例如:
public void other()
{
/* 准备一个List */
List<Integer> intList = Arrays.asList(1,2,3,4,5);
/* 准备一个Stream */
Stream<Integer> stream = intList.stream() ;
try
{
/* 把流里面的数据变成数组 */
// Object[] arrays = stream.toArray();
// System.out.println("==toString==>" + Arrays.toString(arrays));
/* 数据的数量 */
// long count = stream.count() ;
// System.out.println("==count==>" + count);
/* 查询每一个元素 */
// Optional<Integer> findFirst = stream.findFirst();
// System.out.println("==findFirst==>" + findFirst.get());
/* 获取到迭代器 */
// for (Iterator<Integer> iterator = stream.iterator(); iterator.hasNext();)
// {
// Integer intValue = (Integer) iterator.next();
// System.out.println("==循环==" + intValue);
// }
/* 获取最大值
* 先排序:(不管是升序还是降序,他始终认的是第一个)
* */
// Optional<Integer> maxOption = stream.max( (t1,t2) -> t1 - t2);
// System.out.println("==max=>" + maxOption.get());
/* 获取最小值 */
// Optional<Integer> minOption = stream.min( (t1,t2) -> t2 - t1);
// System.out.println("==min=>" + minOption.get());
/* 缩减
* 参数是:java.util.function.BinaryOperator<T>;得找有抽象方法的接口:
* java.util.function.BiFunction<T, U, R>
* 抽象方法是:R apply(T t, U u);
* 泛型T,U是输入参数,R是输出参数
* */
Optional<Integer> reduce = stream.reduce( (t1,t2) -> t1 + t2);
System.out.println("==reduce=>" + reduce.get());
}catch(Exception e)
{
if(stream != null)
{
/* 关闭 */
stream.close();
stream = null ;
}
}
}
}
action算子
public class DataStreamAction
{
/**
* 数据流测试的基本代码
*/
@Test
public void base()
{
/* 准备一个List */
List<Integer> intList = Arrays.asList(100,2,38,12,50,87);
/* 准备一个Stream */
Stream<Integer> stream = intList.stream() ;
try
{
/* 循环 */
stream.forEach( t -> System.out.println("==循环==" + t));
}catch(Exception e)
{
if(stream != null)
{
/* 关闭 */
stream.close();
stream = null ;
}
}
}
/**
* 数据流测试的基本代码
* 数据流,只能碰到一个action算子
*/
@Test
public void other()
{
/* 准备一个List */
List<Integer> intList = Arrays.asList(1,2,3,4,5);
/* 准备一个Stream */
Stream<Integer> stream = intList.stream() ;
try
{
/* 把流里面的数据变成数组 */
// Object[] arrays = stream.toArray();
// System.out.println("==toString==>" + Arrays.toString(arrays));
/* 数据的数量 */
// long count = stream.count() ;
// System.out.println("==count==>" + count);
/* 查询每一个元素 */
// Optional<Integer> findFirst = stream.findFirst();
// System.out.println("==findFirst==>" + findFirst.get());
/* 获取到迭代器 */
// for (Iterator<Integer> iterator = stream.iterator(); iterator.hasNext();)
// {
// Integer intValue = (Integer) iterator.next();
// System.out.println("==循环==" + intValue);
// }
/* 获取最大值
* 先排序:(不管是升序还是降序,他始终认的是第一个)
* */
// Optional<Integer> maxOption = stream.max( (t1,t2) -> t1 - t2);
// System.out.println("==max=>" + maxOption.get());
/* 获取最小值 */
// Optional<Integer> minOption = stream.min( (t1,t2) -> t2 - t1);
// System.out.println("==min=>" + minOption.get());
/* 缩减
* 参数是:java.util.function.BinaryOperator<T>;得找有抽象方法的接口:
* java.util.function.BiFunction<T, U, R>
* 抽象方法是:R apply(T t, U u);
* 泛型T,U是输入参数,R是输出参数
* */
Optional<Integer> reduce = stream.reduce( (t1,t2) -> t1 + t2);
System.out.println("==reduce=>" + reduce.get());
}catch(Exception e)
{
if(stream != null)
{
/* 关闭 */
stream.close();
stream = null ;
}
}
}
}