欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java多線程 原子性操作類的使用

 更新時間:2021年10月27日 15:56:54   作者:冬日毛毛雨  
這篇文章主要介紹了Java多線程 原子性操作類的使用,在java5以后,我們接觸到了線程原子性操作,也就是在修改時我們只需要保證它的那個瞬間是安全的即可,經(jīng)過相應(yīng)的包裝后可以再處理對象的并發(fā)修改,本文總結(jié)一下Atomic系列的類的使用方法,下面一起進(jìn)入文章了解詳細(xì)內(nèi)容

前言:

在java5以后,我們接觸到了線程原子性操作,也就是在修改時我們只需要保證它的那個瞬間是安全的即可,經(jīng)過相應(yīng)的包裝后可以再處理對象的并發(fā)修改,本文總結(jié)一下Atomic系列的類的使用方法,其中包含:

1. 基本類型的使用

public class AtomicTest {
    /**
     * 常見的方法列表
     *
     * @see AtomicInteger#get()             直接返回值
     * @see AtomicInteger#getAndAdd(int)    增加指定的數(shù)據(jù),返回變化前的數(shù)據(jù)
     * @see AtomicInteger#getAndDecrement() 減少1,返回減少前的數(shù)據(jù)
     * @see AtomicInteger#getAndIncrement() 增加1,返回增加前的數(shù)據(jù)
     * @see AtomicInteger#getAndSet(int)    設(shè)置指定的數(shù)據(jù),返回設(shè)置前的數(shù)據(jù)
     * @see AtomicInteger#addAndGet(int)    增加指定的數(shù)據(jù)后返回增加后的數(shù)據(jù)
     * @see AtomicInteger#decrementAndGet() 減少1,返回減少后的值
     * @see AtomicInteger#incrementAndGet() 增加1,返回增加后的值
     * @see AtomicInteger#lazySet(int)      僅僅當(dāng)get時才會set
     * @see AtomicInteger#compareAndSet(int, int) 嘗試新增后對比,若增加成功則返回true否則返回false
     **/

    public static void main(String[] args) {

        final AtomicTicket ticket = new AtomicTicket();
        for (int i = 0; i < 3; i++) {
            new Thread(new Runnable() {
                @Override
                public void run() {
                    while (ticket.getCount() > 0) {
                        System.out.println(Thread.currentThread().getName() + " count: " + ticket.decrement());
                    }
                }
            }).start();
        }
    }
}


class AtomicTicket {

    public AtomicInteger count = new AtomicInteger(100);

    public int decrement() {

        return count.getAndDecrement();
    }

    public int getCount() {
        return count.get();

    }
}

Thread-0 count: 100
Thread-2 count: 98
Thread-1 count: 99
Thread-2 count: 96
Thread-0 count: 97
Thread-2 count: 94
Thread-2 count: 92
Thread-1 count: 95
中間省略...
Thread-1 count: 12
Thread-2 count: 7
Thread-0 count: 9
Thread-2 count: 5
Thread-1 count: 6
Thread-2 count: 3
Thread-0 count: 4
Thread-2 count: 1
Thread-1 count: 2

2. 數(shù)組類型的使用

public class AtomicIntegerArrayTest {

    /**
     * 常見的方法列表
     * @see AtomicIntegerArray#addAndGet(int, int) 執(zhí)行加法,第一個參數(shù)為數(shù)組的下標(biāo),第二個參數(shù)為增加的數(shù)量,返回增加后的結(jié)果
     * @see AtomicIntegerArray#compareAndSet(int, int, int) 對比修改,參數(shù)1:數(shù)組下標(biāo),參數(shù)2:原始值,參數(shù)3,修改目標(biāo)值,修改成功返回true否則false
     * @see AtomicIntegerArray#decrementAndGet(int) 參數(shù)為數(shù)組下標(biāo),將數(shù)組對應(yīng)數(shù)字減少1,返回減少后的數(shù)據(jù)
     * @see AtomicIntegerArray#incrementAndGet(int) 參數(shù)為數(shù)組下標(biāo),將數(shù)組對應(yīng)數(shù)字增加1,返回增加后的數(shù)據(jù)
     *
     * @see AtomicIntegerArray#getAndAdd(int, int) 和addAndGet類似,區(qū)別是返回值是變化前的數(shù)據(jù)
     * @see AtomicIntegerArray#getAndDecrement(int) 和decrementAndGet類似,區(qū)別是返回變化前的數(shù)據(jù)
     * @see AtomicIntegerArray#getAndIncrement(int) 和incrementAndGet類似,區(qū)別是返回變化前的數(shù)據(jù)
     * @see AtomicIntegerArray#getAndSet(int, int) 將對應(yīng)下標(biāo)的數(shù)字設(shè)置為指定值,第二個參數(shù)為設(shè)置的值,返回是變化前的數(shù)據(jù)
     */
    private final static AtomicIntegerArray ATOMIC_INTEGER_ARRAY = new AtomicIntegerArray(new int[]{1,2,3,4,5,6,7,8,9,10});

