并发编程线程基础(一)
1.1 什么是线程
这种老生常谈的问题在这里简单说一下啊,线程是操作系统调度的最小单元,也叫作轻量级进程。 在一个进程中可以创建多个线程
,这些线程都应有各自的计数器、堆栈和局部变量等属性,并且都能够访问共享的内存变量。
1.2 线程的状态
1. New ,新创建状态。线程被创建,还没有调用start方法,在线程运行之前还有一些基础工作要做。
2. Runnable,可运行状态。一旦调用start方法,线程线程就处于Runnable状态。一个可运行的线程可能正在运行也节能没有运行,这取决于操作系统给线程提供运行的时间。
3. Blocked,阻塞状态。表示线程被锁阻塞,他暂时不活动。
4. Waiting,等待状态。线程暂时不活动,并且不运行任何代码,这消耗最少的资源,直到线程调度器重新激活它。
5. Time waiting,超市等待状态。和等待状态不同的是,它是可以在指定的时间自行返回的。
6. Terminated,终止状态。表示当前线程已经执行完毕。导致线程终止有两种情况:第一种就是run方法执行完毕正常退出;第二种就是因为一个没有捕获的异常而终止了run方法,导致线程进入终止状态。
1.3 线程的创建与运行
线程创建有三种方法
1. 继承Thread类,重写run方法
2. 实现Runnable接口,并实现该接口的run方法
3. 实现Callable接口,重写call方法
1.4 线程的通知与等待
1.4.1 wait()函数
当一个线程调用一个共享变量的wait()方法时,该调用线程被阻塞挂起直到发生下面几件事情之一才返回:(1)其他线程调用了该共享变量的notify()或者notifyAll()方法;
(2)其他线程调用了该线程的interrupt()方法,该线程抛出InterruptException异常返回还有一个需要注意的是如果调用wait()方法的线程没有
事先获取该对象的监视器锁,则调用wait()方法时调用线程会抛出IllegalMonitorStateException异常。
1.4.2 wait(long timeout)函数
如果一个线程调用共享对象的该方法挂起后,没有在指定timeout ms 时间内被其他线程调用该共享变量的notify(),或者notifyAll()方法唤醒,那么该函数还是会因为超时而返回。
1.4.3 wait(long timeout,int nanos)函数
在其内部调用的是wait(long timeout )函数。只有nanos>0时才使参数timeout 递增1.
1.4.5 notify()函数
一个线程调用共享对象的notify()方法后,会唤醒一个在该共享变量上调用wait系列方法后被挂起的线程。一个共享变量上可能会有多个线程在等待,具体唤醒哪个等待的线程是随机的。
被唤醒的线程不能马上从wait方法返回并继续执行,必须在获取了共享对象的监视器锁后才可以返回。
1.4.6 notifyAll()函数
不同于咋共享变量上调用notify()函数会唤醒被阻塞到该共享变量上的一个线程,notifyAll()方法则会唤醒所有在该共享变量上由于调用wait系列方法而被挂起的线程。
1.4 等待线程执行终止的join方法
join方法是Thread类提供的,可以等待多个线程全部执行完毕后再做处理。
1.5 让线程睡眠的sleep方法
Thread类中有一个静态的sleep方法,当一个执行中的线程调用了Thread的sleep方法后,调用线程会暂时让出指定时间 的执行权,
也就是在这期间不参与CPU的调度,但是该线程所拥有的监视器资源,比如锁还是持有不让出的。
指定的睡眠时间到了后该函数会正常返回,线程就处于就绪状态,然后参与ZPU的调度获取到CPU的资源后就可以继续运行了。
如果在睡眠期间其他线程调用了该线程的interropt方法中断了该线程则该线程会在调用sleep方法的地方抛出InterruptedException异常而返回。
1.6 让出CPU执行权的yield方法
当一个线程调用yield方法时,当前线程会让出CPU使用权,然后处于就绪状态,线程调度器会从就绪队列里面获取一个线程优先级最高的线程,
当然也有可能会调度到刚刚让出CPU的那个线程来获取CPU执行权。
1.7 线程终断
Java中的线程中断是一种线程间的协作模式,通过设置线程的中断标志并不能直接终止该线程的执行,而是被中断的线程根据中断状态自行处理。
void interrupt()方法:中断线程。例如当线程A运行时,线程B可以调用A的interrupt方法来设置线程A的中断标志为true并立即返回。
设置标志仅仅是设置标志,线程A实际并没有被中断,它会继续往下执行。如果线程A因为调用了wait系列函数 join方法或者sleep方法而被阻塞挂起这时候B线程调用线程A的interrupt方法,线程A会在调用这些方法的地方抛出InterruptedException异常而返回。
void isInterrupted()方法
检测当前线程是否被中断,如果是返回true,否则返回false。
void interrupted()方法
检测当前线程是否被中断,如果是返回true 否则返回false。与isInterrupted不同的是该方法如果发现当前线程被中断,则会清楚中断标志,并且该方法是static方法,可通过Thread类直接调用。
1.8 理解线程上下文切换
多线程编程中线程个数一般都大于CPU个数,而每个CPU同一时刻只能被一个线程使用,为了让用户感觉多个线程是在同时执行的,
CPU资源的分配采用了时间片轮转的策略,也就是给每个线程分配一个时间片,线程在时间片内占用CPU执行任务。当前线程使用完
时间片后,就会处于就绪状态并让出CPU让其他线程占用,这就是上下文切换,从当前线程的上下文切换到了其他线程。
线程上下文切换时机有 : 当前线程的CPU时间片使用完处于就绪状态时,当前线程被其他线程中断时。