零基础学 Java
臧萌
PayPal 数据处理组技术负责人
46665 人已学习
新⼈⾸单¥68
课程目录
已完结/共 170 讲
第二章 Java面向对象编程 (74讲)
时长 07:13
时长 12:08
时长 06:26
时长 05:30
时长 14:16
时长 08:30
零基础学 Java
登录|注册
留言
13
收藏
沉浸
阅读
分享
手机端
回顶部
当前播放: 124 | Lambda V.S. 匿名类(上)
00:00 / 00:00
高清
  • 高清
1.0x
  • 2.0x
  • 1.5x
  • 1.25x
  • 1.0x
  • 0.75x
  • 0.5x
网页全屏
全屏
00:00
付费课程,可试看
01 | 课程介绍
02 | 内容综述
03 | 开发环境搭建(macOS)
04 | HelloWorld程序编译和运行(macOS)
05 | 开发环境搭建(Windows)
06 | HelloWorld程序编译和运行(Windows)
07 | 详解HelloWorld程序
08 | IntelliJ IDEA集成开发环境的安装和使用(macOS)
09 | IntelliJ IDEA集成开发环境的安装和使用(Windows)
10 | 从加减乘除到变量
11 | 再探计算加减乘除的程序
12 | Java中的基本数据类型
13 | Java中的运算符
14 | Java中的位运算符
15 | 基本数据类型的更多语法点
16 | 字符集编码和字符串
17 | 操作符和数据类型总结
18 | 程序执行流程之if-else语句(上)
19 | 程序执行流程之if-else语句(下)
20 | 程序循环之for语句
21 | 代码块和变量的作用域
22 | 程序循环之while语句
23 | 程序执行流程之switch语句
24 | 循环和判断的总结(上)
25 | 循环和判断的总结(下)
26 | 用数组保存成绩
27 | 认识变量和数组(上)
28 | 认识变量和数组(下)
29 | 多维数组
30 | 用数组灵活处理程序
31 | 类(class)
32 | 初探类和对象
33 | 认识引用类型(上)
34 | 认识引用类型(下)
35 | 类、对象和引用的关系
36 | 认识数组类型
37 | 引用的缺省值null
38 | 像自定义类型一样使用类
39 | Java中的包和访问修饰符(上)
40 | Java中的包和访问修饰符(下)
41 | 打造一个小超市
42 | IntelliJ调试程序初探
43 | 方法:让Merchandise对象有行为
44 | 返回值:让Merchandise计算毛利润
45 | 参数:让Merchandise计算多件商品的总价
46 | 参数和返回值是怎么传递的
47 | 分清参数、局部变量和实例的地盘
48 | 隐藏的this自引用
49 | 理解方法:一种特殊的代码块
50 | 理解方法的调用:代码的一种特殊跳转
51 | 给类和方法加Java注释
52 | 成熟的类的对象要自己做事情
53 | 方法的签名和重载
54 | 重载的参数匹配规则
55 | 构造方法:构造实例的方法
56 | 构造方法的重载和互相调用
57 | 静态变量
58 | 静态方法
59 | 静态方法的重载
60 | static代码块和static变量初始化
61 | 方法和属性的可见性修饰符
62 | 重新认识老朋友:Math和Scanner(上)
63 | 重新认识老朋友:Math和Scanner(下)
64 | 最熟悉的陌生人:String (上)
65 | 最熟悉的陌生人:String (下)
66 | 重新认识老朋友: main方法和System类
67 | String类的好兄弟
68 | 继承:方便让商品增加新的类别
69 | 子类对象里藏着一个父类对象
70 | 覆盖:子类想要一点不一样
71 | super:和父类对象沟通的桥梁
72 | super:调用父类的构造方法
73 | 父类和子类的引用赋值关系
74 | 多态:到底调用的哪个方法?(上)
75 | 多态:到底调用的哪个方法?(下)
76 | 多态里更多的语法点(上)
77 | 多态里更多的语法点(下)
78 | instanceof操作符
79 | 继承专属的访问控制:protected
80 | final修饰符(上)
81 | final修饰符(下)
82 | 继承里的静态方法
83 | 插曲:for循环的另一种写法
84 | 万类之祖:Object类
85 | hashCode和equals 方法(上)
86 | hashCode和equals 方法(下)
87 | toString方法
88 | 初探Class类
89 | 初探反射(上)
90 | 初探反射(下)
91 | 面向对象三要素:封装、继承和多态
92 | 枚举:定义商品的门类
93 | 接口:让商品类型更丰富(上)
94 | 接口:让商品类型更丰富(下)
95 | 抽象类:接口和类的混合体
96 | 有方法代码的接口
97 | 接口内代码的更多内容
98 | 静态内部类
99 | 成员内部类
100 | 局部内部类
101 | 匿名类
102 | 特殊类的总结
103 | 让我们的超市运转起来:设计篇
104 | 让我们的超市运转起来:代码篇
105 | 初识异常:try catch
106 | Java中异常的分类
107 | 抛出异常的语法
108 | Java异常的传递
109 | 自定义异常
110 | 异常传递不是凌波微步
111 | try catch finally语句
112 | 自动回收资源的try语句
113 | Java中的常见异常
114 | Collection类族简介
115 | Collection中的List (上)
116 | Collection中的List(下)
117 | Collection中的Set
118 | 泛型简析(上)
119 | 泛型简析(下)
120 | 再探泛型
121 | Iterator接口
122 | Map:key和value的映射
123 | 定义自己的注解
124 | Lambda V.S. 匿名类(上)
125 | Lambda V.S. 匿名类(下)
126 | 基本类型的自动装箱和拆箱
127 | Java中的File类
128 | Java I/O简介
129 | 写文件内容小程序
130 | 读文件内容小程序
131 | 网络通讯名词简介
132 | 简单的网络通讯小程序(上)
133 | 简单的网络通讯小程序(下)
134 | 简单的抓取网页内容的程序
135 | JDK和JRE
136 | 初识线程
137 | 创建自己的线程
138 | 再探线程
139 | 多线程:混乱开始了
140 | 同步控制之synchronized
141 | 同步控制之wait notify
142 | 多线程经典模型:生产者消费者
143 | 线程同步之join
144 | 死锁
145 | ThreadLocal线程专属的变量
146 | 定时任务
147 | volatile关键字的作用
148 | concurrent包基本原理
149 | concurrent包中的Atomic类族
150 | concurrent包中的锁
151 | concurrent包中的数据结构
152 | concurrent包中的线程池
153 | 聊天室开张喽 (上)
154 | 聊天室开张喽 (下)
155 | 什么是学习一门语言
156 | Java平台简介
157 | Maven概念简介
158 | Maven的安装和配置
159 | 创建一个简单的Maven项目
160 | 一个从pptx文件中抽取文字的小工具
161 | Maven常用命令和插件
162 | Intellij更多功能介绍
163 | 值得学习的类库简介
164 | 如何在Stack Overflow上提问才不会被骂
165 | 浅谈程序设计
166 | 游戏小程序功能定义
167 | 游戏小程序设计和模块划分
168 | 游戏小程序代码分析
169 | 使用Swagger创建一个Spring Boot的Web服务
170 | 结课测试&结束语
本节摘要

