C语言:9.6 3个人围成一圈, 从第1个人开始顺序报号1, 2, 3。凡报到3者退出圈子。 找出最后留在圈子中的人原来的序号。要求用链表实现。

/*
 9.6 3个人围成一圈, 从第1个人开始顺序报号1, 2, 3。凡报到3者退出圈子。
找出最后留在圈子中的人原来的序号。要求用链表实现。
***解题思路:
创建一个环形链表,给链表中的每一个节点从1~13编号,
然后开始淘汰过程,对于淘汰的节点,序号置为0,淘汰完成之后,
找到序号不为0的即为最后留下的。*/
#include <stdio.h>
#define NUM 13
typedef struct people
{
	int num;
	struct people *next;
} people;

int main()
{
	int count = NUM;
	people p[NUM];
	people *head;
	head = p; //head 指向p[0]
	//1~13编号
	for (int i = 0; i < NUM; i++)
	{
		head->num = i + 1;
		head->next = &p[i + 1];
		head = head->next;
	}
	//最后一个元素指向第一个元素 , 形成环
	p[NUM - 1].next = p;

	int i = 1;
	head = p;
	while (count > 1)
	{
		//跳过已经被淘汰的节点
		if (head->num == 0)
		{
			head = head->next;
			continue;
		}
		if (i == 3)
		{
			//被淘汰的节点,num置为0
			printf("第 %d 位置被淘汰\n", head->num);
			head->num = 0;
			count--;
		}
		head = head->next;
		i++;
		if (i > 3)
		{
			i = 1;
		}
	}
	printf("--------------\n");
	while (head->num == 0)
	{
		//非0节点即为最后留下的
		head = head->next;
		if (head->num != 0)
		{
			printf("留到最后的是 %d \n", head->num);
		}
	}

	return 0;
}

在这里插入图片描述

//13个人围成一圈,从第1个人开始顺序报号1,2,3。凡报到3者退出圈子。
//找出最后留在圈子中的人原来的序号。要求用链表实现。
#include <stdio.h>
#define N 13
struct One
{
	int n;
	struct One *nextp;
};
int main()
{
	int i,count=N;
	struct One one[N];
	struct One *head;
	head = one;
	for (i = 0; i < N; i++) //将链表首尾相连 并1-13赋值。
	{
		one[i].n = i + 1;
		one[i].nextp = &one[i + 1];
	}
	one[N-1].nextp = &one;
	//for (i = 0; i<N; i++)  //这两种赋值方式都成功。
	//{
	//	head->n = i + 1;
	//	head->nextp = &one[i + 1];
	//	head = head->nextp;
	//}
	//one[N-1].nextp = one;
	i = 1;
	head = one;
	while (count>1)
	{
		if (head->n == 0)
		{
			head = head->nextp;  
			//head=head+1;  //这两种也都成功。
			continue;
		}
		if (i == 3)
		{
			printf("out %d\n", head->n);
			head->n = 0;
			count--;
		}
		head = head->nextp;
		i++;
		if (i > 3)
		{
			i = 1;
		}
	}
	printf("-----------\n");
	while (head->n == 0)
	{
		head = head->nextp; //head->nextp是一个指针(地址),故可以直接赋值给head。
		if (head->n != 0)
		{
			printf("the last is %d\n", (*head).n);
		}
	}
	return 0;
}

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

lihongli000

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值