    public static void main(String []args) throws InterruptedException {
        Thread []threads = new Thread[10];
        for(int i = 0 ; i < 10 ; i++) {
            final int index = i;
            threads[i] = new Thread() {
                public void run() {
                    int original =  ATOMIC_INTEGER_ARRAY.get(index);
                    int result = ATOMIC_INTEGER_ARRAY.addAndGet(index, index + 1);
                    System.out.println("currentThread:" + Thread.currentThread().getName() + " , 原始值為:" + original + ",增加后的結(jié)果為:" + result);
                }
            };
            threads[i].start();
        }
        for(Thread thread : threads) {
            thread.join();
        }
        System.out.println("=========================>\n執(zhí)行已經(jīng)完成,結(jié)果列表:");
        for(int i = 0 ; i < ATOMIC_INTEGER_ARRAY.length() ; i++) {
            System.out.println(ATOMIC_INTEGER_ARRAY.get(i));
        }
    }
}

currentThread:Thread-0 , 原始值為:1,增加后的結(jié)果為:2
currentThread:Thread-3 , 原始值為:4,增加后的結(jié)果為:8
currentThread:Thread-2 , 原始值為:3,增加后的結(jié)果為:6
currentThread:Thread-1 , 原始值為:2,增加后的結(jié)果為:4
currentThread:Thread-5 , 原始值為:6,增加后的結(jié)果為:12
currentThread:Thread-4 , 原始值為:5,增加后的結(jié)果為:10
currentThread:Thread-6 , 原始值為:7,增加后的結(jié)果為:14
currentThread:Thread-7 , 原始值為:8,增加后的結(jié)果為:16
currentThread:Thread-8 , 原始值為:9,增加后的結(jié)果為:18
currentThread:Thread-9 , 原始值為:10,增加后的結(jié)果為:20
=========================>

執(zhí)行已經(jīng)完成,結(jié)果列表:
2
4
6
8
10
12
14
16
18
20

3. 引用類型的使用

public class AtomicReferenceTest {

    public static void main(String[] args) {
        People people1 =new People("Bom", 0);
        People people2 =new People("Tom",10);

        //先初始化一個值,如果不初始化則默認(rèn)值為null
        AtomicReference<People> reference = new AtomicReference<>(people1);
        People people3 = reference.get();
        if (people3.equals(people1)) {
            System.out.println("people3:" + people3);
        } else {
            System.out.println("else:" + people3);
        }

        /**
         * 當(dāng)前值:拿當(dāng)前值和reference.get()獲取到的值去比較,如果相等則true并更新值為期望值
         * 期望值:如果返回true則更新為期望值,如果返回false則不更新值
         */
        boolean b = reference.compareAndSet(null, people2);
        System.out.println("myClass.main-"+b+"--"+reference.get());

        boolean b1 = reference.compareAndSet(people1, people2);
        System.out.println("myClass.main-"+b1+"--"+reference.get());


        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());

