主类模板
gcc从4.1.2版本开始提供了__sync_*系列的内置API,用于加减和逻辑运算,可以对1,2,4,8字节长度的数值或指针类型进行原子操作,为方便使用,笔者对这些API作了简单的封装。
1
template<typename T>
2
class atomic_base
3data:image/s3,"s3://crabby-images/f86b7/f86b7e502a0580d5e24db72fe38f81dda2bc052d" alt=""
data:image/s3,"s3://crabby-images/3ee79/3ee79ec5a9b7f3dd33bbbdc97980715db1aa9f00" alt=""
{
4
public:
5
typedef T int_type;
6
typedef atomic_base<T> self_type;
7data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
8
public:
9
atomic_base(int_type val = int_type())
10
: val_(val)
11data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
12
}
13data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
14
~atomic_base()
15data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
16
BOOST_STATIC_ASSERT(sizeof(int_type)==1||sizeof(int_type)==2||sizeof(int_type)==4||sizeof(int_type)==8);
17
}
18data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
19
atomic_base(const self_type& other)
20data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
21
int_type val = other;
22
__sync_lock_test_and_set(&val_, val);
23
}
24data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
25
self_type& operator = (const self_type& other)
26data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
27
if (this!=&other)
28data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
29
int_type val = other;
30
__sync_lock_test_and_set(&val_, val);
31
}
32
return *this;
33
}
34
35
operator int_type() const
36data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
37
return __sync_val_compare_and_swap(const_cast<volatile int_type*>(&val_),0,0);
38
}
39
40
self_type operator++(int)
41data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
42
return __sync_fetch_and_add(&val_,1);
43
}
44data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
45
self_type& operator++()
46data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
47
__sync_add_and_fetch(&val_,1);
48
return *this;
49
}
50data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
51
self_type operator--(int)
52data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
53
return __sync_fetch_and_sub(&val_,1);
54
}
55data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
56
self_type& operator--()
57data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
58
__sync_sub_and_fetch(&val_,1);
59
return *this;
60
}
61data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
62
self_type& operator+=(int_type val)
63data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
64
__sync_add_and_fetch(&val_,val);
65
return *this;
66
}
67data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
68
self_type& operator-=(int_type val)
69data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
70
__sync_sub_and_fetch(&val_,val);
71
return *this;
72
}
73data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
74
self_type& operator&=(int_type val)
75data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
76
__sync_and_and_fetch(&val_, val);
77
return *this;
78
}
79data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
80
self_type& operator|=(int_type val)
81data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
82
__sync_or_and_fetch(&val_,val);
83
return *this;
84
}
85
86
self_type& operator^=(int_type val)
87data:image/s3,"s3://crabby-images/db282/db282e9ea79ad6a7617774c9b676a45b33d46480" alt=""
{
88
__sync_xor_and_fetch(&val_,val);
89
return *this;
90
}
91data:image/s3,"s3://crabby-images/6c6b8/6c6b84e662455f8092d9c42e3a86036cd3a28be1" alt=""
92
private:
93
T val_;
94
};
上面析构函数中,使用了BOOST_STATIC_ASSERT宏来作编译期约束,确保类型T大小为1,2,4,8字节长度。由于提供了T到atomic_base的隐式转换和转型操作,因此没有必要再定义operato +,operator - ,opeator ^,operator |,operator &运算符成员或友元重载操作。
指针类型偏特化
由于主模板定义了operator+=,operator-=,operator&=,operator|=,operator^=运算符重载操作,但当T为指针类型时,指针的加减和逻辑运算没有意义,但对于指针加减一个数值是有意义的,因此前两者操作保留且重定义,但没有后三者操作,需要特化实现:
1
template<typename T>
2
class atomic_base<T*>
3data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
data:image/s3,"s3://crabby-images/c9e2b/c9e2bc817d66f0a3894ba04ea7703b8e0b7b6162" alt=""
{
4
public:
5
typedef T* pointer_type;
6
typedef atomic_base<T*> self_type;
7data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
8
public:
9
atomic_base(pointer_type val = NULL)
10
: val_(val)
11data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
12
}
13data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
14
atomic_base(const self_type& other)
15data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
16
pointer_type val = other;
17
__sync_lock_test_and_set(&val_, val);
18
}
19data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
20
~atomic_base()
21data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
22
BOOST_STATIC_ASSERT(sizeof(pointer_type)==1||sizeof(pointer_type)==2||sizeof(pointer_type)==4||sizeof(pointer_type)==8);
23
}
24data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
25
self_type& operator=(const self_type& other)
26data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
27
if (this!=&other)
28data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
29
pointer_type val = other;
30
__sync_lock_test_and_set(&val_, val);
31
}
32
return *this;
33
}
34data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
35
operator pointer_type() const
36data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
37
return __sync_val_compare_and_swap(const_cast<volatile pointer_type*>(&val_),0,0);
38
}
39data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
40
self_type operator++(int)
41data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
42
return __sync_fetch_and_add(&val_,1);
43
}
44data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
45
self_type& operator++()
46data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
47
__sync_add_and_fetch(&val_,1);
48
return *this;
49
}
50data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
51
self_type operator--(int)
52data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
53
return __sync_fetch_and_sub(&val_,1);
54
}
55data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
56
self_type& operator--()
57data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
58
__sync_sub_and_fetch(&val_,1);
59
return *this;
60
}
61data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
62
self_type& operator+=(std::ptrdiff_t d)
63data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
64
__sync_add_and_fetch(&val_,d);
65
return *this;
66
}
67data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
68
self_type& operator-=(std::ptrdiff_t d)
69data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
70
__sync_sub_and_fetch(&val_,d);
71
return *this;
72
}
73data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
74
private:
75
T* val_;
76
};
同理,上面析构函数中,使用了BOOST_STATIC_ASSERT宏来作编译期约束,确保类型T的指针大小为1,2,4,8字节长度。
bool类型完全特化
由于大小为1字节的类型包括bool布尔类型,它的取值只有true或false两种可能,不存在加减运算操作,也没有意义,但逻辑运算是有意义的,因此需要特化实现:
1
template<>
2
class atomic_base<bool>
3data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
data:image/s3,"s3://crabby-images/c9e2b/c9e2bc817d66f0a3894ba04ea7703b8e0b7b6162" alt=""
{
4
public:
5
typedef atomic_base<bool> self_type;
6
public:
7
atomic_base(bool val = false)
8
: val_(val)
9data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
10
}
11
12
atomic_base(const self_type& other)
13data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
14
__sync_lock_test_and_set(&val_, other.val_);
15
}
16
17
self_type& operator=(const self_type& other)
18data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
19
if (this!=&other)
20
__sync_lock_test_and_set(&val_, other.val_);
21
return *this;
22
}
23
24
operator bool() const
25data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
26
return __sync_val_compare_and_swap(const_cast<volatile bool*>(&val_),0,0);
27
}
28data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
29
self_type& operator&=(bool val)
30data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
31
__sync_and_and_fetch(&val_, val);
32
return *this;
33
}
34data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
35
self_type& operator|=(bool val)
36data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
37
__sync_or_and_fetch(&val_,val);
38
return *this;
39
}
40data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
41
self_type& operator^=(bool val)
42data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
43
__sync_xor_and_fetch(&val_,val);
44
return *this;
45
}
46data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
47
private:
48
bool val_;
49
};
整数类型别名
针对常见的基本数值类型,定义了其类型别名:
1
typedef atomic_base<bool> atomic_bool;
2
typedef atomic_base<signed char> atomic_char;
3
typedef atomic_base<unsigned char> atomic_uchar;
4
typedef atomic_base<short> atomic_short;
5
typedef atomic_base<unsigned short> atomic_ushort;
6
typedef atomic_base<int> atomic_int;
7
typedef atomic_base<unsigned int> atomic_uint;
8
typedef atomic_base<long> atomic_long;
9
typedef atomic_base<unsigned long> atomic_ulong;
10
typedef atomic_base<long long> atomic_llong;
11
typedef atomic_base<unsigned long long> atomic_ullong;
posted on 2012-06-08 00:19
春秋十二月 阅读(4558)
评论(1) 编辑 收藏 引用 所属分类:
C/C++