java多线程--老先生

java多线程

进程和线程的区别

进程是资源分配的最小单位,线程是CPU调度的最小单位所有与进程相关的资源,都被存在PCB中

进程是抢占处理机的调度单位,线程属于某个进程,共享其资源

线程只由堆栈寄存器/程序计数器/TCB组成

线程不能看做独立应用,进程可看作独立应用

进程有独立地址空间,相互不影响,线程只是进程不同执行路径线程没有独立地址空间,多进程程序比多线程程序更健壮

进程切换开销更大

java进程和线程的关系

java对操作系统的功能进行封装,包括进程和线程

运行一个程序会产生一个进程,进程包括至少一个线程每个进程对应一个JVM实例,多个线程共享JVM里的堆Java采用单线程编程模型,程序会自动创建主线程

主线程可以创建子线程,原则上后于子线程完毕执行

Thread中的start和run的区别

start调用的是start0方法---jvm的JVM_StartThread方法----创建新的thread 之后调用run()方法启动线程

所以调用start()方法会创建一个新的子线程并启动,run()方法只是Thread的一个普通方法的调用,所以run()只是调用了当前线程方法。

二者不存在可比性

Thread和Runnable关系

Thread实现了Runnable接口,使run支持多线程

因为类单一继承原则,推荐多使用Runnable接口

如何给run()方法传参

构造函数/成员变量/回调函数

如何实现处理线程的返回值

主线程等待法

 

使用Thread类的join()阻塞当前线程以等待子线程处理完毕

 

 
 

通过Callable接口实现:通过FutureTask Or线程池获取

 

 
 
 
 

 

没有显式等待,但会在is.Done没有返回true之前,一直等待,直到等了5秒,获取到了返回的value才会打印

 

先建立一个线程池,提交任务到线程池,返回Future类型对象,调用isDone()get()方法线程池可以提交多个任务并发执行

线程的状态

六个状态:

新建NEW:创建后尚未启动的线程的状态

运行Runnable:包含Running(可运行线程中,等待被CPU选中)Ready(线程池中,被选中就进入Running状态)

无限制等待Waiting:不会被分配CPU执行时间,需要显式被唤醒

(没有设置Timeout参数的Object.wait()方法///没有设置Timeout参数Thread.join()方法///LockSupport.park()方法)

限期等待Time Waiting:在一定时间后会由系统自动唤醒

(设置Timeout参数的Object.wait()方法///设置Timeout参数的Thread.join()方   法///LockSupport.parkNanos()方法///LockSupport.parkUntil()方法)

阻塞Blocked:等待获取排他锁

结束状态Terminated:已终止线程的状态,线程已经结束执行

sleep和wait区别

sleep是Thread类的方法

Thread.sleep只会让出CPU,不会导致锁行为的改变Object.wait不仅让出CPU,还会释放已经占有的同步资源锁

notify和notifyAll区别

锁池

假如线程A已经拥有某个对象(不是类)的锁,而其它线程B/C想要调用这个对象的某个synchronized方法(或者块),由于B/C线程在进入对象的synchronized方法之前必须先获得该对象锁   的拥有权,而恰巧该对象的锁目前正在被线程A所占用,此时B/C线程就会被阻塞,进入该对象的锁池去等待

等待池

假设线程A调用了某个对象的wait()方法,线程A就会释放该对象的锁,同时线程A就进入到该对象的等待池中,进入到等待池中的线程不回去竞争该对象的锁。

 

notifyAll会让全部处于等待池的线程全部进入锁池去竞争获取锁。(没得到锁的线程会留在锁池而不是回到等待池

notifyAll会随机挑选一个处于等待池的线程全部进入锁池去竞争获取锁。

yield

当调用Thread.yield()函数,会给线程调度器一个当前线程愿意让出CPU使用的暗示,但是线程调度    器可能会忽视这个暗示

yield不会对锁状态有影响

如何中断线程

stop()已经被抛弃

调用interrupt(),通知线程应该中断了

如果线程处于阻塞状态,那么线程将立即退出被阻塞状态,并抛出一个InterruptedException异常

如果线程处于正常活动状态,那么会将该线程额中断标志设置为true。被设置中断标志的线程将继续正常运行,不受影响

 

需要被调用线程配合中断

在正常运行任务时,经常检查本线程的中断标志位,如果被设置了中断标志就自行停止线程

如果线程处于正常活动状态,那么会将该线程额中断标志设置为true。被设置中断标志的线程将继续正常运行,不受影响

这里的逻辑是新建线程(状态为new)start线程(Timed waiting),没设置为中断就i++,之后打印i 的值和线程状态。休眠300毫秒中断

 

这个检查到中断,并且线程位于阻塞状态,则catch异常并中断。但继续打印发现线程还是RUNNABLE状态,所以这就是被设置中断标志的线程将继续正常运行,不受影响300毫秒继续打印,发现线程终止。

线程状态以及线程状态之间的转换

通过thread或继承Runnable接口new出线程即新建

新建的线程调用start方法进入Runnable状态,可运行状态

Runnable状态被os时间片选中进入Running状态,运行状态

Running状态调用yield方法可能会让出cpu进入Runnable状态,也可能是时间片用完了,也会进  入Runnable状态

Running状态如果等待用户输入或者调用sleep会进入阻塞状态,不会释放锁。

Runnable状态和Running状态如果调用synchronized方法,发现自己没权限,进入锁池等待Running状态调用wait会进入等待队列,如果被notifyAll或者notify唤醒,就进入锁池

锁池获得锁,就进入Runnable状态

Running状态执行完毕或者异常退出,就进入死亡状态TERMINATED

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值