PDF 课件和源代码下载地址:
https://gitee.com/geektime-geekbang/LetsJava

登录 后留言

全部留言(13)

  • 最新
  • 精选
jibu
谈一下对本节使用的泛型的理解: List接口继承自Iterable接口,获得Iterable接口的forEach方法。 forEach方法的参数Consumer也是一个接口。 因此最终的实现类继承自List接口,调用forEach时,得让这个Consumer接口引用指向实现该Consumer接口的对象(代码中使用匿名类实现)。 foreach方法的Consumer引用使用了逆变泛型,要求实际使用的类型为T类或其父类。举个例子,假设B类继承自A类,则List泛型类型为B类(则Iterable泛型类型也为B类)时,用到的Consumer的泛型类型可以为A类。这时,Consumer的accept方法要求参数为A类引用,实际传入的是B类引用,符合“子类引用可以给父类引用赋值”的规范。

作者回复: ✅👍 设 M super T,那么M就是T的父类或者就是T本身(重),那么对于List<T>来说,范型本身的约束就是让List里的元素是T类型或者其子类(重)。 所以M肯定是T的父类/本身,List里的元素肯定是子类/本身。因此M的引用可以指向List里的元素👌,所以下面代码里,t当作Consumer的参数没问题。 default void forEach(Consumer<? super T> action) { Objects.requireNonNull(action); for (T t : this) { action.accept(t); } }

