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

登录 后留言

全部留言(20)

  • 最新
  • 精选
彧豪
老师你好,这节课我有这么几个问题: 1. Storage storage = new Storage(); storage.producer = "Intel"; storage.capacity = 512; Screen screen = new Screen(); screen.size = 15.6; screen.producer = "戴尔"; 这段代码创建了2个对象,那创建了几个引用呢?是4个吗?storage storage.producer screen screen.producer这4个? 2. 情形一: Phone phone = new Phone(); phone.price = 1999; phone.prePhone = phone; phone.prePhone.price = 2999; System.out.println(phone.price); //2999.0 System.out.println(phone.prePhone.price); //2999.0 情形二: Phone phone = new Phone(); phone.price = 1999; phone.prePhone = new Phone(); phone.prePhone.price = 2999; System.out.println(phone.price); //1999.0 System.out.println(phone.prePhone.price); //2999.0 两种情形输出的结果不一样,是因为情形一种phone和phone.prePhone都是指向的同一个引用:phone;而情形二中phone是new Phone(), phone.prePhone又是另一个new Phone(),phone和phone.prePhone指向的是两个不同的引用的缘故吗?

作者回复: 1)对 2)对 你理解的很对,比起我当年理解的快多了,赞👍 其实这俩问题都和对象创建以及引用指向对象有关。 1)里面,四个引用分别是代码创建了两个,用来指向对象(指向公告板上的小本子),创建每个对象时,又分别创建了一个,是对象内部的成员变量。 2)也是一样,情况1只有一个本子(对象),改变price都是在操作同一个本子(对象)的同一张纸(成员变量),情况2是两个本子(对象),每个本子(对象)都有price。

