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

登录 后留言

全部留言(10)

  • 最新
  • 精选
因为热爱
这是老师的程序~ public class Create2 { private static final String TEXT = "太阳在这个平静的小村庄缓缓升起,又是开始了平常的一天。我们故事的主人公睡眼惺忪的起来\n" + "......"; public static void main(String[] args) { System.out.println("程序开始,执行的线程名字叫做" + Thread.currentThread().getName()); for (int i = 1; i <= 3; i++) { Thread thread = new Thread(new PrintStoryRunnable(TEXT, 200 * i), "我的线程:" + i); thread.start(); } System.out.println("启动线程结束,名字叫做" + Thread.currentThread().getName()); } static class PrintStoryRunnable implements Runnable { private String text; private long interval; public PrintStoryRunnable(String text, int interval) { this.text = text; this.interval = interval; } @Override public void run() { try { double num = Math.random(); System.out.println("执行这段代码的线程名字叫做" + Thread.currentThread().getName()); printSlowly(text, interval); System.out.println(Thread.currentThread().getName() + "执行结束"); } catch (InterruptedException e) { e.printStackTrace(); } } } public static void printSlowly(String text, long interval) throws InterruptedException { for (char ch : text.toCharArray()) { Thread.sleep(interval); System.out.print(ch); } System.out.println(); } } 老师这个程序我可不可以这么理解,第一开始Java只有main一个线程,然后通过for循环创建了3个线程,并且几乎同时,这三个线程都开始了,之后main这个线程就结束了,然后里面创建的这3个线程依次结束,因为线程依次睡眠的时间增长,所以会依次结束,然后整个程序执行完毕。 老师您看我理解的对不对~,如果理解错误麻烦老师告诉我一下~谢谢老师🤣🤣🤣 还有就是我对老师之前讲的Java进程和线程直接的细微差别有点不太理解,如果用到这个程序中那又该如何去理解呢? 那我可以这样理解嘛~ 这个Java程序就是一个进程,当这个程序在运行的时候就是这个进程在运行,里面的一个个方法就是一个个线程, 首先只有一个main方法的线程,然后依次出现了3个线程,然后main方法线程结束,刚刚那3个线程依次结束, 最后是整个进程结束,整个程序结束,运行完毕。 老师你看看我理解的对不对~😂🤣🤣🤣

作者回复: 1)理解正确 2)线程和进程的差别是操作系统的范畴。简单来说,一个进程就是一个Java程序的运行时,线程则是进程可以创建出来的执行代码的最小单元。但是并非是一个方法就是一个线程。方法是方法,是代码,等着被执行。线程就是去执行代码的。一个线程被赋予一个方法之后,就会去执行这个方法,直到这个方法结束。执行过程中,这个方法可能会去调用别的方法,这时候并不会创建新的线程。

2020-12-31
6
Geek_b597bc
例程中线程中new了一个PrintStoryRunnable对象,而该类中只定义了run()方法,但没有看到调用的地方,最后怎么也执行了呢?

作者回复: Runnable的方法是被Thread类中的start方法调用的。 一个小技巧,如果你想知道一个方法是在哪里被调用的,在这个方法第一行设一个断点(breakpoint),然后debug运行程序,等程序停在断点之后,看IntelliJ左下角的thread frame就知道是谁调用的它了。

2020-08-15
3
100执行%
为什么断点打在System.out.println("启动线程结束,名字叫做" + Thread.currentThread().getName()); debug的时候 我们新建的线程没有执行run方法呢

作者回复: 断点停住的时候整个Java进程都是停止的。

2019-11-26
3
笨笨
老师如果有多个线程的话,它们是一起执行呢,还是有等某个线程执行完之后再去执行另外的线程呢?

作者回复: 线程会受操作系统的调度,互相切换着执行。也就是每个线程都会执行一会儿,然后让出CPU给别的线程执行。

2020-11-25
2
林伟华
可否这么理解 线程就是一个干活的人。方法就是一件事。 干活的这个人领到一件事 他干(执行)这件事(方法)。 在干(执行方法)过程中用到了其他的方法(其他事) 实际就是干活的人不变 变的只是事 可以这么理解吗。 如果是这有的话 我对于进程也是一个对象有点不大理解。这个对象是JAVA里面的还是 操作系统的啊

作者回复: 你绝对是有慧根哦!对线程的理解很到位。进程/线程是一个对象的意思是,Java可以用一个对象来封装一下进程/线程。并不是说进程/线程就是一个普通的Java对象,它核心的东西就是对应操作系统里的进程/线程

2022-07-19
1
朱家华
老师,我看之前的留言上,有一条回答说可以打个断点看一下自己新建的线程的线程栈,来看一下是谁调用的run()方法。我尝试了一下,下面是“我的线程-1”的栈信息: run:30, CreateThreadAppMain$MyPrintSlowly (com.cebj.javaBasic.whatisthread) run:748, Thread (java.lang) 第一行里的run方法是:自己定义的run{},即class MyPirintSlowly implements Runnable{run(){...}} 第二行里的run方法是,java.lang.Thread里定义的run()方法,方法内容为: /** * If this thread was constructed using a separate * <code>Runnable</code> run object, then that * <code>Runnable</code> object's <code>run</code> method is called; * otherwise, this method does nothing and returns. * <p> * Subclasses of <code>Thread</code> should override this method. * * @see #start() * @see #stop() * @see #Thread(ThreadGroup, Runnable, String) */ @Override public void run() { if (target != null) { target.run(); } } JDK版本号是:1.8.0_144 问题是: 从上边的线程栈可以发现,并不是像之前某个留言中老师说的那样,是Thread thread = new Thread(...); thread.start()中的start()方法调用的我们的run()方法,而是Thread.run()方法。我想问之所以和老师说的不一样,是因为JDK版本不一致吗?

作者回复: start调用了run方法是一种比较形象的理解方式。如果严格来说,start和run直接还隔着一个操作系统呢。 事情是这样的,start方法是和操作系统对接的,最终是要通过操作系统,启动一个线程。而这个启动的线程执行的方法就是run方法。 也就是说,从Java代码的角度来说,run方法才是调用别的方法的那个人,是一段程序执行的起点(也是终点,run方法执行完,线程也就结束了)。

2021-05-12
2
1
追光
老师这个多线程程序是并发还是并行,并发并行经常傻傻分不清

作者回复: 多线程就是很多条“流水线”在执行代码,concurrent还是 parallel可以不用分那么清楚

2021-11-30
笨笨
老师这个例子中是有两个线程吗?一个是main的一个是new()出来后,Thread.start().

作者回复: 对的。其实还有一些别的,不过跟我们的应用没关系了,是JVM自己创建的。

2020-11-25
张二旦
萌老师,你好哈,我想问下,调一个方法就启动一个新的线程,run里面调用了printSlowly方法,这是线程里面套线程吗(如果我这个问题问的有问题,别骂我啊)

作者回复: 调用方法不会启动新的线程,只会让当前的线程在当前的stack上创建一个新的frame,用来给方法里的局部变量,参数等分配内存空间。 只有Thread对象的start方法才会创建一个新的线程。 线程和线程之间是有父子关系的,还有thread group的概念,但是基本没啥大用。从执行的角度来说,线程就是执行代码的单元,不存在嵌套关系。

2020-08-26
追光
看了一下runnable和Thread代码,发现后者继承了前者
2021-11-30
收起评论