当前位置: 首页 > news >正文

内部类与Lambda的衍生关系(了解学习内部类,Lambda一篇即可)

内部类

在这里插入图片描述

成员内部类

特点:内部类可以访问外部类的所有成员 包括私有成员,同时拥有自己的成员,成员内部类的实列必须依赖于外部类的实例存在。

示例如下:

@Data
@AllArgsConstructor
@NoArgsConstructor
public class OutClass {private String name;String age;class InnerClass{public void innerMethod(){System.out.println(name);}}
}

方法调用

    OutClass outClass = new OutClass("张三","13");OutClass.InnerClass innerClass = outClass.new InnerClass();innerClass.innerMethod();
}

适用场景

匿名内部类适用于内部类的功能与外部类紧密相关,需要频繁访问外部类的实列对象,这种情况下 就是和使用成员内部类,比如说,图形绘制,内部类用于处理图形的颜色,线条等,这些属性与图形紧密相关。

静态内部类

使用static修饰的内部类,静态内部类属于对象本身 而不是对象的某个实例。静态内部类只能访问外部类的静态成员,不能访问实例成员。

示例如下

@Data
@AllArgsConstructor
@NoArgsConstructor
public class OutClass {private static String name = "学习静态内部类";String age;static class  InnerClass{public void innerMethod(){System.out.println(name);}}}

方法调用

    OutClass.InnerClass innerClass = new OutClass.InnerClass();innerClass.innerMethod();
}

适用场景

静态内部类一般用于工具类的创建,静态方法直接调用,如我们日常使用的一些工具方法包里面,就是提供一些静态方法,而静态内部类则是在此做了一层封装,它独属于当前类,而非实列对象。一般使用于,一些业务场景下,处理一些特殊情况。
局部内部类
定义在方法,代码块,构造函数中的类。作用域仅限于定义它的代码块内。局部内部类不能使用public,protected,private,static修饰。

示例如下

@Data
@AllArgsConstructor
@NoArgsConstructor
public class OutClass {private static String name = "学习静态内部类";String age;public void outMethod(){class  InnerClass{public void innerMethod(){System.out.println(name);}}InnerClass innerClass = new InnerClass();innerClass.innerMethod();}
}

方法调用

public static void main(String[] args) {OutClass outClass = new OutClass();outClass.outMethod();
}

适用场景

适用于某个方法实现一个特定的功能,这个功能只在该方法中使用,就可以使用局部内部类。例如:排序中 实现一个局部内部类用于处理特殊规则的排序。
匿名内部类
匿名内部类没有显示的名称,反编译后可以看到就是当前类名+$+序号。一般用于创建接口或者抽象类的实现类实列。匿名内部类的定义和实例化同时完成。
需要注意的是 ,如果对接口使用匿名内部类来创建,则需要保证接口中一定只有一个抽象方法。
如果不确定 还可以使用@FunctionalInterface进行检查 如果有其余方法 则会报错。

不过,这些不包含 jdk8提供的,写在接口中的方法,其中,static修饰的方法 与接口实现类无关,只能通过接口名.方法名()调用,而default修饰的方法 则是默认方法,是一种补充接口的方法,适用于 一些接口被很多类实现,后期维护需要新增一个方法,又不想在所有实现类中都写相同的代码,则提供了default方法。
这些改进,打破了接口只能包含抽象方法的限制,增强了接口的功能性和灵活性。
注意,default方法在实现类是可以重写的,与static修饰的方法不同。如:

static void people(){System.out.println("人");}
default void say(){people();System.out.println("=========");
}

示例如下

@FunctionalInterface
public interface MyInterface {void myMethod();
}

代码调用

public static void main(String[] args) {MyInterface myInterface = new MyInterface(){@Overridepublic void myMethod() {System.out.println("这是一个匿名内部类!");}};myInterface.myMethod();
}

使用场景

匿名内部类一般用于,事件创建,线程创建等场景。
Lambda表达式
静态方法引用
如果某个Lambda表达式只是调用一个静态方法 并且→前后参数一致 则可以使用静态方法引用。
通俗来说 就是只调用一个静态方法 并且入参和返回值与接口方法相匹配,就可以使用静态方法引用。
格式:类名::静态方法

示例如下

先写一个匿名内部类看看。

list.sort(new Comparator<Integer>() {@Overridepublic int compare(Integer o1, Integer o2) {return MathUtils.compareBy(o1,o2);}
});