                People people = reference.get();
                people.setName("Tom"+Thread.currentThread().getName());
                people.setAge(people.getAge()+1);
                reference.getAndSet(people);
                System.out.println(Thread.currentThread().getName()+reference.get().toString());
            }
        }).start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());

                People people = reference.get();
                people.setName("Tom"+Thread.currentThread().getName());
                people.setAge(people.getAge()+4);
                reference.getAndSet(people);
                System.out.println(Thread.currentThread().getName()+reference.get().toString());
            }
        }).start();

    }

}

 class People {
    private String name;
    private int age;

    public People(String name, int age) {
        this.name = name;
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "People{" +
                "name='" + name + ''' +
                ", age=" + age +
                '}';
    }
}

4.字段類型的使用

public class AtomicIntegerFieldUpdaterTest {

    /**
     * 可以直接訪問對應(yīng)的變量,進(jìn)行修改和處理
     * 條件:要在可訪問的區(qū)域內(nèi),如果是private或挎包訪問default類型以及非父親類的protected均無法訪問到
     * 其次訪問對象不能是static類型的變量(因為在計算屬性的偏移量的時候無法計算),也不能是final類型的變量(因為根本無法修改),必須是普通的成員變量
     * <p>
     * 方法(說明上和AtomicInteger幾乎一致,唯一的區(qū)別是第一個參數(shù)需要傳入對象的引用)
     *
     * @see AtomicIntegerFieldUpdater#addAndGet(Object, int)
     * @see AtomicIntegerFieldUpdater#compareAndSet(Object, int, int)
     * @see AtomicIntegerFieldUpdater#decrementAndGet(Object)
     * @see AtomicIntegerFieldUpdater#incrementAndGet(Object)
     * @see AtomicIntegerFieldUpdater#getAndAdd(Object, int)
     * @see AtomicIntegerFieldUpdater#getAndDecrement(Object)
     * @see AtomicIntegerFieldUpdater#getAndIncrement(Object)
     * @see AtomicIntegerFieldUpdater#getAndSet(Object, int)
     */
    public final static AtomicIntegerFieldUpdater<A> ATOMIC_INTEGER_UPDATER = AtomicIntegerFieldUpdater.newUpdater(A.class, "intValue");

    public static void main(String[] args) {
        final A a = new A();
        for (int i = 0; i < 10; i++) {

            new Thread() {
                public void run() {
                    System.out.println(
                            Thread.currentThread().getName() + " " + ATOMIC_INTEGER_UPDATER.get(a));
                    ATOMIC_INTEGER_UPDATER.addAndGet(a, 11);
                    System.out.println(
                            Thread.currentThread().getName() + " " + ATOMIC_INTEGER_UPDATER.get(a));
                    if (ATOMIC_INTEGER_UPDATER.compareAndSet(a, ATOMIC_INTEGER_UPDATER.get(a), 120)) {
                        System.out.println(Thread.currentThread().getName() + " 對應(yīng)的值做了修改!");
                    }
                    System.out.println(
                            Thread.currentThread().getName() + " " + ATOMIC_INTEGER_UPDATER.get(a));
                }
            }.start();
        }
    }

    static class A {
        volatile int intValue = 100;
    }
}

Thread-0 100
Thread-2 100
Thread-1 100
Thread-2 122
Thread-3 111
Thread-5 120
Thread-0 111
Thread-5 142
Thread-3 131
Thread-2 對應(yīng)的值做了修改!
Thread-2 120
Thread-8 120
Thread-4 133
Thread-1 133
Thread-9 142
Thread-4 142
Thread-4 對應(yīng)的值做了修改!
Thread-8 131
Thread-3 對應(yīng)的值做了修改!
Thread-3 120
Thread-7 120
Thread-7 131
Thread-5 對應(yīng)的值做了修改!
Thread-5 120
Thread-6 120
Thread-0 對應(yīng)的值做了修改!
Thread-6 131
Thread-7 對應(yīng)的值做了修改!
Thread-8 對應(yīng)的值做了修改!
Thread-4 120
Thread-9 131
Thread-1 對應(yīng)的值做了修改!
Thread-9 對應(yīng)的值做了修改!
Thread-8 120
Thread-7 120
Thread-6 對應(yīng)的值做了修改!
Thread-0 131
Thread-6 120
Thread-9 120
Thread-1 120

到此這篇關(guān)于Java多線程 原子性操作類的使用的文章就介紹到這了,更多相關(guān)Java多線程 原子性操作類內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Activiti7通過代碼動態(tài)生成工作流實現(xiàn)詳解

    Activiti7通過代碼動態(tài)生成工作流實現(xiàn)詳解

    這篇文章主要為大家介紹了Activiti7通過代碼動態(tài)生成工作流實現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-11-11
  • 簡單了解springboot中的配置文件相關(guān)知識

    簡單了解springboot中的配置文件相關(guān)知識

    這篇文章主要介紹了簡單了解springboot中的配置文件相關(guān)知識,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-11-11
  • maven環(huán)境變量配置以及失敗原因解析

    maven環(huán)境變量配置以及失敗原因解析

    這篇文章主要為大家詳細(xì)介紹了maven環(huán)境變量配置教程,以及為大家解析了安裝失敗的原因,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-10-10
  • Groovy的規(guī)則腳本引擎實例解讀

    Groovy的規(guī)則腳本引擎實例解讀

    這篇文章主要介紹了Groovy的規(guī)則腳本引擎實例解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • Java項目打包發(fā)布到maven私倉常見的幾種方式

    Java項目打包發(fā)布到maven私倉常見的幾種方式

    這篇文章主要介紹了項目打包發(fā)布到maven私倉常見的幾種方式,幫助大家更好的理解和學(xué)習(xí)使用Java,感興趣的朋友可以了解下
    2021-03-03
  • Springboot打包部署項目的過程詳解

    Springboot打包部署項目的過程詳解

    這篇文章主要介紹了Springboot打包部署項目的過程,通過jar包方式打包和war方式打包,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2023-12-12
  • Java獲取上月份最后一天日期8位的示例代碼

    Java獲取上月份最后一天日期8位的示例代碼

    這篇文章主要介紹了Java獲取上月份最后一天日期8位,需要的朋友可以參考下
    2022-09-09
  • 最新評論