Java线程安全的计数器简单实现代码示例

2025-05-29 0 34

前几天工作中一段业务代码需要一个变量每天从1开始递增。为此自己简单的封装了一个线程安全计数器,可以让一个变量每天从1开始递增。当然了,如果项目在运行中发生重启,即便日期还是当天,还是会从1开始重新计数。所以把计数器的值存储在数据库中会更靠谱,不过这不影响这段代码的价值,现在贴出来,供有需要的人参考。

?

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

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169
package com.hikvision.cms.rvs.common.util;

import java.text.SimpleDateFormat;

import java.util.ArrayList;

import java.util.Date;

import java.util.List;

import java.util.concurrent.CountDownLatch;

import java.util.concurrent.atomic.AtomicInteger;

import java.util.concurrent.atomic.AtomicReferenceFieldUpdater;

/**

* Created by lihong10 on 2017/8/9.

* 一个循环计数器,每天从1开始计数,隔天重置为1。

* 可以创建一个该类的全局对象,然后每次使用时候调用其get方法即可,可以保证线程安全性

*/

public class CircularCounter {

private static final AtomicReferenceFieldUpdater<CircularCounter, AtomicInteger> valueUpdater =

AtomicReferenceFieldUpdater.newUpdater(CircularCounter.class, AtomicInteger.class, "value");

//保证内存可见性

private volatile String key;

//保证内存可见性

private volatile AtomicInteger value;

private static final String DATE_PATTERN = "yyyy-MM-dd";

public CircularCounter() {

/**

* 这里将key设置为getCurrentDateString() + "sssssssssss" 是为了测试addAndGet()方法中日期发生变化的情况

* 正常使用应该将key初始化为getCurrentDateString()

*/

this.key = getCurrentDateString() + "sssssssssss";

this.value = new AtomicInteger(0);

}

/**

* 获取计数器加1以后的值

*

* @return

*/

public Integer addAndGet() {

AtomicInteger oldValue = value;

AtomicInteger newInteger = new AtomicInteger(0);

int newVal = -1;

String newDateStr = getCurrentDateString();

//日期一致,计数器加1后返回

if (isDateEquals(newDateStr)) {

newVal = add(1);

return newVal;

}

//日期不一致,保证有一个线程重置技术器

reSet(oldValue, newInteger, newDateStr);

this.key = newDateStr;

//重置后加1返回

newVal = add(1);

return newVal;

}

/**

* 获取计数器的当前值

* @return

*/

public Integer get() {

return value.get();

}

/**

* 判断当前日期与老的日期(也即key成员变量记录的值)是否一致

*

* @return

*/

private boolean isDateEquals(String newDateStr) {

String oldDateStr = key;

if (!isBlank(oldDateStr) && oldDateStr.equals(newDateStr)) {

return true;

}

return false;

}

/**

* 如果日期发生变化,重置计数器,也即将key设置为当前日期,并将value重置为0,重置后才能接着累加,

*/

private void reSet(AtomicInteger oldValue, AtomicInteger newValue, String newDateStr) {

if(valueUpdater.compareAndSet(this, oldValue, newValue)) {

System.out.println("线程" + Thread.currentThread().getName() + "发现日期发生变化");

}

}

/**

* 获取当前日期字符串

*

* @return

*/

private String getCurrentDateString() {

Date date = new Date();

String newDateStr = new SimpleDateFormat(DATE_PATTERN).format(date);

return newDateStr;

}

/**

* 计数器的值加1。采用CAS保证线程安全性

*

* @param increment

*/

private int add(int increment) {

return value.addAndGet(increment);

}

public static boolean isBlank(CharSequence cs) {

int strLen;

if(cs != null && (strLen = cs.length()) != 0) {

for(int i = 0; i < strLen; ++i) {

if(!Character.isWhitespace(cs.charAt(i))) {

return false;

}

}

return true;

} else {

return true;

}

}

public static void test() {

CircularCounter c = new CircularCounter();

AtomicInteger count = new AtomicInteger(0);

List<Thread> li = new ArrayList<Thread>();

int size = 10;

CountDownLatch latch1 = new CountDownLatch(1);

CountDownLatch latch2 = new CountDownLatch(size);

for (int i = 0; i < size; i++) {

Thread t = new Thread(new CounterRunner(c, latch1, latch2, count), "thread-" + i);

li.add(t);

t.start();

}

System.out.println("start");

latch1.countDown();

try {

latch2.await();

} catch (InterruptedException e) {

e.printStackTrace();

}

System.out.println(count.get());

System.out.println(c.get());

if(count.get() == c.get()) {

System.out.println("该计数器是线程安全的!!!");

}

}

public static void main(String... args) {

for(int i = 0; i < 15; i++) {

test();

}

}

}

/**

* 测试使用的Runnable对象

*/

class CounterRunner implements Runnable {

private CircularCounter counter;

private CountDownLatch latch1;

private CountDownLatch latch2;

private AtomicInteger count;

public CounterRunner(CircularCounter counter, CountDownLatch latch1, CountDownLatch latch2, AtomicInteger count) {

this.latch1 = latch1;

this.latch2 = latch2;

this.counter = counter;

this.count = count;

}

@Override

public void run() {

try {

latch1.await();

System.out.println("****************");

for (int i = 0; i < 20; i++) {

counter.addAndGet();

count.addAndGet(1);

}

latch2.countDown();

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

总结

以上就是本文关于Java线程安全计数器简单实现代码示例的内容,希望对大家有所帮助,有什么问题可以随时留言,欢迎大家一起交流讨论。感谢朋友们对快网idc网站的支持!

原文链接:http://blog.csdn.net/nmgrd/article/details/77015206

收藏 (0) 打赏

感谢您的支持,我会继续努力的!

打开微信/支付宝扫一扫,即可进行扫码打赏哦,分享从这里开始,精彩与您同在
点赞 (0)

声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。

快网idc优惠网 建站教程 Java线程安全的计数器简单实现代码示例 https://www.kuaiidc.com/114568.html

相关文章

发表评论
暂无评论