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

1. 基本類(lèi)型的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
|
public class AtomicTest { /** * 常見(jiàn)的方法列表 * * @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時(shí)才會(huì)set * @see AtomicInteger#compareAndSet(int, int) 嘗試新增后對(duì)比,若增加成功則返回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ù)組類(lèi)型的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
public class AtomicIntegerArrayTest { /** * 常見(jiàn)的方法列表 * @see AtomicIntegerArray#addAndGet(int, int) 執(zhí)行加法,第一個(gè)參數(shù)為數(shù)組的下標(biāo),第二個(gè)參數(shù)為增加的數(shù)量,返回增加后的結(jié)果 * @see AtomicIntegerArray#compareAndSet(int, int, int) 對(duì)比修改,參數(shù)1:數(shù)組下標(biāo),參數(shù)2:原始值,參數(shù)3,修改目標(biāo)值,修改成功返回true否則false * @see AtomicIntegerArray#decrementAndGet(int) 參數(shù)為數(shù)組下標(biāo),將數(shù)組對(duì)應(yīng)數(shù)字減少1,返回減少后的數(shù)據(jù) * @see AtomicIntegerArray#incrementAndGet(int) 參數(shù)為數(shù)組下標(biāo),將數(shù)組對(duì)應(yīng)數(shù)字增加1,返回增加后的數(shù)據(jù) * * @see AtomicIntegerArray#getAndAdd(int, int) 和addAndGet類(lèi)似,區(qū)別是返回值是變化前的數(shù)據(jù) * @see AtomicIntegerArray#getAndDecrement(int) 和decrementAndGet類(lèi)似,區(qū)別是返回變化前的數(shù)據(jù) * @see AtomicIntegerArray#getAndIncrement(int) 和incrementAndGet類(lèi)似,區(qū)別是返回變化前的數(shù)據(jù) * @see AtomicIntegerArray#getAndSet(int, int) 將對(duì)應(yīng)下標(biāo)的數(shù)字設(shè)置為指定值,第二個(gè)參數(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. 引用類(lèi)型的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
|
public class AtomicReferenceTest { public static void main(String[] args) { People people1 = new People( "Bom" , 0 ); People people2 = new People( "Tom" , 10 ); //先初始化一個(gè)值,如果不初始化則默認(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.字段類(lèi)型的使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
|
public class AtomicIntegerFieldUpdaterTest { /** * 可以直接訪(fǎng)問(wèn)對(duì)應(yīng)的變量,進(jìn)行修改和處理 * 條件:要在可訪(fǎng)問(wèn)的區(qū)域內(nèi),如果是private或挎包訪(fǎng)問(wèn)default類(lèi)型以及非父親類(lèi)的protected均無(wú)法訪(fǎng)問(wèn)到 * 其次訪(fǎng)問(wèn)對(duì)象不能是static類(lèi)型的變量(因?yàn)樵谟?jì)算屬性的偏移量的時(shí)候無(wú)法計(jì)算),也不能是final類(lèi)型的變量(因?yàn)楦緹o(wú)法修改),必須是普通的成員變量 * <p> * 方法(說(shuō)明上和AtomicInteger幾乎一致,唯一的區(qū)別是第一個(gè)參數(shù)需要傳入對(duì)象的引用) * * @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() + " 對(duì)應(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 對(duì)應(yīng)的值做了修改!
Thread-2 120
Thread-8 120
Thread-4 133
Thread-1 133
Thread-9 142
Thread-4 142
Thread-4 對(duì)應(yīng)的值做了修改!
Thread-8 131
Thread-3 對(duì)應(yīng)的值做了修改!
Thread-3 120
Thread-7 120
Thread-7 131
Thread-5 對(duì)應(yīng)的值做了修改!
Thread-5 120
Thread-6 120
Thread-0 對(duì)應(yīng)的值做了修改!
Thread-6 131
Thread-7 對(duì)應(yīng)的值做了修改!
Thread-8 對(duì)應(yīng)的值做了修改!
Thread-4 120
Thread-9 131
Thread-1 對(duì)應(yīng)的值做了修改!
Thread-9 對(duì)應(yīng)的值做了修改!
Thread-8 120
Thread-7 120
Thread-6 對(duì)應(yīng)的值做了修改!
Thread-0 131
Thread-6 120
Thread-9 120
Thread-1 120
到此這篇關(guān)于Java多線(xiàn)程 原子性操作類(lèi)的使用的文章就介紹到這了,更多相關(guān)Java多線(xiàn)程 原子性操作類(lèi)內(nèi)容請(qǐng)搜索服務(wù)器之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持服務(wù)器之家!
原文鏈接:https://juejin.cn/post/7017409081920995364