Java 并发编程(3): JAVA 并发编程基础

1. JAVA 并发编程基础

从启动一个线程到线程间不同的通信方式.

1.1 线程

线程是系统调度的最小单位, 拥有各自的计数器, 堆栈和局部变量等属性.

1.1.1.1 为什么需要多线程

  • 更多的CPU 核心.
    • 一个线程同一个时刻只能运行在一个CPU 核心上.
  • 更快的响应时间.
  • 更好的编程模型.

1.1.2 线程优先级

  • OS 采用时分的形式调度线程的运行.
    • OS 会分出一个个的时间片, 线程会被分配到若干个时间片.
    • 当线程的时间片用完了就发生线程调度, 并等待下次调度.
  • 线程的优先级决定了线程需要被分配的CPU 资源的多少.
    • 频繁阻塞(休眠或I/O)的线程设置较高的优先级.
    • 偏重计算(需要较多CPU 时间的) 设置较低优先级, 以防止CPU 被独占.
  • 但是, OS 和JVM 对线程优先级不做任何保证, 程序的正确性不能依赖于线程的优先级高低.

1.1.3 线程的状态

1.1.4 Daemon 线程

  • 支持型线程: 主要用于程序中后台调度及支持性工作.
  • 当一个JVM 中不存在非Daemon 线程的时候, JVM 将会退出.
  • 当JVM 退出时, Daemon 线程中的finally 块并不一定会被执行.
    • Daemon 线程不能依赖于finally 块来执行关闭或清理资源的逻辑.

1.2 启动和终止线程

1.2.1 构造和启动线程

  • 新构造的线程由其parent 线程来进行空间分配, 同时child 继承了parent 的优先级, 是否为Daemon, contextClassLoader 等属性.
  • 启动线程时, 最好设置线程名称, 这样在使用jstack 分析问题时, 能获得更多有用的信息.

1.2.2 中断.

  • 中断是线程的一个标示位属性, 它表示一个运行中的线程, 是否被其他线程进行了中断操作.
  • 线程通过isInterrupted() 来检查自身是否被中断, 同时调用Thread.interrupted() 进行中断复位.
  • 方法在抛出InterruptedException 之前, JVM 会将该线程的中断标示位清除, 然后再抛出异常.

1.2.3 过期的suspend(), resume(), stop().

  • 过期的主要原因:
    • 调用suspend() 后, 线程不会释放已经占有的资源(如锁), 而是占用着资源进入睡眠状态, 易引发死锁.
    • 调用stop() 在终结一个线程时不会爆炸线程的资源正常释放, 通常是没有给与线程完成资源释放的机会, 从而导致程序运行在不确定的状态下.
  • 使用新的等待/通知机制来代替它们.

1.2.4 安全地终止线程

  • 通过标识位或中断操作的方式, 能够使线程在终止时有机会去清理资源. 而不是武断地终止线程.
    • 中断是简便的线程间交互方式, 适合于取消或停止任务.
    • 同时, 利用一个boolean 变量来控制是否需要停止任务并终止进程.
// class Runner implements Runnable
public void run(){
    while (on && !Thread.currentThread().isInterrupted(){
        // do your job.
    }
}
public void cancel(){
    on = false;
}

// main method
thread.interrupt();
runner.cancel();

1.3 线程间通信

1.3.1 volatile/synchronized 关键字

1.3.2 等待/通知机制

  • 生产者/消费者模型, 在功能层面上解耦了How & What.
  • 循环检查预期的方案
    • 难以同时保证及时性和降低开销. 循环的sleep 时长很难把握.
  • 等待/通知机制依托于同步机制, 其目的是确保等待线程从wait()方法返回时能够感知到通知线程对变量做出的修改.
    • 使用wait()/notify()/notifyAll()时需要先对调用对象加锁.
    • wait(): 放弃锁并进入对象的WaitQueue中, 进入Waiting状态.
    • notify(): 将WaitThread 从WaitQueue 移到SynchronizedQueue中, 并将其状态变为Blocked状态. 在notifyThread 释放了锁后, WaitThread 再次获取到锁并从wait() 返回并继续执行.
  • 经典范式:
// >>>> 等待方
synchronized(对象){
    while(条件不满足){
        对象.wait();
    }
    对应的处理逻辑.
}

// >>>> 通知方
synchronized(对象){
    改变条件
    对象.notifyAll();
}

1.3.3 管道输入/输出流

  • 与普通的I/O流的不同: 以内存为传输媒介, 主要用于线程之间的数据传输.
    • 有面向字节/字符的PipedInput/OutputStream, PipedReader/Writer.
  • 必须先调用connect() 进行绑定, 才能进行访问.

1.3.4 Thread.join() 的使用

  • 每个join线程等待前驱线程终止后, 才从join()返回.
public final synchronized void join(){
    //条件不满足, 继续等待
    while(isAlive()){
        wait(0);
    }
    //条件符合, 方法返回.
}
  • 当线程终止时,会调用自身的notifyAll(), 来通知所有等待在该线程对象上的线程.

1.3.5 ThreadLocal 的使用

  • 线程变量: 以ThreadLocal 对象为键, 任意对象为值的存储结构.
    • 该结构被附带在线程上.

1.4 生产者和消费者模式

使用阻塞队列(容器)做第三方来解耦生产者和消费者, 两者通过阻塞队列进行通信.

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

推荐阅读更多精彩内容

  • 一、进程和线程 进程 进程就是一个执行中的程序实例,每个进程都有自己独立的一块内存空间,一个进程中可以有多个线程。...
    阿敏其人阅读 2,611评论 0 13
  • 前言 多线程并发编程是Java编程中重要的一块内容,也是面试重点覆盖区域,所以学好多线程并发编程对我们来说极其重要...
    嘟爷MD阅读 7,310评论 21 272
  • 第1章 并发编程的挑战 1.1 上下文切换 即便是单核CPU也支持多线程并发,CPU通过给每个线程分配时间片(几十...
    卑鄙的鹿尤菌阅读 4,758评论 1 22
  • 1 下公交车的时候 站台人很多 刚好后面有一辆洒水车开了过来 本以为这下完了 肯定会被洒一身水 突然间 洒水车的车...
    卅槑阅读 132评论 0 0
  • 今天妹妹心血来潮,要给我表演最好的考级节目,我赶紧录了下来,留住这珍贵的一刻。 星期八小镇玩了...
    擎天柱_6e9a阅读 123评论 0 0