我们可以看到,compare方法的入参是两个,返回值要求是int类型。而我们提供的静态方法 compareBy他也是要求 两个形参 然后返回一个int类型的值,如此 就满足静态方法引用,可以写成这样。

list.sort((o1,o2) -> {return MathUtils.compareByAbs(o1,o2);
});

我们可以看到 箭头前后的参数是一直的,因此,直接缩写。

public class MathUtils {public static int compareBy(int a, int b) {return Integer.compare(a, b);}
}
public static void main(String[] args) {List<Integer> list = Arrays.asList(1, -1, 2, 5, 3);list.sort(MathUtils::compareByAbs);System.out.println(list);
}

适用场景

适用于排序,过滤以及调用工具类中的静态方法来完成特定的逻辑。

实例方法引用

如果某个Lanbda表达式只是通过对象名称调用一个实例方法,并且 →前后参数一致 可以使用实例方法引用
格式:对象名::实例方法

示例如下

我们可以看到 ,只是对字母进行替换,全部改为大写后返回,使用匿名内部类完成这个操作。

public static void main(String[] args) {List<String> str = Arrays.asList("a","b","v");str.replaceAll(new UnaryOperator<String>() {@Overridepublic String apply(String s) {return s.toUpperCase();}});System.out.println(str);
}

紧接着 我们就可以简化,写一个实例方法,箭头前后参数一致,返回值满足replaceAll要求的String类型,如下:

public class StringProcessor {public String processString(String str){return str.toUpperCase();}
}public static void main(String[] args) {List<String> str = Arrays.asList("a","b","v");str.replaceAll(new StringProcessor()::processString);System.out.println(str);
}

当然,这个场景也适合使用静态类引用,都可以。

适用场景

适用于对集合中的数据进行处理,利用已有对象的方法,实现特定的功能。

特定类引用

如果某个Lanbda表达式只是调用一个特定类型的实例方法,并且前面参数列表中的第一个参数作为方法的主调,后面所有参数都是作为该实例方法入参的,则可以使用特定类型的方法引用。
格式:特定类名::方法

示例如下

public static void main(String[] args) {//字符串比较,忽略大小写List<String> str =Arrays.asList("b","A","d","c");str.sort(String::compareToIgnoreCase);System.out.println(str);
}

适用场景

适用于一些特定场景下,元素进行映射,过滤。

构造器引用

类型::new
某个Lanbda表达式只是在创建对象,并且→前后一致,就可以使用构造器引用了。
日常开发中 我们几乎用不到这种方式 了解一下即可。

示例如下

System.out.println(myInterface.say(););
http://www.lqws.cn/news/602533.html

相关文章:

  • Windows10/11 轻度优化 纯净版,12个版本!
  • 【分治思想】归并排序 与 逆序对
  • Nginx重定向协议冲突解决方案:The plain HTTP request was sent to HTTPS port
  • CertiK《Hack3d:2025年第二季度及上半年Web3.0安全报告》(附报告全文链接)
  • OEM怎么掌握软件开发能力
  • 记本好书:矩阵力量:线性代数全彩图解+微课+Python编程
  • Python OrderedDict 用法详解
  • 学习昇腾开发的第11天--主要接口调用流程
  • CMU-15445(6)——PROJECT#2-BPlusTree-Task#1
  • 记一次Ubuntu22安装MongoDB8并同步本地数据过程
  • 应急响应类题练习——玄机第四章 windows实战-emlog
  • 微信小程序使用秋云ucharts echarts
  • 高阶数据结构------并查集
  • 数据结构day7——文件IO
  • STM32——存储器映射(Memory mapping)
  • 反向传播 梯度消失
  • OSE3.【Linux】练习:编写进度条及pv命令项目中的进度条函数
  • 07CSRF 漏洞保护
  • vite项目中引入tailwindcss,难倒AI的操作
  • Modbus协议
  • 数字图像处理学习笔记
  • Spring IOC容器核心阶段解密:★Bean实例化全流程深度剖析★
  • 菜谱大全——字符串处理艺术:从文本解析到高效搜索 [特殊字符][特殊字符]
  • 城市灯光夜景人像街拍摄影后期Lr调色教程,手机滤镜PS+Lightroom预设下载!
  • 自由学习记录(66)
  • RESTful API 设计原则深度解析
  • 转录组分析流程(六):列线图
  • 笨方法学python-习题12
  • JavaScript 安装使用教程
  • 解码知识整理,使您的研究更高效!