下载APP
登录
关闭
讲堂
算法训练营
Python 进阶训练营
企业服务
极客商城
客户端下载
兑换中心
渠道合作
推荐作者
当前播放: 104 | 让我们的超市运转起来:代码篇
00:00 / 00:00
标清
  • 标清
1.0x
  • 2.0x
  • 1.5x
  • 1.25x
  • 1.0x
  • 0.5x
网页全屏
全屏
00:00
付费课程,可试看

零基础学Java

共170讲 · 170课时,约1500分钟
2657
免费
01 | 课程介绍
免费
02 | 内容综述
免费
03 | 开发环境搭建(macOS)
免费
04 | HelloWorld程序编译和运行...
免费
05 | 开发环境搭建(Windows)
免费
06 | HelloWorld程序编译和运行...
免费
07 | 详解HelloWorld程序
08 | IntelliJ IDEA集成开发环...
09 | IntelliJ IDEA集成开发环...
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和Sc...
63 | 重新认识老朋友:Math和Sc...
64 | 最熟悉的陌生人:String ...
65 | 最熟悉的陌生人:String ...
66 | 重新认识老朋友: main方...
67 | String类的好兄弟
68 | 继承:方便让商品增加新的...
69 | 子类对象里藏着一个父类对...
70 | 覆盖:子类想要一点不一样
71 | super:和父类对象沟通的...
72 | super:调用父类的构造方...
73 | 父类和子类的引用赋值关系
74 | 多态:到底调用的哪个方法...
75 | 多态:到底调用的哪个方法...
76 | 多态里更多的语法点(上)
77 | 多态里更多的语法点(下)
78 | instanceof操作符
79 | 继承专属的访问控制:prot...
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 | 同步控制之synchronize...
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创建一个Spri...
170 | 结束语
本节摘要

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