2020-05-10
5
朱家华
继续上一个留言的解释部分: 这里我们不在单独定义实现类,有两种方法: 1.匿名类 2.lambda 匿名类的实现原理本质就是定义了一个Consumer的实现类,然后new该类实例并作为ForEach()的参数。匿名类只是一种简便写法,本质没差别。 而lambda又是什么底层原理呢? 看着strs.forEach(s -> System.out.println(s + "_test")); 这种语法总觉得有点茫然。以前往方法里传过基本数据类型,传过引用数据类型,忽然传了串乱码一样的东西就不知道怎么回事了。 我尝试放开Consumer接口中acceptInt(int s)的注释,让Consumer接口有两个抽象方法。然后发现AppMain中的lambda表达式出错了。 报错信息是: Multiple non-overriding abstract methods found in interface com.cebj.javaBasic.learnLambda.demo1.Consumer 意思是说Consumer接口中有多个没有实现的抽象方法。感觉好像对lambda底层的机制有点眉目了。 前面用匿名类对象作为forEach()参数,本质上传递的是Consumer的实现类的对象。而lambda表达式由于让Consumer有了两个抽象方法就报错了,所以我推测lambda也是在底层自己搞了个类实现了Consumer接口(不然不会报错)。而 "s -> System.out.println(s + "_test")" 这个lambda表达式则变成了这个实现类实现的方法体 。然后Java将这个实现类的实例作为参数传到了forEach()中。 所以如果想使用lambda,需要满足以下条件 1.调用方法(forEach)的参数要定义成接口类型,即这里的Consumer要是一个接口: 2.接口中只能有一个抽象方法:Consumer中只能有一个抽象方法,不然Java去创建这个实现类的时候会不知道要实现哪个方法。因为lambda表达式其实只能反映一个方法的内容。 另外在定义lambda表达式的时候需要注意,要让lambda表达式满足接口中唯一抽象方法的参数、返回值、异常类型。

作者回复: 相信我,你这种学不明白就写写看,写不明白就问问看的方法,就没有学不会的东西👍 回到你的问题,你说的完全正确。lambda的接口只能有一个抽象方法。Java也是用这个抽象方法,再移花接木的一顿操作,最终调用了你写的代码。 你这学习方式太赞了!

2021-05-21
2
朱家华
老师,感觉视频里有个问题: IterateListLambdaAppMain这个类的代码中,讲到: myList.forEach(IterateListLambdaAppMain::processString); 这行代码时,说到: 如果不使用外部变量,可以使用上述的调用方式:类名::方法名 但是IterateListLambdaAppMain::processString这个函数是有参数的。 private static void processString(String str) { System.out.println(str); } 我的理解是这样的: 这里可以使用lambda表达式,即 (s) -> {System.out.println(s);} 也可以有替代的选择,那就是传入一个自己定义的方法,比如这里的IterateListLambdaAppMain::processString。 这个自己定义的方法可以有参数,可以有返回值,也可以是静态的以及非静态的,但不论什么样,这个方法要与接口Consumer接口中的抽象方法保持一致(参数,返回值,异常类型一致) 不知道我的理解对不对

作者回复: 在我看来,这其实是一种函数式编程的“恶趣味”,或者是一个标榜的优势:极力压缩代码量,压缩局部变量的创建(当然也挺好,毕竟我们写代码百分之九十的时间是在给变量方法类起名字😄)。 回到你的问题,我说的外部参数,就是除了lambda要遍历和处理的那个值之外的参数。这样就可以在没有二义性的前提下极力压缩代码量了

2021-05-21
2
1
Geek_6a0727
老师您好!lambda在Java的表现形式一共分多少种?

作者回复: 接着看,后面都有的

2020-02-24
1
追光
老师,为什么WindowAdapter()无法使用lambda替换?是因为需要覆写多个方法吗

作者回复: 是的,lambda只能有一个方法

