Java多线程与并发

1. volatitle

volatitle对共享变量进行同步。在写入volatitle变量值之后,CPU缓存中的内容会被写回主存,再读取volatitle变量值时,缓存值为失效状态,然后重新从主存读取已改变过的值。

2. synchronized 关键字

所有的Java对象都有一个与之关联的监视器对象,允许在该监视器上进行加锁和解锁

这里一定要理解清楚,我们加锁的是监视器对象,而不是代码code

  • synchronized静态方法
    监视器对象是所在Java类对应的Class
  • synchronized实例对象方法
    监视器对象是当前对象实例
  • synchronized代码块
    代码块声明中的对象

2.1 synchronized方法

2.1.1. synchronized关键字的继承性

synchronized关键字是不能继承的,父类的 synchronized方法在子类中并不是synchronized,子类需要显式地为某个方法加synchronized,以变成同步方法

2.1.2. synchronized实例对象方法

synchronized method(){}的监视器对象是这个实例对象,加锁的对象不是这个方法,而是实例对象

public class Foo
{
   /**
    * 加锁的监视器对象是class Foo 的  实例对象   Foo foo = new Foo()后的foo
    * 而不是这个方法mothodA
    */
     public synchronized void mothodA()
     {}
       
     public synchronized void mothodB()
     {}
}
  1. 有多个线程去访问 foo.mothodA() 时,同时只有一个线程能访问foomothodA()方法
  2. 一个类的实例对象有多个synchronized的方法时,只要一个线程访问了其中一个synchronized方法,其它线程就不能访问这个对象中的任何一个synchronized方法
  3. 不同实例对象间的synchronized方法调用时互不影响的
Foo foo1 = new Foo();
Foo foo2 = new Foo();

不同的线程分别访问foo1foo2中的synchronized mothodA()方法是互不影响的,因为它们的监视器对象分别是foo1foo2,同一个监视器对象才会阻塞同步

2.1.3 synchronized static 静态方法

synchronized static staticMethodA()的监视器对象是Foo.class类,所有访问这个类的静态同步方法的线程,都在同一个 Foo.class上加锁和解锁,所以对所有线程中的类实例对象的同步起作用。

class Foo 
{
   public synchronized static methodA();
   public synchronized static staticMethodA();
}

一个线程里调用了Foo.staticMethodA(),会对其它线程调用foo.methodA()造成同步

2.2 synchronized代码块

  1. 多个并发线程访问一个对象的synchronized(this)同步代码块时,同一时间只有一个线程执行
  2. 当一个线程访问一个对象的synchronized(this)同步代码块时,其它线程仍然可以访问这个对象的非synchronized(this)代码块,而对对象中其它所有的synchronized(this)同步代码块的访问都将被阻塞
class Foo 
{
    public void methodA()
    {
         /**
           * 加锁对象是 实例对象  
           * this 关键字代表实例对象本身
           */
          synchronized (this) {
          }
    }
}

public class Foo extends Thread
{
    private int val;
        //全局
    private static Object lock = new Object();

    public Foo(int v)
    {
        val = v;
    }

    @Override
    public void run()
    {
        printVal(val);
    }

    public void printVal(int v)
    {
        synchronized (lock)
        {
            while(true)
                System.out.println(v);
        }
    }
}

3. Object类的wait、notify、notifyAll

/**
 * 以以下代码顺序执行的方式就能确保异步执行的过程正确的获取到looper,
 * 当前线程里调用 new Worker(),如果looper还未创建调用线程就陷入wait状
 * 态,构造函数里启动另一线程,创建looper后会唤醒new Worker()的调用线
 * 程,这时new Worker()才执行完,接着执行下面的getLooper()就能正常获取
 * <pre>
 * mAlbumArtWorker = new Worker("album art worker");
 * mAlbumArtHandler = new AlbumArtHandler(mAlbumArtWorker.getLooper());
 * </pre>
 */
public class Worker implements Runnable
{
    private final Object mLock = new Object();
    private Looper mLooper;

    Worker(String name)
    {
        //调用线程里构造函数启动另一个线程
        Thread t = new Thread(null,this,name);
        t.setPriority(Thread.MIN_PRIORITY);
        t.start();
        synchronized (mLock)
        {
            //如果当前looper对象未创建
            while(mLooper == null)
            {
                try
                {
                    //调用构造函数的线程wait,并释放监视对象mLock持有的锁
                    mLock.wait();
                } catch (InterruptedException ex)
                {}
            }
        }
    }