精选留言(12)

  • 2019-08-17
    臧萌老师好,请教例程中一段代码“ Merchandise[] all = new Merchandise[Category.values().length * merchandisePerCategory]; ”,Merchandise是Interface类型,接口的定义里并没有构造函数,语法角度如何去理解这段代码的含义?谢谢。

    作者回复: 假设Merchandise是接口。
    这里是定义了一个接口的引用的数组。将来这些接口的引用,可以指向任何一个实现了这个接口的类的实例。
    就好像:
    Merchandise m1 = null;
    Merchandise m2 = null;
    ....

    m1 = new Phone();
    m2 = new 任意实现了Merchandise接口的类。

    2
    1
  • 2019-08-01
    老师你写的接口里有参数在实现方法中没用到,为什么定义那几个参数,这不白输入码??

    作者回复: 接口的定义是对问题的抽象,并非是迁就于实现的。

    就好像最常见的USB接口,提供了数据传输的功能,同时也提供了供电的功能。但是它并没有要求每个USB设备都必须同时用到所有的它提供的功能。

    一个USB小风扇,使用供电的功能就够了。不能强求它说这个接口是USB的,不传输点数据就是白用USB了。接上USB,能供电,风扇可以转,这就已经实现了这个USB设备的应有的功能了。没有什么白不白的。因为这个接口,不是为了一个设备设计的。改天插上U盘,就用得上数据传输的功能了。

    如果专门在电脑上设计一个仅供充电的USB口,一个仅供数据传输的USB口,一个既能充电又能传输数据的USB口。不是不可以,但是没必要。

    最后一个题外话。问问题,一个问号和两个问号,没有什么不一样。我发现你问的问题,都是俩问号。那第二个问号,才是真的白输入了哦亲。

    1
  • 2019-09-15
    有个疑问,为啥要把scanner声明为 static final的呀,这样的话不是会变成字面值吗?
    花了两天才模仿老师的例程实现了一片,感觉平时在实验室写C++和matlab就是调一调cv现成的库,或者就是写个类(函数)也就是吧公式的计算封装一下,完全不算是面向对象,学习这种面向对象的时候,类和类之间的联系和嵌套好多啊,类和类的关系总是捋不清楚,好不习惯.学习速度比预期要慢很多.....

    作者回复: 字面值是literal value。只有基本数据类型和string才有。

    final只是不可变值。用它是这个变量的值确实不需要变化。

    写独立完整的程序确实需要心中有对整个系统的把控和理解。调api和写完整的程序,就像装修和盖楼的区别,都有各自的场景,面对的问题不同,解决方式也就不一样了。

  • 2019-08-03
    /////老师每次购物不是都会新建一个customer吗,
    double moneyEarned = customer.payFor(shoppingCart, finalCost);
     public double payFor(ShoppingCart shoppingCart, double totalCost) {

            // TODO 留给大家的思考题,买不起怎么办?

            moneySpent += totalCost;///这里不是购物结束了,下次就是另一个customer直接=不是跟简单吗+有什么用吗,不懂?

            return totalCost;
    展开

    作者回复: 和语法无关的代码有没有用的问题,你觉得它没有用,那就是没有用。这个没有什么标准或者准则。

    比如说这里,如果调试代码的时候把它的值输出出来,那就有用了。

    比如如果要检查是不是买得起,可能就有用了。

  • 2019-08-03
    public void serveCustomer(Customer customer) {

            int maxTypeToBuy = MAX_BUY_DEFAULT;/////老师这句什么用下句不是会赋值吗,你说的缺省
    跟这有什么关系不懂?

            if (customer instanceof AbsCustomer) {

                maxTypeToBuy = ((AbsCustomer) customer).getGuangCount();

            }
    展开

    作者回复: 编程行有个名词叫做“保护性编程”。意思就是,要假设别人会犯错,当然,自己也会犯错。

    比如上面的代码。只有在customer instanceof AbsCustomer的情况下,maxTypeToBuy才会被赋值。

    或者纵观例程里现有的所有代码,给maxTypeToBuy 赋值 MAX_BUY_DEFAULT都是白费。因为都会在别的地方给覆盖掉。

    但是保护性编程的理念是,无论现在代码是什么样子,我要在当时当下,当行代码,尽力做到万无一失。

    来跟我重复一遍:尽力做到万无一失

    maxTypeToBuy对于程序来说是相当重要的一个变量。它直接控制了后面程序会不会运行。如果maxTypeToBuy在某些情况下(可能是未来,可能是bug),没有被赋值,那么maxTypeToBuy的值将会是0。后面所有购买的情况也就不会发生了。

    同学其实你很多问题都是站在“面对当下编程”的角度看代码。接口里有个参数用不到,某个赋值是“白赋值了”,都是这种想法造成的。

    多写代码,多看代码,找到感觉之后你就会知道为什么需要“保护性编程”,“面向接口编程”,因为人都会犯错误,因为人的想法总会变化。如果“面对当下编程”,程序的灵活性和健壮性会很不足。代码里一个疏忽,程序可能就不能正常运转了;小小的一个需求变化,可能会需要代码从头到尾改一遍。

  • 2019-07-31
    for (Merchandise m : toChoose) {

                    if (m == null) {

                        continue;

                    }

                    int buyCount = customer.buyMerchan
    ////////////////如果customer没有指向SuiYuanCustomer或ThinkAndBuyCustomer那customer.buyMerchandise(m)是指哪个

    public void serveCustomer(Customer customer) {

            int maxTypeToBuy = MAX_BUY_DEFAULT; ////////// 如果customer指向SuiYuanCustomer或ThinkAndBuyCustomer 这个不是会被下句盖掉吗??

            if (customer instanceof AbsCustomer) {

                maxTypeToBuy = ((AbsCustomer) customer).getGuangCount();

            }
    展开
  • 2019-07-31
    接口的参数并非都要在某一个实现里全用到的。 public double processCardDiscount(double totalCost, double totalCostAfterDiscount,Customer customer, ShoppingCart shoppingCart)
    可是这个接口不就两个实现吗,都没用到啊??
  • 2019-07-31
    Category category = customer.chooseCategory();

                // 简单的导购员,顾客说不想买就算了,不做推荐

                if (category == null) {

                    continue;

                }老师这里category 没有可能为空啊这不是白问吗??
    展开
  • 2019-07-31
    public double processCardDiscount(double totalCost, double totalCostAfterDiscount,

                                          Customer customer, ShoppingCart shoppingCart) {

            return totalCostAfterDiscount * (1 - discount);

     public double processCardDiscount(double totalCost, double totalCostAfterDiscount,

                                          Customer customer, ShoppingCart shoppingCart) {

            // 如果折扣下来剩下的钱比点数少,那么就抵扣掉需要付的剩下的钱

            if (totalCostAfterDiscount < point) {

                point -= totalCostAfterDiscount;

                return totalCostAfterDiscount;

            } else {

                // 否则就抵扣掉所有的点

                point = 0;这个不应该去掉吗,要不返回的不就是0吗??

                return point;

            }

        }double totalCost,Customer customer, ShoppingCart shoppingCart这三个参数方法中没用到
    啊??
    展开

    作者回复: 接口的参数并非都要在某一个实现里全用到的。

  • 2019-07-31
    public void serveCustomer(Customer customer) {

            int maxTypeToBuy = MAX_BUY_DEFAULT;

            if (customer instanceof AbsCustomer) {

                maxTypeToBuy = ((AbsCustomer) customer).getGuangCount();

            }
    老师你说的缺省是指customer没有指向AbsCustomer或它的子类吗,
     for (Merchandise m : toChoose) {

                    if (m == null) {

                        continue;

                    }

                    int buyCount = customer.buyMerchandise(m);
    可是如果customer缺省的话没有指向SuiYuanCustomer或ThinkAndBuyCustomer那customer.buyMerchandise(m)是指哪个
    展开

    作者回复: 缺省应该是指的AbsCustomer里定义的方法,没有被其子类覆盖掉

  • 2019-07-26
    getRandomMerchandiseOfCatagory()if里为啥要有MAath.Random>0.5

    作者回复: 这才是方法里Random的奥义,否则每次选出来的商品对于同一种类型都是确定的了。

  • 2019-07-26
    老师那些代码网站上没有啊,只有接口

    作者回复: 少年你不打算自己写写看吗?哈哈,下周一会同步上去的。