零基础学 Java
臧萌
PayPal 数据处理组技术负责人
46665 人已学习
新⼈⾸单¥68
课程目录
已完结/共 170 讲
第二章 Java面向对象编程 (74讲)
时长 07:13
时长 12:08
时长 06:26
时长 05:30
时长 14:16
时长 08:30
零基础学 Java
登录|注册
留言
11
收藏
沉浸
阅读
分享
手机端
回顶部
当前播放: 69 | 子类对象里藏着一个父类对象
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

登录 后留言

全部留言(11)

  • 最新
  • 精选
朱家华
如果使用组合,对于作为属性的类对象,我们无法修改其方法。有时候是因为没有权限,有时候是因为我们的习惯:我们不该为了一个类去修改作为其属性的另一个类的方法。对于组合这种方式,我们只有使用的权限,而没有修改的权限。 而对于继承,因为有覆写的存在,因此,虽然某个方法属于父类,但如果子类想要拥有区别于父类的属于自己的独特的方法,那么可以使用覆写机制。所以,对于继承这种方式,我们不仅可以使用,还可以修改。 即组合,我们从其他类简单地拿来了属性和方法;而继承,我们不仅仅从其他类拿来了属性和方法,我们还可以修改拿来的方法。 组合让两个类相对保持自己的独立性,而继承则让两个类相互渗透,因为继承更确切地说是子类和父类的融合。 类存在的目的是封装,而继承在某种程度上破坏了这种封装,让两个类紧紧耦合在了一起。

作者回复: 手动点赞,说的很好~

2021-01-04
9
Leo
has-a和is-a , 🤩我又成长了!

作者回复: 优先使用has a,谨慎使用is a。

2019-09-27
4
st
老师您好, 以HuaweiPhone和Phone的关系举例,如果不用继承,而是在HuweiPhone的属性里面引用Phone, 这种方式算是一种依赖注入吗?这种组合的方式会有一个问题是HuweiPhone的逻辑要依赖于Phone, 怎么解耦这种关系呢?

作者回复: 这个不算是典型的依赖注入,因为HuaweiPhone会有很多实例。比较典型的依赖注入是只有单个实例。 从组合的角度来说,并不存在耦合关系。因为HuaweiPhone是负责创建Phone的实例的。 以这个例子举例,如果Phone是别的代码创建的,然后HuaweiPhone需要用到这个Phone的实例。那么就算是依赖注入。 所以这个点在于,被依赖的对象(Phone)是谁创建的。在这个例子里,就是单纯的组合。 再举个例子。比如,A模块初始化好,才能初始化B模块。那么就是Spring擅长的依赖注入。如果B模块在初始化的过程中,自己用代码创建了一个A模块,那不需要依赖注入。

2019-08-03
4
Geek2808
老师请问为什么创建了一个Merchandise类型的属性,可以直接通过点运算符来调用方法呢?而我自己试了一下,它报错说merchandise没有初始化

作者回复: 创建一个Merchandise类型的属性变量,不是属性。 属性是指的Merchandise类内部的成员变量。 在方法里这样写是不行的,因为m是局部变量,局部变量没有初始化不可以使用。 Merchandise m; m.xxx 这样是可以的: Merchandise m = null; // 虽然肯定会报错,但是是可以通过编译的 m.xxx 或者这样 Merchandise m = new Merchandise(); m.xxx

2020-10-22
1
想问一下,为什么组合的时候不能在Phone类里定义一个buy方法,来实现对原Merchandise类Buy方法的覆盖?

作者回复: 这个问题,不确定我理解的对不对。我想你说的是,在Phone里定义一个buy方法,然后创建一个Merchandise的实力,用这个buy方法覆盖Merchandise实例的buy方法。 如果是这样的话,做不到的原因是Java里只有集成才有覆盖,也就是说,如果两个类没有继承关系,那就没法覆盖对方的方法。如果要Phone里的buy能够覆盖Merchandise类Buy方法,那么Phone就必须继承Merchandise类。但是有继承,就不叫组合了。 这个从理论上说,就牵扯到继承的具体实现了。像JS这种语言,是基于原型的继承(多态)。可以灵活的插拔更换各种方法/函数。Java是基于类这种类型的继承(多态),不能那么灵活。 但是换个角度,从设计的角度说,Phone本身,也不应该有buy方法。Phone就是Phone,有的是功能。只有Phone作为商品的时候,才有buy这种行为。当然,如果把Phone看作商品,那么继承Merchandise并覆盖buy方法没问题。

2020-05-28
1
小飞机
老师,new PhhoneHasAMerchandis的时候,后边的括号里为啥有 merchandise

作者回复: 构造方法的参数

2020-03-28
1
jibu
老师好!如果是两个属性高度重合的VO,比如新增VO和修改VO,后者只比前者多了个属性id,那么按照您的说法,这其实是用组合更合适,而非继承对吧?

作者回复: 如果只是属性重用,是的,组合更合适。 如果是属性和用到属性的方法被重用,进而需要被覆盖,继承/传递给子类这种,可以考虑继承。

2019-11-28
2
1
Geek_37542e
继承:手机是商品的一种 组合:商品属性是手机的一种属性 这样理解对吗?

作者回复: 🉑️

2022-03-24
追光
老师,看完了以后感觉对于什么时候用组合,什么时候用继承,还是有点不太明白 因为继承平常经常用,但是没怎么接触组合,所以能不能详细说说继承和组合的适用场景

作者回复: 这个没有严格的标准。用什么都是出于自己的判断。虽然道理很简单,is-a,用继承,has-a,用组合。 但是如果有人就是觉得“手机是一个有芯片和电池的屏幕”,是“is-a”,你也没法说他就是不对,他就是从那个角度看问题的。

2021-11-17
2
FRJ
老师很有趣,没什么废话效率很高,刷起来很舒服!

作者回复: :-)

2020-03-17
收起评论