多线程上篇(解析需勇气,打脸不在意)

了解线程的前世今生:

现在的操作系统都是多任务操作系统,多线程是实现多任务的一种方式。

进程和线程是一种什么关系:

进程是指内存中运行的应用程序,每个进程都有自己独立的一块内存空间,而一个进程中可以启动多个线程。线程是指进程中的一个执行流程单元。进程中的多个线程共享进程的内存,因此线程间通信很容易,速度快。

每个正在系统上运行的程序都是一个进程。每个进程包含一到多个线程。进程也可能是整个程序或者是部分程序的动态执行。线程是一组指令的集合,或者是程序的特殊段,它可以在程序里独立执行。

线程是程序中一个单一的顺序控制流程.在单个程序中同时运行多个线程完成不同的工作,称为多线程.

(个人理解:进程相当于软件,线程相当于软件的一个功能,每个软件都有自己的内存空间,软件可以启动多个功能,软件中的多个功能共享软件的内存。)

线程调度原理:依靠线程栈模型实现。

线程栈是指某时刻时内存中线程调度的栈信息,当前调用的方法总是位于栈顶。线程栈的内容是随着程序的运行动态变化的,因此研究线程栈必须选择一个运行的时刻(实际上指代码运行到什么地方),一旦创建一个新的线程,就产生一个新的调用栈。

线程的分类:用户线程和守护线程(系统线程)。

守护线程:守护进程(Daemon)是运行在后台的一种特殊进程。它的作用是为其他线程的运行提供便利服务,它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。也就是说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。因此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要任何非守护线程还在运行,程序就不会终止。一个普通的用户线程调用Thread对象的setDaemon方法就变成了守护线程。(典型,垃圾回收线程)

(个人理解:守护线程作用是为其他的用户线程进行服务,当用户线程结束时,程序也会终止,守护线程也会死去,反之。但守护线程死去,用户线程不会死去。)

实现线程的方式:继承Thread类和实现Runnable接口
线程的状态:创建、就绪、运行、阻塞、死亡

线程状态间的切换:

线程创建之后,不会立即进入就绪状态,因为线程的运行需要一些条件(比如内存资源,在前面的JVM内存区域划分一篇博文中知道程序计数器、Java栈、本地方法栈都是线程私有的,所以需要为线程分配一定的内存空间),只有线程运行需要的所有条件满足了,才进入就绪状态。当线程进入就绪状态后,不代表立刻就能获取CPU执行时间,也许此时CPU正在执行其他的事情,因此它要等待。当得到CPU执行时间之后,线程便真正进入运行状态。线程在运行状态过程中,可能有多个原因导致当前线程不继续运行下去,比如用户主动让线程睡眠(睡眠一定的时间之后再重新执行)、用户主动让线程等待,或者被同步块给阻塞,此时就对应着多个状态:1.time waiting(睡眠或等待一定的事件),2.waiting(等待被唤醒),3.blocked(阻塞),Sleep方法和wait方法以及yield方法join方法。

sleep()方法和对象的wait()方法都可以让线程暂停执行,它们有什么区别?
sleep()方法(休眠)是线程类(Thread)的静态方法,调用此方法会让当前线程暂停执行指定的时间,将执行机会(CPU)让给其他线程,但是对象的锁依然保持,因此休眠时间结束后会自动恢复(线程回到就绪状态)。

(个人理解:打比喻跟排队一样。sleep:比如<线程1>排在第三,现在让出来给别人排队并给出指定的时间,又可以重新进入        <线程1>回到就绪状态)
wait()是Object类的方法,调用对象的wait()方法导致当前线程放弃对象的锁(线程暂停执行),进入对象的等待池(wait pool),只有调用对象的notify()方法(或notifyAll()方法)时才能唤醒等待池中的线程进入等锁池(lock pool),如果线程重新获得对象的锁就可以进入就绪状态。线程的sleep()方法和yield()方法有什么区别?
sleep()方法给其他线程运行机会时不考虑线程的优先级,因此会给低优先级的线程运行的机会;yield()方法只会给相同优先级或更高优先级的线程以运行的机会;线程执行sleep()方法后转入阻塞(blocked)状态,而执行yield()方法后转入就绪(ready)状态;
 sleep()方法声明抛出InterruptedException,而yield()方法没有声明任何异常;
 sleep()方法比yield()方法(跟操作系统CPU调度相关)具有更好的可移植性。
线程同步以及线程调度相关的方法. wait():使一个线程处于等待(阻塞)状态,并且释放所持有的对象的锁;
sleep():使一个正在运行的线程处于睡眠状态,是一个静态方法,调用此方法要处理InterruptedException异常;
notify():唤醒一个处于等待状态的线程,当然在调用此方法的时候,并不能确切的唤醒某一个等待状态的线程,而是由JVM确定唤醒哪个线程,而且与优先级无关;
notityAll():唤醒所有处于等待状态的线程,该方法并不是将对象的锁给所有线程,而是让它们竞争,只有获得锁的线程才能进入就绪状态;

 

 

    

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值