gpt4 book ai didi

java - 我的代码线程不安全吗?

转载 作者:搜寻专家 更新时间:2023-10-30 21:10:09 25 4
gpt4 key购买 nike

我已经编写了代码来理解 CyclicBarrier。

我的应用程序模拟选举。每轮选出得票少的候选人,该候选人退出比赛以取得胜利。

来源:

class ElectoralCommission {
public volatile boolean hasWinner;
public volatile String winner;
private List<String> candidates;
private Map<String, Integer> results = new ConcurrentHashMap<>();

ElectoralCommission(List<String> candidates) {
this.candidates = candidates;
}

public void acceptVote(int index) {
Integer currentResult = results.get(candidates.get(index));
results.put(candidates.get(index), currentResult == null ? 1 : currentResult + 1);
}

public void clearResults() {
results.clear();
}

public synchronized List<String> getCandidates() {
return candidates;
}

public String getWinner() {
return winner;
}

public boolean isHasWinner() {
return hasWinner;
}

public void printResults() {
System.out.println("result:");
for (Map.Entry<String, Integer> entry : results.entrySet()) {
System.out.println("key=" + entry.getKey() + ", value=" + entry.getValue());
}
}

public void removeWeakCandidates() {
int minVoteValue = Collections.min(results.values());
int maxVoteValue = Collections.max(results.values());
if (maxVoteValue == minVoteValue) {
System.out.println("all candidates got same votes count");
} else {
List<String> loosers = results.entrySet().stream().filter(i -> i.getValue() == minVoteValue).map(i -> i.getKey()).collect(Collectors.toList());
candidates.removeAll(loosers);
if (candidates.size() == 1) {
hasWinner = true;
winner = candidates.get(0);
}
}
clearResults();
}
}

class Voter implements Runnable {
ElectoralCommission electoralCommission;
CyclicBarrier cyclicBarrier;

Voter(CyclicBarrier cyclicBarrier, ElectoralCommission electoralCommission) {
this.cyclicBarrier = cyclicBarrier;
this.electoralCommission = electoralCommission;
}

@Override
public void run() {
while (!electoralCommission.hasWinner) {
List<String> candidates = electoralCommission.getCandidates();
int voteIndex = new Random().nextInt(candidates.size());
electoralCommission.acceptVote(voteIndex);
try {
cyclicBarrier.await(); //wait while all voters vote
} catch (InterruptedException e) {
e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates.
} catch (BrokenBarrierException e) {
e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates.
}
}
}
}

class Main {
public static final int VOTE_COUNT = 10000;

public static void main(String[] args) {
List<String> candidates = new ArrayList<>();
candidates.add("candidate_1");
candidates.add("candidate_2");
candidates.add("candidate_3");
candidates.add("candidate_4");
candidates.add("candidate_5");
candidates.add("candidate_6");
candidates.add("candidate_7");
candidates.add("candidate_8");
candidates.add("candidate_9");
candidates.add("candidate_10");
ElectoralCommission electoralCommission = new ElectoralCommission(candidates);
CyclicBarrier cyclicBarrier = new CyclicBarrier(VOTE_COUNT, new Summarizer(electoralCommission));
for (int i = 0; i < VOTE_COUNT; i++) {
new Thread(new Voter(cyclicBarrier, electoralCommission)).start();
}

}
}

class Summarizer implements Runnable {
ElectoralCommission electoralCommission;

Summarizer(ElectoralCommission electoralCommission) {
this.electoralCommission = electoralCommission;
}

@Override
public void run() {
System.out.println("summarizer");
electoralCommission.printResults();
electoralCommission.removeWeakCandidates();
if (electoralCommission.hasWinner) {
String winnerName = electoralCommission.getWinner();
System.out.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
System.out.println("candidate " + winnerName + " won!");
System.out.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
}
}
}

示例输出:

summarizer
result:
key=candidate_2, value=984
key=candidate_3, value=1004
key=candidate_1, value=1018
key=candidate_6, value=1002
key=candidate_7, value=969
key=candidate_4, value=1031
key=candidate_5, value=1038
key=candidate_10, value=915
key=candidate_8, value=1003
key=candidate_9, value=1034
summarizer
result:
key=candidate_2, value=1096
key=candidate_3, value=1133
key=candidate_1, value=1088
key=candidate_6, value=1144
key=candidate_7, value=1078
key=candidate_4, value=1136
key=candidate_5, value=1100
key=candidate_8, value=1147
key=candidate_9, value=1078
summarizer
result:
key=candidate_2, value=1429
key=candidate_3, value=1488
key=candidate_1, value=1430
key=candidate_6, value=1445
key=candidate_4, value=1410
key=candidate_5, value=1397
key=candidate_8, value=1399
summarizer
result:
key=candidate_2, value=1668
key=candidate_3, value=1697
key=candidate_1, value=1684
key=candidate_6, value=1661
key=candidate_4, value=1679
key=candidate_8, value=1610
summarizer
result:
key=candidate_2, value=2079
key=candidate_3, value=1954
key=candidate_1, value=2003
key=candidate_6, value=2022
key=candidate_4, value=1941
summarizer
result:
key=candidate_2, value=2504
key=candidate_3, value=2481
key=candidate_1, value=2500
key=candidate_6, value=2514
summarizer
result:
key=candidate_2, value=3299
key=candidate_1, value=3284
key=candidate_6, value=3417
summarizer
result:
key=candidate_2, value=4961
key=candidate_6, value=5036
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
candidate candidate_6 won!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!

如您所见,我的最后一个汇总器调用输出如下:

summarizer
result:
key=candidate_2, value=4961
key=candidate_6, value=5036
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
candidate candidate_6 won!
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!

我有 10000 个选民,但 4961 + 5036 = 9997。

哪个丢了 3 票?

附言

我尝试了带锁的解决方案:

class ElectoralCommission {
public volatile boolean hasWinner;
public volatile String winner;
private List<String> candidates;
private Map<String, AtomicInteger> results = new ConcurrentHashMap<>();

ElectoralCommission(List<String> candidates) {
this.candidates = candidates;
}

public void acceptVote(int index) {
AtomicInteger currentResult = results.get(candidates.get(index));
if(currentResult!=null){
currentResult.incrementAndGet();
}
}

public void clearResults() {
results.clear();
}

public synchronized List<String> getCandidates() {
return candidates;
}

public String getWinner() {
return winner;
}

public boolean isHasWinner() {
return hasWinner;
}

public void printResults() {
System.out.println("result:");
for (Map.Entry<String, AtomicInteger> entry : results.entrySet()) {
System.out.println("key=" + entry.getKey() + ", value=" + entry.getValue());
}
}

public void removeWeakCandidates() {
int minVoteValue = Collections.min(results.values().stream().map(i->i.intValue()).collect(Collectors.toList()));
int maxVoteValue = Collections.max(results.values().stream().map(i->i.intValue()).collect(Collectors.toList()));
if (maxVoteValue == minVoteValue) {
System.out.println("all candidates got same votes count");
} else {
List<String> loosers = results.entrySet().stream().filter(i -> i.getValue().intValue() == minVoteValue).map(i -> i.getKey()).collect(Collectors.toList());
candidates.removeAll(loosers);
if (candidates.size() == 1) {
hasWinner = true;
winner = candidates.get(0);
}
}
clearResults();
}
}

class Voter implements Runnable {
ElectoralCommission electoralCommission;
CyclicBarrier cyclicBarrier;

Voter(CyclicBarrier cyclicBarrier, ElectoralCommission electoralCommission) {
this.cyclicBarrier = cyclicBarrier;
this.electoralCommission = electoralCommission;
}

@Override
public void run() {
while (!electoralCommission.hasWinner) {
List<String> candidates = electoralCommission.getCandidates();
int voteIndex = new Random().nextInt(candidates.size());
electoralCommission.acceptVote(voteIndex);
try {
cyclicBarrier.await(); //wait while all voters vote
} catch (InterruptedException e) {
e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates.
} catch (BrokenBarrierException e) {
e.printStackTrace(); //To change body of catch statement use File | Settings | File Templates.
}
}
}
}

class Main {
public static final int VOTE_COUNT = 10000;

public static void main(String[] args) {
List<String> candidates = new ArrayList<>();
candidates.add("candidate_1");
candidates.add("candidate_2");
candidates.add("candidate_3");
candidates.add("candidate_4");
candidates.add("candidate_5");
candidates.add("candidate_6");
candidates.add("candidate_7");
candidates.add("candidate_8");
candidates.add("candidate_9");
candidates.add("candidate_10");
ElectoralCommission electoralCommission = new ElectoralCommission(candidates);
CyclicBarrier cyclicBarrier = new CyclicBarrier(VOTE_COUNT, new Summarizer(electoralCommission));
for (int i = 0; i < VOTE_COUNT; i++) {
new Thread(new Voter(cyclicBarrier, electoralCommission)).start();
}

}
}

class Summarizer implements Runnable {
ElectoralCommission electoralCommission;

Summarizer(ElectoralCommission electoralCommission) {
this.electoralCommission = electoralCommission;
}

@Override
public void run() {
System.out.println("summarizer");
electoralCommission.printResults();
electoralCommission.removeWeakCandidates();
if (electoralCommission.hasWinner) {
String winnerName = electoralCommission.getWinner();
System.out.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
System.out.println("candidate " + winnerName + " won!");
System.out.println("!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!");
}
}
}

但我明白了

java.util.concurrent.BrokenBarrierException
at java.util.concurrent.CyclicBarrier.dowait(CyclicBarrier.java:250)
at java.util.concurrent.CyclicBarrier.await(CyclicBarrier.java:362)
at lection2.task3.Voter.run(Main.java:88)

第 88 行:cyclicBarrier.await();

最佳答案

acceptVote(int index) 方法似乎有错误

尝试使用方法ConcurrentHashMap.putIfAbsentAtomicInteger 重写它:

private final ConcurrentHashMap<String, AtomicInteger> results = new ConcurrentHashMap<>();

public void acceptVote(int index) {
AtomicInteger currentResult = results.get(candidates.get(index));
if (currentResult == null) {
currentResult = results.putIfAbsent(candidates.get(index), new AtomicInteger(1));
if (currentResult != null) {
currentResult.getAndIncrement();
}
} else {
currentResult.getAndIncrement();
}
}

像这样的……

关于java - 我的代码线程不安全吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31938613/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com