Concurrent Basic

Concurrent Basic

本文引自《Java多线程编程实战指南(设计模式篇)》

基础


无处不在的线程

进程(Process)代表运行中的程序。一个运行的Java程序就是一个进程。

从操作系统地角度来看,线程(Thread)是进程中可独立执行的子任务。一个进程可以包含多个线程。同一个进程中的线程共享该进程所申请到的资源,入内存空间和文件句柄。

从JVM的角度来看,线程是进程中的一个组件(Component),它可以看作执行Java代码的最小单位。Java程序中任何一段代码总是执行在某个确定的线程中。

JVM在启动的时候会创建一个main线程,该线程负责执行Java程序的入口方法(main方法)。

在多线程编程中,弄清楚一段代码具体由哪个(或者哪种)线程去负责执行的这点很重要,这关系到性能问题,线程安全问题等。

Java中的线程可以分为守护线程(Deamon Thread)和用户线程(User Thread)。用户线程会阻止JVM的正常停止,即JVM正常停止前应用程序中的所有用户线程必须先停止完毕。否则JVM无法停止。而守护线程则不会影响JVM的正常停止,即应用程序中所有守护线程在运行也不影响JVM的正常停止。因此守护线程用来做一些重要性不是很高的任务,例如监视其他线程的运行情况。


线程的创建和运行


线程的状态和上下文切换

Thread.State :

  • NEW - 新建状态。
  • RUNNABLE - 该状态是一个复合状态。包括两个子状态:READYRUNNING
  • BLOCKED - 一个线程发起一个阻塞式I/O(Blocking I/O)操作后,或者试图去获得一个由其他线程持有的锁时,相应的线程会处于该状态。处于该状态的线程不会占用CPU资源。当相应的I/O操作完成后,或者相应的锁被其他线程释放后,该线程的状态又可以转换为RUNNABLE
  • WAITING - 一个线程执行了某些方法之后就会处于这种无限等待其他线程执行特定操作的状态。这些方法包括Object.wait(),Thread.join()和LockSupport.park()。能够使相应的线程从WAITING状态转换为RUNNABLE状态的相应方法包括:Object.notify(),Object.notifyAll()和LockSupport.unpark(thread);
  • TIMED_WAITING - 该状态可WAITING状态类似,差别在于处于该状态的线程并非无限等待其他线程执行特定操作,而是处于带有时间限制的等待状态。当其他线程没有在指定时间内执行该线程所期望的特定操作时,该线程的状态自动转换为RUNNABLE状态。
  • TERMINATED - 已经执行结束的线程处于该状态。由于一个线程实例只能被启动一次,因此一个线程也只可能有一次处于该状态。Thread实例的run方法正常返回或者由于抛出异常而提前终止都会导致相应的线程处于该状态。

从上述描述可知,一个线程在其整个生命周期中,只可能一次处于NEW的状态和TERMINATED状态。而一个线程的状态从RUNNABLE状态转换为BLOCKED、WAITING和TIMED_WAITING这几个状态中的任何一个状态都意味着上下文切换(Context Switch)的产生。

上下文切换 : 对线程上下文信息进行保存和恢复的过程就称之为上下文切换


线程的监视

  • jvisualvm
  • jstack
  • Java Mission Control

原子性、内存可见性、指令重排序、synchronized、volatile

  • 原子操作 - Atomic,相应的操作时单一不可分割的操作。在多线程环境中,非原子操作可能会收到其他线程的干扰。

  • synchronized - 可以实现操作的原子性。其本质时通过该关键字所包括的临界区(Critical Section)的排他性保证在任何一个时刻只有一个线程能够执行临界区内的代码,这使得临界区中的代码代表了一个原子操作。synchronized关键字代表的另外一个作用是内存的可见性(Memory Visibility),即保证了一个线程执行临界区代码时所修改的变量值对于稍后执行临界区中的代码的线程来说是可见的。

  • 临界区 - Critical Section,示例代码如下:

synchronized(syncObject) { 
    //critical section
}
  • 内存可见性 - Memory Visibility,CPU在执行代码的时候,为了减少变量访问的时间消耗可能将代码中访问的变量的值缓存道该CPU的缓存区(如L1 CacheL2 Cache等)中。因此,相应代码再次访问某个变量时,相应的值可能时从CPU缓存区而不是主内存中读取的。同样的,代码对这些被缓存锅的变量的值的修改也可能仅是被写入CPU缓存区,而没有被写回主内存。由于每个CPU都有自己的缓存区,因此一个CPU混村去中的内容对于其他CPU而言是不可见的。这就导致了其他CPU上运行的其他线程可能无法“看到”该线程对某个变量值所做的更改。这就是所谓的内存可见性。

  • volatile - 保证内存可见性,但是不保证操作的原子性。其另一个作用是禁止了指令重排序

  • 指令重排序 - 编译器和CPU为了提高指令的执行效率,可能会进行指令重排序操作。

synchronizedvolatile的区别:synchronized既能保证操作的原子性,又能保证内存的可见性。而volatile仅能保证内存可见性。但是前者会导致上下文切换,而后者不会。


线程的优势和风险

优势 :

  • 提高系统的吞吐率(Throughput
  • 提高响应性(Responsiveness
  • 充分利用多核(Muticore)CPU资源
  • 最小化对系统资源的占用
  • 简化程序的结构

风险 :

  • 线程安全问题
  • 线程的生命特征问题
  • 上下文切换
  • 可靠性

常用术语

术语 释义
任务 Task 把线程比作公司员工的,那么任务就可以被看作员工所要完成的工作内容。任务与线程并非一一对应的,通常一个线程可以用来执行多个任务。任务是一个相对的概念。一个文件可以被看作是一个任务,一个文件中的多个记录可以被看作一个任务,多个文件也可以看作一个任务
并发 Concurrent 表示多个任务同一时间段内被执行,这些任务并不是顺序执行的,而往往是以交替的方式被执行
并行 Parallel 表示多个任务在同一时刻被执行
客户端线程 Client Thread 从面向对象编程的角度来看,一个类总是对外提供某些服务(这也是这个类存在的意义)。其它类是通过调用该类的响应方法来使用这些服务的。因此,一个类的方法的调用方代码就被称为该类的客户端代码。相应地,执行客户端代码的线程就被称为客户端线程。因此,客户端线程也是一个相对的概念,某个类的客户端线程随着执行该方法调用方代码的线程的变化而变化
工作者线程 Worker Thread 工作者线程是相对于客户端线程而言的。它表示客户端线程之外的用于特定用途的其他线程
上下文切换 Context Switch 对线程的上下文信息进行保存和恢复的过程就称为上下文切换
显示锁 Explicit Lock 在Java代码中可以使用和控制的锁,即不是编译器和CPU内部使用的锁。包括synchronizedjava.util.concurrent.locks.Lock接口的所有实现类
线程安全 Thread Safe 一段操纵共享数据的代码能够保证在同一时间内被多个线程执行而保持其正确性的,就被称为是线程安全的

———————————————————— 结束 ————————————————————

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,402评论 6 499
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,377评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,483评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,165评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,176评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,146评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,032评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,896评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,311评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,536评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,696评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,413评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,008评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,815评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,698评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,592评论 2 353

推荐阅读更多精彩内容