0

0

如何正确实现用户循环选择唯一神祇对象并动态更新列表

心靈之曲

心靈之曲

发布时间:2026-01-14 19:00:02

|

920人浏览过

|

来源于php中文网

原创

如何正确实现用户循环选择唯一神祇对象并动态更新列表

本文详解如何用 while 循环替代嵌套 for 循环,安全、健壮地实现用户从动态变化的神祇列表中重复选取 6 个不重复 id 的对象,并实时移除已选项,避免索引越界与逻辑错判。

在 Java 游戏模拟器类项目中,常见的需求是让用户从一个初始加载的神祇(God)列表中手动挑选固定数量(如 6 个)且互不重复的对象,每选中一个后需立即将其从候选池中移除,防止重复选择。原始代码中使用 for (int i = 0; i <= 5; i++) 外层循环 + 内层遍历匹配的结构存在根本性缺陷:

  • 内层 else 分支逻辑错误:只要当前遍历位置的 id 不匹配,就立即触发“重输”提示——这导致即使目标 ID 存在于列表靠后位置,也会在检查第一个元素时误判为无效;
  • 边遍历边删除引发索引偏移:在 for (j = 0; j < list.size(); j++) 中调用 list.remove(j) 会导致后续元素前移,但循环变量 j 仍自增,造成跳过下一个元素或 IndexOutOfBoundsException;
  • 缺乏重复选择校验:未检查用户是否已选过该 ID(即是否已在 selectedGods 中存在),违背“不可重复选取”需求。

✅ 正确解法是放弃固定次数的 for 循环,改用条件驱动的 while 循环,以 selectedGods.size() < 6 为唯一终止条件,并将“查找+验证+移动”逻辑封装为独立方法,提升可读性与健壮性。

✅ 推荐实现方案

首先,添加一个通用查找工具方法(建议置于 Player 类中):

Cutout.Pro
Cutout.Pro

AI驱动的视觉设计平台

下载
/**
 * 在指定神祇列表中查找 id 匹配的 God 对象索引
 * @param godList 待搜索的 God 列表
 * @param targetId 目标 id 值
 * @return 匹配项的索引(>=0),未找到返回 -1
 */
private int findGodById(List<God> godList, int targetId) {
    for (int i = 0; i < godList.size(); i++) {
        if (godList.get(i).getId() == targetId) {
            return i;
        }
    }
    return -1;
}

然后,在 selectGodsForTeam() 方法中重构核心选择逻辑:

void selectGodsForTeam() {
    Scanner scanner = new Scanner(System.in);
    System.out.println("请依次输入您要选择的 6 位神祇的 ID(每次输入后按回车):");

    while (selectedGods.size() < 6) {
        System.out.print("您已选择 " + selectedGods.size() + "/6 位神祇。请输入下一位神祇的 ID > ");

        int chooseGodId;
        try {
            chooseGodId = scanner.nextInt();
        } catch (InputMismatchException e) {
            System.out.println("❌ 输入错误:请输入有效的整数 ID!");
            scanner.nextLine(); // 清空非法输入
            continue;
        }

        // 检查是否已选(防重复)
        if (findGodById(selectedGods, chooseGodId) >= 0) {
            System.out.println("⚠️  提示:ID " + chooseGodId + " 的神祇已被选中,请选择其他神祇。");
            continue;
        }

        // 检查是否在候选池中存在
        int indexInPool = findGodById(listOfAllGods, chooseGodId);
        if (indexInPool == -1) {
            System.out.println("❌ 错误:ID " + chooseGodId + " 的神祇不存在或已被选走,请重新输入。");
            continue;
        }

        // 安全添加并移除
        God selectedGod = listOfAllGods.get(indexInPool);
        selectedGods.add(selectedGod);
        listOfAllGods.remove(indexInPool); // 此时 indexInPool 是有效索引,无偏移风险
        System.out.println("✅ 成功选择:" + selectedGod.getName() + "(ID:" + chooseGodId + ")");
    }

    System.out.println("\n? 神祇队伍组建完成!共选择 " + selectedGods.size() + " 位神祇。");
}

? 关键设计说明

  • 单一层级查找:findGodById() 一次性遍历并返回索引,避免嵌套循环中“一不匹配就报错”的误判;
  • 双重校验机制:先查 selectedGods(防重复),再查 listOfAllGods(防无效 ID),语义清晰;
  • 安全移除:使用 remove(int index) 而非 remove(Object),确保移除的是当前匹配项,且因查找与删除分离,无并发修改风险;
  • 异常防护:捕获 InputMismatchException 处理非数字输入,提升用户体验;
  • 即时反馈:每次成功/失败均给出明确提示,符合交互式 CLI 应用最佳实践。

⚠️ 注意事项

  • 若 God 类未重写 equals() 和 hashCode(),请勿使用 list.contains(new God(...)) 进行重复判断,应始终基于 id 字段比对;
  • Scanner 实例建议在类级别复用或显式关闭(本例中因生命周期短可忽略,生产环境建议 try-with-resources);
  • 后续扩展时,可将 findGodById() 抽离为静态工具方法,或改用 Java 8+ 的 Stream API(如 list.stream().filter(g -> g.getId() == id).findFirst()),但需注意性能与可读性权衡。

此方案彻底规避了原始逻辑中的索引错乱、提前中断和重复漏洞,是处理“用户交互式动态集合筛选”场景的标准实践。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

107

2023.09.25

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1051

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

615

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

49

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

88

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

272

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

59

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.4万人学习

C# 教程
C# 教程

共94课时 | 11.4万人学习

Java 教程
Java 教程

共578课时 | 82.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号