2020-01-11
3
7
彧豪
老师,我有个关于String的问题想问你 1. 如下代码,为何没有输出内容, 直接报错: public class MyPhone { public static void main(String[] args) { String str; System.out.println(str); //Error: 可能尚未初始化变量str } } 2. 而如下代码却能输出初始值null; public class OnlyString { String str; } public class MyPhone { public static void main(String[] args) { OnlyString oS = new OnlyString(); System.out.println(oS.str); //null } } 3. String数组的初始值也都是null: public class MyPhone { public static void main(String[] args) { String[] strArr = new String[3]; for (int i = 0; i < strArr.length; i++) { System.out.println(strArr[i]); //null null null } } } str被创建出来,不是默认初始值是null吗,为何第一种那样会直接报错呢?

作者回复: 成员变量的引用类型会被初始化为null,局部变量(方法里创建的变量)不会被初始化,没有初始化就使用的话会有编译错误

2020-01-10
6
carZycriss
public class myphone { public static void main(String[] args) { phone xiaomi = new phone(); zhuban zhuban1 = new zhuban(); xiaomi.price = 1999; xiaomi.zhuban.daxiao=1; System.out.println(xiaomi.price + xiaomi.zhuban.daxiao.n); } } 老师只是我的main方法函数, public class zhuban { String producer; int daxiao; int chuchang; } public class phone { zhuban zhuban; int price; } 这是另外几个类 然后报错这样 Exception in thread "main" java.lang.NullPointerException at myphone.main(myphone.java:8) 我不知道错在哪,请问一下老师。

作者回复: phone xiaomi = new phone(); zhuban zhuban1 = new zhuban(); 然后你应该让xiaomi对象的zhuban这个引用指向你新创建出来的对象, 也就是下面这一句: xiaomi.zhuban = zhuban1 P.S. 类名建议按照习惯, 使用首字母大写.

2019-10-31
3
6
因为热爱
老师12.05分的时候你说有几个new就有几个对象,我数了一下是6个,你说成了5个 发现一处口误,哈哈哈哈~������

作者回复: bingo,是的我说错了,应该是6个(不包括String在内)

2020-09-24
4
bahao
老师你好,视频中看到你给对象成员属性赋值是用 对象.属性 的方式,这让我想到工作中一般情况是:成员属性设置为private,而开放public的get/set方法来对成员属性赋值,不知道这样的设计用意是什么,就好像家长想把小孩锁在家里好好学习,然后又开了一扇窗,暗示可以跳出窗外玩耍,这样的话,直接不锁门不就得了吗?还请赐教

作者回复: 工作中的方式是对的,这个在后面也会说。 从单纯的存取变量值这个功能的角度来说,声明为public当然最方便。 但是这从可管理可扩展上来说就不方便。方法是代码,代码就可以执行逻辑。比如后面会学到的synchronized,或者在set方法里检查一下value的正确性,在get的时候做个复制等等。getset分开也更好控制,比如只给get不给set方法等。更不用说以后可以用aop做各种钩子骚操作。 所以你举的那个例子,不是很贴切。你可以这么想。你有一台电脑,要借给别人用。那么你是直接给TA一个root账号,让TA爱怎么用怎么用呢?还是给他创建一个普通账号,然后在他需要使用root权限的时候,请求你的允许呢? 直接给root权限,就是完全不可控,别人完全可以把你的电脑给弄的面目全非,对比这里就是给成员变量随便赋值,不受任何限制。当然,这确实“方便”。但是方便都有代价。 还有一种想法就是“反正这个类只有我一个人用”。其实这也是不对的。绝大多数人学编程,最终都是要和别人合作写代码的。养成习惯很重要。 我这里说的也不全面,很多东西是只可意会,不可言传,或者要在某个具体的场景下才能体会到。用get和set访问成员变量,是面向对象编程的一个最佳时间,代码写的越久越会有感受。

2021-05-15
1
Cooper
public class MyPhoneMaker { public static void main(String[] args) { Phone phone = new Phone(); phone.hasFigurePrintUnlocker = true; phone.price=1999; phone.screen = new Screen(); phone.screen.producer= "京东方"; phone.screen.size=8.8; CPU cpu = new CPU(); cpu.producer = "三星"; cpu.speed = 3.5; Memory memory = new Memory(); memory.producer = "三星"; memory.capacity = 4; Storage storage = new Storage(); storage.capacity = 128; storage.producer = "三星"; phone.mainboard = new Mainboard(); phone.mainboard.cpu = cpu; phone.mainboard.memory = memory; phone.mainboard.storage = storage; phone.mainboard.model = "EA888"; phone.mainboard.year = 2019; System.out.println(memory.producer); } } 这个代码里 phone.mainboard.cpu = cpu; phone.mainboard.memory = memory; phone.mainboard.storage = storage; 这几行代码是什么意思呢?删掉对程序有影响吗?

作者回复: 这是给对象的属性赋值 比如phone.mainboard.cpu=cpu,就是给phone的mainboard的cpu属性赋值,赋为什么呢?局部变量cpu代表的值。 其余的类似。

2020-08-29
1
Geek_199e7e
public class PhoneMaker { public static void main(String[] args) { // 创建一个Phone的实例 Phone phone = new Phone(); // 可以设置断点查看实例内部状态 phone.hasFigurePrintUnlocker = true; phone.price = 2999.9; // 创建一个Screen的实例,给phone实例的screen引用赋值 phone.screen = new Screen(); // 通过引用phone找到Phone的对象,通过screen找到Screen对象,通过.producer给之赋值 phone.screen.producer = "京东方"; phone.screen.size = 6.6; Mainboard mainboard = new Mainboard(); mainboard.model = "EA888"; mainboard.year = 2019; mainboard.cpu = new CPU(); mainboard.cpu.producer = "德州仪器"; mainboard.cpu.speed = 3.5; mainboard.memory = new Memory(); mainboard.memory.producer = "三星"; mainboard.memory.capacity = 4; mainboard.storage = new Storage(); mainboard.storage.producer = "Intel"; mainboard.storage.capacity = 256; // 想想我们创建了几个对象,有创建了几个引用? } } 老师,这段代码为什么是创建了5个对象? 你说有几个new就是有几个对象, 那么明明有6个new啊,phone.screen = new Screen();不算创建对象吗?为什么?

作者回复: Sorry,算的。数数数错了……

2020-08-23
1
Hugo
String str1 = new String("abc"); String str2 = "abc"; String str3 = "abc"; System.out.println(str1.equals(str2)); //true System.out.println(str1==str2);//false System.out.println(str3==str2);//true 臧萌老师,你好 str1.equals(str2) //true 是因为equals方法 比较str1和str2地址指向的值, str1==str2 //false 是因为 “==” 比较的是内存地址,String str1 = new String("abc"); 是在内存中开辟新的存储空间,而 String str2 = "abc"; 在不同的存储空间,所以就是false, str3==str2 //true str3、str2 值是“adc” 在内存中str3、str2都指向了“adc”。 String str1 = new String("abc"); String str2 = "abc"; str1 和str2 在内存中是怎么存储的了?new 关键字,没搞清楚new关键字, 实例化对象 在内存中是怎么存储,str1和str2 在 不同的堆栈中吗?

作者回复: 首先,Java中所有的对象都是在堆(heap)中的,new 操作符就是在堆中分配一块内存,然后存放对象的数据。 FAQ里的《02章-Java 中的内存是怎么分配的 .md》供你参考。 https://github.com/geektime-geekbang/LetsJava/blob/master/FAQ/02%E7%AB%A0-Java%20%E4%B8%AD%E7%9A%84%E5%86%85%E5%AD%98%E6%98%AF%E6%80%8E%E4%B9%88%E5%88%86%E9%85%8D%E7%9A%84%20.md

2020-03-09
2
1
彧豪
哈哈哈,谢谢老师! 以及,仔细看了你的回复,我又有了一个疑问,哈哈,或者是想法吧: public class Storage { long capacity; String producer; String model; } Storage storage = new Storage(); 此时这里,创建了1个对象没有疑问,而引用是不是就是3个了呢?就是创建Storage的实例/对象的时候创建出了3个对象, 就是它里面的非基础类型的成员变量,和后面是否对Storage的成员变量赋值无关了呢?

作者回复: Java中除了string对象,所有对象都是new操作符创建出来的。storage对象里只有引用,创建出来时引用没有指向任何对象。

2020-01-12
2
1
深水蓝
请问一下老师,基本的数据类型可以通过强制类型转换实现转换。那么自定义类型是不是也有类似强制类型转换的方法的呢?

作者回复: 后面会提到,有强制类型转换

2022-02-28
收起评论