https://blog.csdn.net/weixin_72388638/article/details/131691554
目录
一、前言
二、JAVA8
Lambda表达式
Stream API
创建方式
中间操作
终止操作
Optional类
三、JAVA9
模块机制
JShell交互式编程
接口
新增集合工厂方法
四、JAVA10
局部变量类型判断
五、JAVA11
Lambda表达式补充
String方法的补充
全新的HttpClient 使用
六、JAVA12-16
新的switch语法
文本块
新的instanceof语法
空指针异常的改进
记录类型(record)
七、Java 17
密封类型(sealed关键字)
总结
springboot3 最低dk17,java8估计要被弃用了
相当于匿名内部类的简写
public void test() {
Thread t1 = new Thread(()-> System.out.println("你好")); //lambda 表达式
t1.start();
}
规范:
(名称)-> {代码语句,包括返回值}
和匿名内部类不同,只支持接口,不支持抽象类
接口有且只有一个抽象方法
接口上有@FunctionalInterface 标识
自定义接口
@FunctionalInterface
public interface myBy {
Integer getAdd(Integer a,Integer b);
}
使用
public void test02() {
myBy myBy=(a,b)-> a+b; //实现接口
System.out.println(myBy.getAdd(1,2));
}
有现成的实现可以直接代替
private static Integer Add(Integer a, Integer b){ //加法
return a+b;
}
public void test02() {
// myBy myBy=(a,b)-> a+b;
myBy my=(OneTest::Add); //借用别人的实现方法
System.out.println(myBy.getAdd(1,2));
}
线程安全性,(十分局限,不允许对外部数据进行赋值操作)
错误案例
List
Integer a=0;
list.forEach(i->{
a+=i; //请注意,这个写法是错误的,lambda表达式为了保证线程安全不允许对外部数据进行赋值操作,当前编译阶段就会报错
});
System.out.println(a);
函数式编程
作用:对集合、数组操作
对比Collections(集合操作工具类),Arrays(数组操作工具类) :
Collections和Arrays会改变原来的集合数据;
Stream则不会,它会返回一个新的带有结果的Stream;
不想改变原有数据的情况下操作集合和数组可以使用Stream;
操作集合和数组建议使用这种方式,进一步减少被优化的可能,能少写for就少写一个
public void test012() {
List
// 顺序流 按元素顺序返回
Stream
// 并行流 ,同时取数据
Stream
//数组类型
int[] arr =new int[]{1,2,3,4,5};
IntStream stream3 = Arrays.stream(arr);
Stream
}
//Math.random() 生成 [0,1)的 double数
//生成10个数
Stream.generate(Math::random).limit(10).forEach(System.out::println);
数据过滤(filter)
//数据过滤 保留大于10的数据
// 当filter(e->{return true;}) 返回true才保留当前操作数据
List
list.stream().filter(e->e>10).forEach(System.out::println);
//结果 23,456,12,678,23
截取片段(limit)
List
//获得前4条数据 结果 1,23,456,9
list.stream().limit(4).forEach(System.out::println);
舍弃片段(skip)(截取片段 相反)
List
//丢弃前面4个数据,结果为12,678,23
list.stream().skip(4).forEach(System.out::println);
去重(distinct)
List
//去重 根据equals和hashCode, 结果为:1,23,456,9,12,678
list.stream().distinct().forEach(System.out::println);
映射(map)
自定义操作
map(e->{return "结果"}) 和filter类似, 只是return是自定义的结果
List
// 还是操作数据 ,自定义操作数据,用来返回一些特定数据; str.toUpperCase()转化成大写
// map(e->{return "结果"}) 和filter类似 只是return是自定义的结果
//结果 AA,BB,CC,DD
list.stream().map(str->str.toUpperCase()).forEach(System.out::println);
排序(sorted)
List
// 自然排序 ,结果:1,1,9,9,12,23,23,23,456,678
// 对类对象的排序需要实现 Comparable 接口
list.stream().sorted().forEach(System.out::println);
//自定义排序,即使是类对象也不用实现Comparable 接口
//自定义排序结果为:687,456,23,23,23,12,9,9,1,1
list.stream().sorted((a,b)->{
return Integer.compare(b,a);
})
.forEach(System.out::println);
匹配与查找
allMatch(lambda) 检查是所有元素满足条件
anyMatch(lambda) 检查是否至少匹配一个元素
noneMatch(lambda) 检查是否没有匹配的元素
findFirst() 返回第一个元素
findAny() 返回当前流中的任意元素
count() 返回流元素的总个数
max(Double::compare) 返回流中最大值
min(Double::compare)返回流中最小值
forEach(lambda) 内部迭代
匹配与查找示例:
// 1、allMatch 验证是否全部元素满足条件
List
// 是否全部元素大于 30;返回false,true;当前结果为false
boolean b = list.stream().allMatch(e -> e > 30);
System.out.println(b);
//2、noneMatch 检查全部元素,是否全部不满足条件
// 全部不满足条件为true
List
// 判断全部元素是否没有0,结果为true
boolean b1 = list.stream().noneMatch(e -> e == 0);
System.out.println(b1);
//3、findFirst 返回第一个元素
// 当前结果为 1
List
Optional
System.out.println(first);
// 4、 findAny 随机返回一个元素
List
// ?? 有BUG? 每次都是同一个结果
//Collections.shuffle(list); 还是洗牌好用
Optional
System.out.println(first);
//5、count 求元素总数
// 结果为 10
List
long count = list.stream().count();
System.out.println(count);
//5、max最大值 自然排列取右边值
//最大值
Optional
//Optional
System.out.println(max.get());
//6、最小值 自然排列取左边值
Optional
//Optional
System.out.println(min.get());
//7、遍历
// list.forEach(System.out::println); 集合自带了遍历
list.stream().forEach(System.out::println);
归约(reduce)
// 计算结果,返回计算结果
// 0初始值,lambda累加操作 (a是初始值也是下一次结果值当前初始值设置是0,b是索引为i的元素)
//结果为1235
Integer reduce = list.stream().reduce(0, (a, b) -> a + b);
收集(collect)
把结果转化为list或set等
// 把结果转化为list
List
List
System.out.println(collect);
//把结果转化为set
List
Set
System.out.println(collect);
可以预防空指针异常,
有if(a!=null) 时就优先使用这个类;
可以将任何一个类包装进Optional类中
API方法:
//创建
Optional.of(A) // A必须非空,A为空会报错
Optional.empty() //创建一个非空的Optional实例
Optional.ofNullable(A) //A可以为空 ,A为空不会报错; 使用这个
//返回数据
Optional.ofNullable(A).get() 返回包装的A
Optional.ofNullable(A).orElse(B) 当前A不为空就返回A,否则返回B,(A和B需要是同一个类型)
Optional.ofNullable(A).orElseGet() 如果有值则返回,否则返回有接口实现提供的对象
Optional.ofNullable(A).orElseThrow() 如果有值则将其返回,否则抛出由Supplier接口实现提供的异常
示例:
// 如果A为空就返回B
String A=null;
String a = Optional.ofNullable(A).orElse("A为空");
System.out.println(a);
private static void hello(String str){
Optional.ofNullable(str) //包装进入Optional,空的也会放入
.ifPresent(i->{
System.out.println(i); //打印
});
}
Optional不太方便,Lambda里面不能执行对外部属性执行赋值操作还是挺多限制的不如使用Objects.isNull(A)
module-info.java文件
//暴露出去
module module.a {
exports com.test;
}
// 只暴露给指定的模块
module module.a {
exports com.test to module.b;
}
//导入module.a 中暴露的内容
module module.b {
requires module.a;
}
//依赖的传递性
module module.a {
exports com.test;
requires transitive java。logging; // 传递到引用的模块
}
//暴露出去,并且开放反射权限
open module module.a {
exports com.test;
}
接口可以存在私有方法(私有方法必须要实现)
public void test04() {
Map
List
//list.add(6); 错误 list是只读的
System.out.println(map);
}
怎么变成js了,
public void test05() {
var a="hello,word"; // 只能定义在方法局部变量
// var b; 错误,不能定义未赋值的变量
System.out.println(a);
}
内可以定义 var类型变量;
public void test06() {
String str=" ";
System.out.println(str.isEmpty()); // 判断内容是否为空(有空格也是false)
System.out.println(str.isBlank()); //判断外形是否为空(肉眼观察,即使有空格也是true)
// str.repeat(2) 重复拼接2次成新字符串
// str.lines() 根据 \n分割
//str.strip() 去除首位空格
// str.stripLeading() 去除首部位空格
// str.stripTrailing() 去除尾部空格
}
public void test07() throws URISyntaxException, IOException, InterruptedException {
//创建
HttpClient client=HttpClient.newHttpClient();
//构建请求
HttpRequest request = HttpRequest.newBuilder(new URI("https://www.baidu.com")).GET().build();
HttpResponse
//发送请求
System.out.println(send.body());
}
老方式
private static String grade(int score){
score/=10;
String Gl="";
switch (score){
case 10:
case 9:
Gl="优秀";
break;
case 8:
case 7:
Gl="良好";
break;
case 6:
Gl="及格";
break;
default:
Gl="不及格";
}
return Gl;
}
switch允许有返回值,不写break
为switch表达式
private static String grade2(int score){
score/=10;
return switch (score){
case 10,9->"优秀";
case 8,7->"良好";
case 6->"及格";
default->"不及格";
};
}
返回值yield
private static String grade2(int score){
score/=10;
return switch (score){
case 10,9->{
//可以执行其他逻辑
System.out.println("优秀");
yield "优秀";
}
case 8,7->"良好";
case 6->"及格";
default->"不及格";
};
}
public void test09() {
// 普通字符串
String str="adbc\nsdawd";
System.out.println(str);
//文本块
String str2=
"""
文本块内容:
asaiddawjid
"sdfsef"fsef
mdkaslekf
""";
System.out.println(str2);
}
旧的用法
a instanceof A
true: a是A的实例或者 ,a是A类的后代
新补充:
a instanceof A b
为true时候相当于多执行了一步 A b=(A)a
可以直接转化对应的类型
//旧的写法
@Override
public boolean equals(Object obj) {
if(obj ==this) return true;
if (obj instanceof Student){
Student test=(Student) obj;
return test.name.equals(this.name);
}
return false;
}
//新写法省一行代码
@Override
public boolean equals(Object obj) {
if(obj ==this) return true;
if (obj instanceof Student test){ // ture时候把obj转化成 了(Student)test
return test.name.equals(this.name);
}
return false;
}
这个特性应该最容易感知出来(笑~)
有空指针异常时候打印的错误信息明确的指出了哪个参数空指针异常
在编译时候自动编译出 get、hashcode、equals、toString等方法
支持实现接口,不支持继承(位置已经被占用)
内部不能定义成员变量
//记录类型
public record Account(String user,String pwd) {
//private String adb; 错误,不能定义成员变量
//定义方法
public void Add(){
System.out.println("调用Add");
}
}
//使用
public void test010() {
Account account1=new Account("123","2334");
account1.Add();
System.out.println(account1);
}
以前final修饰的类不能被继承
sealed 修饰的类可以定点继承,只允许指定的类继承
要求:
1、可以基于普通类、抽象类、接口也可以是继承自其他抽象类的子类或是实现其他接口的类等
2、必须有子类继承,且不能是匿名内部类或者是lambda的形式
3、sealed写在原来final的位置,但是不能和final、non-selaled关键字同时出现,只能选择其一
4、继承的子类必须标记为final、sealed、non-sealed类型
示例:
//A类
public sealed class A permits B {
//只允许B继承A (且B需要是final或者sealed 修饰的)
public String name;
}
//B类
public non-sealed class B extends A {
// B为final时候不允许其他类继承
// B为sealed 允许指定类继承
// B为B是non-sealed ,重新开放继承权限,使其他类可以继承B来间接继承A ,(关键字怎么还带个 - ,难看)
}
//C类
public class C extends B{
}
// 测试
public void test011() {
A a=new A();
B b=new B();
C c=new C();
//A是sealed ,B是non-sealed ,c依旧继承了A中name属性
c.name="王";
System.out.println(c.name);
}