1把锁:synchronized
2个并
并发(concurrent)
- 是在同一实体上的多个事件
- 是在一台机器上“同时”处理多个任务(一对多)
- 同一时刻,其实是只有一个事情再发生
并行(parallel)
- 是在不同实体上的多个事件
- 是在多台处理器或多核心上同时处理多个任务
- 同一时刻,大家都在做事情,你做你的,我做我的
3个程
进程
在操作系统中运行的一个应用程序就是一个进程,每个进程都有它自己的内存空间和系统资源
线程
也被称为轻量级进程,在同一个进程内会有1个或多个线程,是大多数操作系统进行时序调度的基本单元
线程分类
用户线程
是系统的工作线程,它会完成这个程序需要完成的业务操作(一般不做特别说明,默认说的都是用户线程)
守护线程(Daemon)
- 守护线程是一种为其他线程服务的特殊线程,在后台默默地完成一些系统性的任务,比如垃圾回收线程就是最典型的例子
- 守护线程作为一个服务线程,没有服务对象就没有必要继续运行了,如果用户线程全部结束了,意味着程序需要完成的业务操作已经结束了,系统可以退出了
- 当系统只剩下守护线程的时候,JVM会自动退出
这段代码有两个线程
- t1
- main主线程
【t1是用户线程】
public class DaemonDemo {
public static void main(String[] args) {
Thread t1 = new Thread(() -> {
System.out.println(Thread.currentThread().getName() + " 开始运行," + (Thread.currentThread().isDaemon() ? "守护线程" : "用户线程"));
while (true) {
}
}, "t1");
t1.start();
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " ----end 主线程");
}
}
【t1改为守护线程】
public class DaemonDemo {
public static void main(String[] args) {
Thread t1 = new Thread(() -> {
System.out.println(Thread.currentThread().getName() + " 开始运行," + (Thread.currentThread().isDaemon() ? "守护线程" : "用户线程"));
while (true) {
}
}, "t1");
// 通过设置属性Daemon来设置当前线程是否为守护线程
t1.setDaemon(true);
t1.start();
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + " ----end 主线程");
}
}
注意:setDaemon(true)方法必须在start()之前设置,否则报IlegalThreadStateException 异常
管程
管程就是我们平时的Monitor锁对象。Monitor其实是一种同步机制,它的义务是保证(同一时间)只有一个线程可以访问被保护的数据和代码,JVM中同步是基于进入和退出监视器(Monitor管程对象)来实现的,每个对象实例都会有一个Monitor对象,Monitor对象和Java对象一同创建并销毁,底层由C++语言实现。
执行线程就要求先成功持有管程,然后才能执行方法,最后当方法完成(无论是正常完成还是非正常完成)时释放管程。在方法执行期间,执行线程持有了管程,其他任何线程都无法再获取到同一个管程。
文章说明
该文章是本人学习 尚硅谷 的学习笔记,文章中大部分内容来源于 尚硅谷 的视频尚硅谷JUC并发编程(对标阿里P6-P7),也有部分内容来自于自己的思考,发布文章是想帮助其他学习的人更方便地整理自己的笔记或者直接通过文章学习相关知识,如有侵权请联系删除,最后对 尚硅谷 的优质课程表示感谢。