登录
首页 >  文章 >  java教程

Java实现投票系统教程详解

时间:2026-04-10 15:21:32 257浏览 收藏

本文深入浅出地讲解了用Java构建线程安全投票系统的核心要点:如何选用ConcurrentHashMap与newKeySet()保障高并发下的票数统计和防重投,为何必须用动态细粒度锁而非全局锁来确保“查重+计票”原子性,以及怎样通过流式排序实时生成准确有序的结果视图——这些看似简单的控制台实现,实则直击分布式系统中数据一致性、并发控制与查询时效性的本质问题,是理解更复杂Web投票架构(如Spring Boot+MyBatis)底层逻辑的坚实起点。

在Java里如何实现简单的投票系统_Java基础业务逻辑实现解析

投票系统核心数据结构怎么选

Map 存候选人名和票数最直接,但要注意键值重复和线程安全。如果只是单线程控制台程序,HashMap 足够;若涉及多线程模拟(比如多个用户并发投票),必须换成 ConcurrentHashMap,否则会出现计数丢失。

候选人名单建议额外用 ListSet 管理,避免仅靠 Map 的 key 集合做校验——因为 Map 可能还没初始化某个候选人,直接 get() 会返回 null,容易引发 NullPointerException

如何防止重复投票

光靠“输入姓名”无法识别同一人,必须引入唯一标识。最简方案是要求用户输入学号/工号(字符串),并用另一个 Set 记录已投票 ID:

private static Set<String> votedIds = ConcurrentHashMap.newKeySet();
// 投票前检查
if (votedIds.contains(userId)) {
    System.out.println("该用户已投过票");
    return;
}
votedIds.add(userId);

注意:ConcurrentHashMap.newKeySet() 是 Java 8+ 提供的线程安全集合;若用 HashSetsynchronized,代码更冗长且易漏锁。

  • 不推荐用 IP 或时间戳做去重——本地测试全是 127.0.0.1,时间戳精度不够
  • 控制台程序中,“用户ID”由人工输入,需在提示语里明确要求(如“请输入学号:”),否则逻辑再严也挡不住乱输

投票操作的原子性怎么保障

一次投票包含两个动作:检查是否已投 + 增加票数。这两步必须原子执行,否则并发时可能 A、B 同时通过检查,然后都给同一候选人 +1,实际只应 +1 次。

解决方式不是加全局锁(性能差),而是对具体候选人加细粒度锁:

private static final Map<String, Integer> votes = new ConcurrentHashMap<>();
private static final Map<String, Object> candidateLocks = new ConcurrentHashMap<>();

public static void vote(String candidateName, String userId) {
    if (!isValidCandidate(candidateName)) return;
    if (votedIds.contains(userId)) return;

    // 获取该候选人的专属锁对象
    Object lock = candidateLocks.computeIfAbsent(candidateName, k -> new Object());
    synchronized (lock) {
        // 再次确认(防止锁外已变更)
        if (votedIds.add(userId)) {
            votes.merge(candidateName, 1, Integer::sum);
        }
    }
}

这里用 computeIfAbsent 动态生成锁对象,避免为不存在的候选人预分配锁;merge 是线程安全的计数更新方式,比 put(k, get(k)+1) 更可靠。

结果查询为什么不能直接遍历 Map

调用 votes.entrySet() 遍历没问题,但若想按票数排序输出,别写 new ArrayList(votes.entrySet()).sort(...) —— 这会创建中间列表,且排序后仍是无序 Map,下次遍历又乱序。

正确做法是每次查询时生成有序视图:

List<Map.Entry<String, Integer>> sorted = votes.entrySet().stream()
    .sorted(Map.Entry.<String, Integer>comparingByValue().reversed())
    .collect(Collectors.toList());

注意点:

  • reversed() 必须显式调用,否则默认升序(得票少的排前面)
  • 不要把 sorted 结果缓存成字段——Map 内容随时可能变,缓存会导致结果过期
  • 如果候选人很多(>1000),流式排序比手动循环 + 数组快,但不用过度优化,业务系统里查票不是高频操作

真实场景中,没人会手敲一百行投票逻辑——但理解这四点,才能看懂 Spring Boot + MyBatis 版本里 DAO 层的 @Select 和 Service 层的 synchronized 块到底在防什么。

理论要掌握,实操不能落!以上关于《Java实现投票系统教程详解》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>