- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
java解法
- 解题思路
- 这个问题本质上是一个变形的约瑟夫问题,要求从 1 到 100 个人中,每次按照 m 报数并淘汰,直到剩余人数少于 m。然后输出剩下的人的编号,按从小到大排序。
解题步骤
输入 m:
如果 m 不满足 1 < m < 100,直接返回 “ERROR!”。
初始化数据结构:
removed[100] 作为布尔数组,用于记录哪些人已被淘汰(true 表示已淘汰)。
count 记录当前报数到多少。
remaining 记录当前剩余人数,初始化为 100。
currentIndex 记录当前正在报数的人的索引,初始化为 0。
模拟报数淘汰过程:
遍历 removed 数组,跳过已淘汰的人,只对未淘汰的人进行报数。
每当 count == m 时,淘汰该人(设置 removed[currentIndex] = true),重置 count 并减少 remaining。
使用 currentIndex = (currentIndex + 1) % 100 来循环遍历整个数组。
筛选剩余人员并输出:
遍历 removed 数组,将未淘汰的人的编号加入 StringJoiner 并返回
import java.util.Scanner;
import java.util.StringJoiner;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
int m = sc.nextInt();
System.out.println(getResult(m));
}
public static String getResult(int m) {
if (m <= 1 || m >= 100) return "ERROR!";
boolean[] removed = new boolean[100];
int count = 0;
int remaining = 100;
int currentIndex = 0;
while (remaining >= m) {
if (!removed[currentIndex]) {
count++;
if (count == m) {
removed[currentIndex] = true;
count = 0;
remaining--;
}
}
currentIndex = (currentIndex + 1) % 100;
}
StringJoiner sj = new StringJoiner(",");
for (int i = 0; i < 100; i++) {
if (!removed[i]) {
sj.add(String.valueOf(i + 1));
}
}
return sj.toString();
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 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
C++解法
若 m 不满足 1 < m < 100,直接输出 “ERROR!” 并退出。
初始化 people 数组:
使用 vector 存储 1~100 的编号。
模拟报数淘汰过程:
设定 idx 作为当前要淘汰的索引,初始值 0。
循环淘汰:当 people 数量 不少于 m 时:
计算下一轮被淘汰者的索引位置:idx = (idx + m - 1) % people.size();
使用 erase() 移除该索引处的人。
排序并输出结果:
sort(people.begin(), people.end()); 保障剩下的编号按升序排列。
逐个打印剩下的编号,并使用 “,” 进行分隔
#include
#include
#include
using namespace std;
int main() {
int m;
cin >> m;
if (m <= 1 || m >= 100) {
cout << "ERROR!" << endl;
return 0;
}
vector<int> people(100);
for (int i = 0; i < 100; ++i) {
people[i] = i + 1;
}
int idx = 0;
while (people.size() >= m) {
idx = (idx + m - 1) % people.size();
people.erase(people.begin() + idx);
}
sort(people.begin(), people.end());
for (size_t i = 0; i < people.size(); ++i) {
cout << people[i];
if (i != people.size() - 1) cout << ",";
}
return 0;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 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
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
若 m 不在有效范围 1 < m < 100,返回 “ERROR!”。
初始化 people 数组:
使用 Array.from({ length: 100 }, (_, i) => i + 1) 创建 1~100 的编号数组。
模拟报数淘汰过程:
设定 index 作为当前要淘汰的索引,初始值 0。
循环淘汰:当 people 数量 不少于 m 时:
计算当前需要淘汰的索引 index = (index + m - 1) % people.length
使用 splice(index, 1) 移除该索引处的人。
返回剩余人员编号:
people.join() 将剩余编号以 , 分隔并返回
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
rl.on("line", (line) => {
const m = parseInt(line);
console.log(calculateSurvivors(m));
});
function calculateSurvivors(m) {
if (m <= 1 || m >= 100) {
return "ERROR!";
}
const people = Array.from({ length: 100 }, (_, i) => i + 1);
let index = 0;
while (people.length >= m) {
index = (index + m - 1) % people.length;
people.splice(index, 1);
}
return people.join();
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 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
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: