【数据结构】实践作业:加里森的任务
【数据结构】实践作业:加里森的任务有n个加里森敢死队的队员要炸掉敌人的一个军火库,谁都不想去,队长加里森决定用轮回数数的办法来决定哪个战士去执行任务。规则如下:如果前一个战士没完成任务,则要再派一个战士上去。现给每个战士编一个号,大家围坐成一圈,随便从某一个编号为x的战士开始计数,当数到y时,对应的战士就去执行任务,且此战士不再参加下一轮计数。如果此战士没完成任务,再从下一个战士开始数数,被数到第
零、效果展示
一、题目要求
有n个加里森敢死队的队员要炸掉敌人的一个军火库,谁都不想去,队长加里森决定用轮回数数的办法来决定哪个战士去执行任务。规则如下:如果前一个战士没完成任务,则要再派一个战士上去。现给每个战士编一个号,大家围坐成一圈,随便从某一个编号为x的战士开始计数,当数到y时,对应的战士就去执行任务,且此战士不再参加下一轮计数。如果此战士没完成任务,再从下一个战士开始数数,被数到第y时,此战士接着去执行任务。以此类推,直到任务完成为止。
加里森本人是不愿意去的,假设加里森为1号,请你设计一程序为加里森支招,求出n,x,y满足何种条件时,加里森能留到最后而不用去执行任务。
要求:
(1)主要数据结构采用链式结构存储。
(2)自拟实验实例验证程序正确性,并讨论n,x,y之间的关系。
二、需求分析
任务描述
设计一个程序解决加里森的任务问题。该问题要求根据给定的条件(n、x、y),找出使加里森不用去执行任务的组合。具体规则如下:
- 有 n 个加里森敢死队的队员要炸掉敌人的一个军火库。
- 每个队员都有一个编号,编号从 1 到 n。
- 以编号为 x 的战士为起始点,从1开始循环数数,每数到第 y 个战士,该战士将去执行任务。
- 如果某个战士执行任务失败,则会重新开始数数,直到任务完成为止。
- 加里森本人是不愿意去执行任务的,他的编号永远为1。
输入要求
- 输入形式:通过命令行或交互界面输入 n、x、y 的值。
- 输入值的范围:n、x、y 均为正整数,且满足 1 <= x <= n,1 <= y <= 2*n(自定义)。
输出要求
- 输出形式:输出符合条件的 x 和 y 的组合。
- 输出内容:程序将输出满足条件的 x 和 y 值,使得加里森不用去执行任务。
功能描述
- 程序能够根据输入的 n、x、y 值,计算出使加里森不用去执行任务的组合。
- 程序能够处理正确的输入,并输出符合条件的 x 和 y 值。
- 程序能够处理错误的输入,并给出相应的错误提示或处理方式。
测试数据
- 正确的输入数据:例如 n=5,y=2,期望输出为 x=6。
- 错误的输入数据:例如 n=0,期望输出为错误提示或处理方式。
二、 概要设计
思路概述
- 使用链表结构模拟加里森的任务问题,每个节点代表一个队员,节点之间通过指针连接成环形链表。
- 设计函数来初始化链表、删除节点、打印链表等操作,根据规则模拟任务执行过程,找出符合条件的 x 和 y 的组合。
数据结构定义
- 定义链表节点结构体
LinkNode
,包含数据data
和指向下一个节点的指针next
。
主程序流程
- 提供交互式界面,让用户输入 n、x、y 的值。
- 根据用户输入的值调用相应的函数进行计算和处理。
- 输出符合条件的 x 和 y 的组合,或者输出错误提示信息。
程序模块层次关系
- 主程序模块:负责处理用户输入、调用相应函数、输出结果。
- 初始化链表模块:负责创建空链表和添加节点。
- 模拟模块:根据规则进行模拟运行并删除节点。
- 输出结果模块:根据计算结果输出符合条件的 x 和 y 的组合,或者输出错误提示信息。
三、详细设计
主体思路
主函数 main
|
|--> function//面向用户的交互模块
|
|--> Garrison_x//基于(n,y)求解(x)
| |--> x++
| |--> CreateHeadNode + InitList//初始化n人链表
| |--> DeleteNode
| |--> PrintList
|
|--> Garrison_x_y//基于(n)求解(x,y)
| |--> x++
| |--> y++
| |--> CreateHeadNode + InitList//初始化n人链表
| |--> DeleteNode
| |--> PrintList
结构体定义
//定义节点结构体
typedef struct linknode {
int data;
struct linknode* next;
}LinkNode;
函数定义
LinkNode* CreateHeadNode(void);//创建头结点
void CreateNewNode(LinkNode* H,int data);//创建元素节点
void PrintList(LinkNode* H);//打印链表
void DeleteNode(LinkNode* H, int num);//删除节点
void InitList(LinkNode* H,int num);//初始化链表
void Garrison_x(int n, int y, int isPrt);//固定n,y求x
void Garrison_x_y(int n, int isPrt);//固定n求x,y
void function();//面向用户的交互模块
执行函数
void Garrison_x(int n, int y, int isPrt)//固定n,y求x
{
for (int x = 1; x <= n; x++)
{
//根据n值初始化链表
LinkNode* H = CreateHeadNode();
pre = H;
InitList(H, n);
if (isPrt) PrintList(H);//打印初始化后的链表
for (int j = 0; j < x-1; j++)//将pre指向上次执行者的前一个人
pre = pre->next;
while (H->data > 1)//当前链表剩余元素个数>1?继续进行删除操作:退出
{
DeleteNode(H, y);
if (isPrt) PrintList(H);
}
if (H->next->data == 1)//如果1号(加里森)活了下来
printf("\n*** x = %d ***\n\n", x);
else//似了
if (isPrt) printf("x = %d is wrong\n", x);
}
}
void Garrison_x_y(int n, int isPrt)//固定n求x,y
{
for (int x = 1; x <= n; x++)
{
for (int y = 1; y <= 2 * n; y++)//设置y的上限是2n,可根据需求进行修改
{
LinkNode* H = CreateHeadNode();
pre = H;
InitList(H, n);
if(isPrt) PrintList(H);
for (int j = 0; j < x-1; j++)
pre = pre->next;
while (H->data > 1)
{
DeleteNode(H, y);
if (isPrt) PrintList(H);
}
if (H->next->data == 1)
printf("\n*** x = %d y = %d***\n\n", x,y);
else
if (isPrt) printf("x = %d y = %d is wrong\n", x,y);
}
}
}
交互模块
void function()
{
int func = 0;
int isPrt = 0;
int n, x, y;
printf("\n请输入想要执行的功能('1'表示(n,y)->(x),'2'表示(n)->(x,y),'0'表示退出程序):" );
scanf_s("%d", &func);
switch (func) {
case 0:
exit(0);
case 1:
printf("\n请输入n=");
scanf_s("%d", &n);
printf("\n请输入y=");
scanf_s("%d", &y);
if (n <= 0 || y <= 0)
{
printf("输入不合法,请重新输入");
break;
}
printf("\n请选择打印模式('1'表示'全过程','0'表示'精简'):");
scanf_s("%d", &isPrt);
Garrison_x(n, y, isPrt);
break;
case 2:
printf("\n请输入n=");
scanf_s("%d", &n);
if (n <= 0)
{
printf("输入不合法,请重新输入");
break;
}
printf("\n请选择打印模式('1'表示'全过程','0'表示'精简'):");
scanf_s("%d", &isPrt);
Garrison_x_y(n, isPrt);
break;
default:
printf("\n输入错误!请检查输入");
break;
}
}
四、 调试分析报告
时间复杂度
对于(n,y)->x,当固定n与y时,对于每一个x,删除节点操作执行n-1次,每次删除会移动工作指针y次,时间复杂度为O(n^2*y)。
对于(n)->(x,y),当固定n,再固定x时,对于每一个y,删除节点操作执行n-1次,每次删除会移动工作指针y次,时间复杂度为O(n^4)。
空间复杂度
链表操作的空间复杂度为O(n)
发现
当(n)->(x,y)时,xy的对数与y的取值范围相同。换句话说,当n一定时,对于任意的y,都能找到唯一确定的x与之对应。
五、用户使用说明
- 运行程序后,程序将会显示如下提示信息:
请输入想要执行的功能('1'表示(n,y)->(x),'2'表示(n)->(x,y),'0'表示退出程序):
-
根据提示输入想要执行的功能对应的数字,例如输入 1表示执行 (n,y)->x 的功能。
-
根据程序提示依次输入相关参数:
- 对于功能
1
,需要依次输入 n 和 y 的值,确保输入的值大于 0。 - 对于功能
2
,需要输入 n 的值,确保输入的值大于 0。
- 对于功能
-
如果输入的值不合法(小于等于 0),程序将会提示重新输入。
-
程序会根据输入的参数执行相应的功能,并输出结果或过程。
-
如果需要继续测试或执行其他功能,可以重复上述操作进行。
-
如果需要退出程序,可以输入
0
并按回车键退出。
六、 测试结果
mode:(n,y)->(x) simply
input:n=-1,y=3
output:
输入不合法,请重新输入
mode:(n,y)->(x) simply
input:n=10,y=3
output:
*** x = 7 ***
mode:(n,y)->(x) all
input:n=3,y=1
output:
List: 1 2 3
List: 1 3
List: 1
*** x = 1 ***
List: 1 2 3
List: 1 2
List: 2
x = 2 is wrong
List: 1 2 3
List: 2 3
List: 3
x = 3 is wrong
mode:(n)->(x,y) simply
input:n=6
output:
*** x = 1 y = 3***
*** x = 1 y = 5***
*** x = 2 y = 1***
*** x = 3 y = 2***
*** x = 3 y = 4***
*** x = 3 y = 7***
*** x = 3 y = 9***
*** x = 4 y = 6***
*** x = 4 y = 11***
*** x = 5 y = 8***
*** x = 5 y = 12***
*** x = 6 y = 10***
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)