    public Looper getLooper()
    {
        return mLooper;
    }

    @Override
    public void run()
    {
        //对检视对象mLock加锁
        synchronized (mLock)
        {
            Looper.prepare();
            mLooper = Looper.myLooper();

            //唤醒监视对象mLock上等待的所有线程,如果调用构造函数的线程在wait状态,将被唤醒
            mLock.notifyAll();
        }
        Looper.loop();
    }

    public void quit()
    {
        mLooper.quit();
    }
}

4.高级实用工具

4.1 java.util.concurrent.locks中的API

锁可在对象之间传递,因此使用更灵活

4.2 java.util.concurrent.Semaphore信号量

可以创建一个new Semaphore(0)零个许可的信号量作为一个阻塞点,另一个工作线程处理完一定的工作后release()释放一个许可出来,让前面阻塞的线程继续执行

4.3 java.util.concurrent.CountDownLatch倒数闸门

4.4 java.util.concurrent.CyclicBarrier循环屏障

4.5 java.util.concurrent.Exchanger<V>对象交换器

适用于两个线程需要交换数据的场景。

  1. 只能有2个线程交换数据
  2. exchange是交换点,另一线程未到达则本线程在此等待
  3. 各自线程exchange函数的返回值是另一线程exchange的参数值
public class FillAndEmpty
{
    Exchanger<DataBuffer> exchanger = new Exchanger<DataBuffer>();
    DataBuffer initialEmptyBuffer = ... a made-up type
    DataBuffer initialFullBuffer = ...

    class FillingLoop implements Runnable
    {
        public void run()
        {
            //生成一个empty 的空缓冲区
            DataBuffer currentBuffer = initialEmptyBuffer;
            try {
                while (currentBuffer != null) {
                    //向空缓冲区填充数据
                    addToBuffer(currentBuffer);
                    if (currentBuffer.isFull())
                    {
                        /**
                         * 1.到达交换点,如果另一个线程还没有到exchanger.exchange交换点,则在此阻塞等待
                         * 2.这里exchanger.exchange  返回的数据,就是另一线程到达exchanger.exchange(dataA)时
                         *   传递的dataA,并把自己exchanger.exchange(dataB)传递的dataB传递给另一线程的exchanger.exchange
                         *   作为返回值
                         */
                        currentBuffer = exchanger.exchange(currentBuffer);
                        
                        //另一线程到达exchanger.exchange后,彼此线程安全的交换数据exchanger.exchange()函数返回后继续执行
                    }
                }
            } catch (InterruptedException ex) { ...handle ...}
        }
    }

    class EmptyingLoop implements Runnable
    {
        public void run()
        {
            DataBuffer currentBuffer = initialFullBuffer;
            try {
                while (currentBuffer != null) {
                    takeFromBuffer(currentBuffer);
                    //缓冲区数据是空的
                    if (currentBuffer.isEmpty())
                    {
                        /**
                         * 1.进入交换点,如果填充线程未到达交换点,则再次阻塞,等待填充线程到达时继续执行
                         * 2.exchanger.exchange(currentBuffer) 把自己空的缓冲区作为参数传递给填充线程,填充
                         * 线程exchanger.exchange的返回值就是本线程(exchanger.exchange(dataA)传递的dataA,
                         * 本线程exchanger.exchange的返回值时另一线程exchanger.exchange(dataB)传递的参数dataB
                         */
                        currentBuffer = exchanger.exchange(currentBuffer);
                    }
                }
            } catch (InterruptedException ex) { ...handle ...}
        }
    }

    void start()
    {
        new Thread(new FillingLoop()).start();
        new Thread(new EmptyingLoop()).start();
    }
}

5. 处理 InterruptedException

异常中断——注意:不可中断的阻塞方法

https://www.ibm.com/developerworks/cn/java/j-jtp05236.html

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,605评论 18 399
  • 该文章转自:http://blog.csdn.net/evankaka/article/details/44153...
    加来依蓝阅读 7,340评论 3 87
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,954评论 1 18
  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,448评论 1 15
  • 当我和你说了晚安的时候,我依然未睡,究竟是为什莫,只是心里不舒服说不出来罢了。
    夜欲行阅读 107评论 0 1