并发编程之美读书笔记
第一章
1.1
-
什么是进程:进程是代码在数据集合上的依次运行活动,是系统进行资源分配和调度的基本单位.
-
什么是线程:线程是进程的一个执行路径,是cpu分配的基本单位,一个进程中可以有多个线程.
-
每个线程都有自己的程序计数器和栈区域,在cpu中通过程序计数器用来记录线程当前要执行的地址,栈区域是存放局部变量和调用栈帧的
-
堆和方法区是线程共享的,堆中存放new的对象实例,方法区存放JVM加载的类,常量及静态变量
1.2
- 线程的三种创建方式
- 1.实现Runnable的run方法
继承Runable类
重写run()方法
main方法
可以使用多个start() - 2.继承Thread类并重写run的方法
继承Thread类
重写run()方法
创建线程
start()启动线程
由于java不支持多继承多个方法需要多份代码 - 3.使用FutureTask方式
创建任务类继承Callable类
重写call()方法
创建异步任务FutureTask
start()启动线程
get()返回结果
此方法与上面两种方法不同的是可以获得返回结果
1.3
- 线程等待调用wait()方法,在调用前需要事先获取该对象的监视器锁
- 唤醒线程方法:
- 其他线程调用该共享的对象notify()或notifyAll()方法
- 其他线程调用该进程的interrupt()方法,抛出InterruptedException异常时返回
- 虚假唤醒:使用循环条件为唤醒条件来解决虚假唤醒
今天就看了前面一点,我了解到了什么是进程与线程,进程与线程关系,创建线程的三种方式Runnable,Thread以及FutureTask,还有1.3的开头一点调用wait()方法需要注意的东西.
当线程调用共享对象的wait()方法时,当前线程只会释放当前共享对象的锁,当前线程持有的其他共享对象的监视器锁并不会被释放
- wait(long timeout)
- timeout>0:多了一个超时参数,没有在指定timeout ms内被其他线程唤醒,还是会超时返回
- timeout=0:与wait()方法效果一样
- timeout<0:会抛出IllegalArgumentException(非法监视器状态异常)异常
- wait(long timeout,int nanos)
- 内部调用了一个wait(long timeout)函数,在nanos>0时才会执行相关操作比如让timeout递增1
- notify()函数:
- 会唤醒一个调用wait方法挂起的线程,具体哪个是随机的
- 被唤醒后不会马上返回并执行,必须先获取共享对象的监视器锁
否则会抛出IllegalArgumentException异常 - notifyAll():会唤醒所有该共享变量上的被挂起的线程
1.4
join方法,由Thread类直接提供无返回值的用来等待线程执行终止的方法
这里推荐使用CountDownLatch
1.5
sleep方法:调用此方法,线程会让出执行权,在这期间不参与CPU调度.
如果在sleep期间被中断会抛出InterruptedException异常
1.6
让出CPU执行权的yield方法,调用此方法只是让出剩余时间片,并没有被阻塞挂起,而是处于就绪状态,线程调度器下一次调度时就有可能调度到当前线程执行
1.7
线程中断是一种线程之间的协作模式.
- void interrupted()方法:中断线程,如果A没有调用一系列方法被挂起,B调用A的interrupt()方法设置中断标记为true并返回
但在此期间A实际没有被中断,只是设置了标记,如果调用了以上方法,会抛出InterruptedException异常 - boolean isInterrupted()方法,检测当前线程是否被中断,是则返回true
可以通过interrupt()让满足条件的线程立即被中断返回激活,可以节约时间
今天看了wait()方法的剩余部分,了解到了调用wait()方法后线程释放锁的状况,还有带参数的wait方法的一些作用,了解了notify和notifyAll的作用和区别.了解了join,sleep,yield方法的作用和不同之处,还有线程中断的一些相关知识 暂时先停一下先看mybatis