2021-11-08
朱家华
为了更好地理解lambda,我把ArrayList的泛型去掉了,单纯模仿Java的Iterable和Consumer抽象,来自己实现一个forEach()。 由于留言只允许2000字,我超了,就分成两部分,代码和解释: 这里是代码部分: public interface Consumer { void acceptStr(String s); // void acceptInt(int s); } *************************** public interface Iterable { default void forEach(Consumer action) { for (int i = 0; i < MyArrayList.LENGTH; i++) { action.acceptStr(get(i)); } } String get(int index); } ****************** public class MyArrayList implements Iterable { public static final int LENGTH = 10; private String[] elements = new String[LENGTH]; public MyArrayList() { for (int i = 0; i < LENGTH; i++) { elements[i] = "str" + i; } } @Override public String get(int index) { return elements[index]; } } ***************************** public class AppMain { public static void main(String[] args) { MyArrayList strs = new MyArrayList(); System.out.println("---------------最原始的for循环---------------"); for (int i = 0; i < MyArrayList.LENGTH; i++) { System.out.println(strs.get(i) + "_test"); } System.out.println("---------------匿名内部类实现---------------"); strs.forEach( new Consumer() { @Override public void acceptStr(String s) { System.out.println(s + "_test"); } }); System.out.println("---------------lambda实现---------------"); strs.forEach(s -> System.out.println(s + "_test")); } } ****************************************

作者回复: 这种刨根问底的学习态度必须点个大大的赞👍👍

2021-05-21
朱家华
这节课13分钟,我看了快3天了,有很多体会,麻烦老师帮忙看一下对不对。 体会1: 对象拥有什么?其拥有属性,也拥有方法。那么当我们把对象作为方法参数,我们就既传递了数据,也传递了方法。虽然这些属性和方法都是对象级别的。但是拥有了对象,方法也可以通过反射得到类的static属性和方法,甚至构造函数。所以只要传递了对象,就得到了类中的一切。无论是哪种属性哪种方法,只要是类里的,我们都可以得到。 而lambda就两个特征:1.函数可以单独存在 2.函数可以作为参数或返回值 在Java8之前,Java做不到1,但部分做得到2。所以我是不是可以这样认为,Java8之前,Java其实可以部分实现lambda功能(即2,通过将方法封装在类里,借助类对象传递方法),但用着不太方便。 那Java8引入lambda,实际上是为了让语法更为简单,让函数式编程在Java中更好用。 体会2: 从接口的设计分析: Iterable和Consumer: Iterable负责遍历Collection中的元素,Consumer则负责对遍历出的每个元素执行操作。从抽象的角度看,Iterable抽象了遍历的功能,Consumer抽象了操作的功能。这两个功能本质上来说,应该是ArrayList的功能。(一个容器类,当然要负责遍历自己呀,遍历自己时执行的操作当然也要负责呀)。那Java为什么要把这两个功能定义成接口呢?这么设计的目的是是为了什么呢?这样做不是让设计更为复杂吗? 之前学到接口的时候,老师说接口其实是规范了一种行为,规范了一批实现类的行为。 就是说接口是多态的,一个接口可以指向不同的实现类,对应多种不同的实现。那调用者不用关注具体实现,只用关注接口中的规范即可。 那把类的行为抽离出来(抽象成接口)的好处就是: 1.将来好引入新的不同的实现,代码扩展性强 2.调用者只关注接口的规范,不用在考虑具体的实现细节,封装了细节。调用起来也舒服、优雅 那么基于上面的说法,我是不是可以这样理解,其实接口本质就是抽取。把一个类的某种行为抽取出来,形成一个接口,将来就可以让这个行为有不同的实现,提高代码可扩展性。 所以前面课程里说 “要面向接口开发,而不是面向实现开发” 本质就是说,我们要学会抽象,拿到一个现实对象,要去分析它有什么行为。然后分析多个现实对象的行为是不是有类似的地方,然后再抽象成更为抽象的行为?

作者回复: 体会1: lambda从最终的功能来说,更多的是方便,而且Java基础类库也增加了lambda接口的支持。相当于是标准化了。其实如果愿意的话接口就是lambda。一点解释:并非是之前的Java无法做到传递代码的功能,只是不标准不方便,就无法得到大规模应用。 体会2: 这俩行为是ArrayList的行为,但是也不仅仅是它的行为呀。Set也可以。所以用接口把这种遍历、操作元素的行为独立出来。谁想实现谁就用这个接口

2021-05-21
朱家华
老师有个问题: 例子1: public class AppMain { public static void main(String[] args) { List<String> strs = addElementsToList(new ArrayList<>()); strs.forEach(new Processer<String>()); strs.forEach(new Processor<String>()); } private static List<String> addElementsToList(List<String> list) { for (int i = 0; i < 5; i++) { list.add("str" + i); } return list; } } ******************* import java.util.function.Consumer; public class Processer<T> implements Consumer<T> { @Override public void accept(T t) { System.out.println(t + "_impl1"); } } ************************ import java.util.function.Consumer; public class Processor<T> implements Consumer<T> { @Override public void accept(T t) { System.out.println(t + "_impl2"); } } 这里在调用strs.forEach(new Processer<String>())时,我定义的Processer类其实实现了Consumer接口,因此才能作为forEach()函数的参数。但是IterateListLambdaWhereAppMain例子中,好像并没有人来实现Consumer接口 老师的例子: public class IterateListLambdaWhereAppMain { public static void main(String[] args) { List<String> myList = addElementsToList(new ArrayList<>()); // action.accept 就直接跳到了我们的方法,其实中间 Java 一顿后台操作 // 帮我们生成了一个匿名类来实现接口,并调用我们提供的方法 myList.forEach(IterateListLambdaWhereAppMain::processString); } public static List<String> addElementsToList(List<String> list) { for (int i = 0; i < 10; i++) { list.add("str" + (i % 5)); } return list; } private static void processString(String str) { throw new RuntimeException(); } } 那么按照上面的说法,如果我们想把自己定义的action作为forEach的参数,那我们定义的类需要实现接口Consumer。 那这里processString方法是定义在类IterateListLambdaWhereAppMain中的,我们并没有看到其实现了Consumer接口。那processString方法为什么能作为forEach()的参数呢? 老师说是Java在后台帮我们生成了匿名类,这点我理解不了,一个匿名类实现了Consumer接口,那他一定是自己实现了自己的accept方法。那么调用的时候应该跳转到该匿名类的accept方法内部去执行,为什么在debug的时候我们发现它跳转到了我们的processString方法中呢?

作者回复: 那你看看是谁调用的process String呢?debugger可能会有一些调用栈上的优化。可以new一个exception,然后printStackTrace看看实际的调用栈。

2021-05-21
杨松
老师,请教个问题;Supplier<String> supplier = String::new; 这条语句是属于静态方法调用,还是参数类型里的方法调用,还是其他的方式,分不清了。。。。。

作者回复: 这个就是lambda语句,最终会翻译成一些完成lambda需要的代码,当然完成功能的还是new String

2021-01-20
因为热爱
滴滴滴~来分享一下我在这节课遇到的问题,以及我现在是怎样把这个问题解决来的 首先我遇到的问题是:老师讲解的那个静态内部类不太理解 那来附上代码: // TODO 匿名内部类版 myList.forEach( new Consumer<String>() { @Override public void accept(String s) { processString(outside + s); } } ); 这个问题我是对着下面的关于lambda方法的简化版去解决的 首先我在myList.forEach(IterateListLambdaAppMain::processString); 这段代码处设置了一个debug 然后去看这个代码是如何调用的,首先呢,我们是往这个forEach里传了一个这个类内部的一个processString方法进去,然后呢这个forEach去调用了Arraylist里的forEach这个方法,这个方法里面有一个accept方法,这个方法呢又去调用了processString方法里设置好的一个输出语句,最后输出,到这里整个调用就完成了。我们就可以把我们往myList里添加的元素,一个个的输出出来 那么对比分析老师讲的那个匿名内部类版的myList.forEach(。。。)的这个方法是不是可以一样的去对比者去想呢? 还是一样首先往for里去传一个参数,这个参数里有一个内部类。我们的myList.forEach就会去调用这个内部类,这个内部类就会执行自己的输出语句操作,从而将我们一开始向我们往mylist里添加的元素,一个个的打印出来。 都是一个道理。但是我想了很久才明白🤣🤣🤣 属实有滴笨勒😅 希望大家如果出了一样滴问题,也都能想的开,哈哈哈哈哈哈哈哈~

作者回复: 很棒棒哦,lambda其实有点绕的。

2020-12-09
2
收起评论