C语言 | 数据结构—约瑟夫环问题

简介: 目录首先什么是约瑟夫环约瑟夫环实现方式一、创建结构体变量二、初始化链表三、构建循环链表四、删除链表 五、完整代码及注释讲解首先什么是约瑟夫环约瑟夫环是循环链表中的一个经典问题;题目描述:n 个人围成一圈,从第一个人开始报数,数到 m 的人出列,再由下一个人重新从 1 开始报数,数到 m 的人再出圈,依次类推,直到所有的人都出圈;假设10个人围成一圈,依次编号1到10,按从小到大顺序报数,报到3的人出局,流程示意图如下约瑟夫环实现方式我个人倾向于循环链表;一、创建结构体变量typedef struct Node{ int data; //数据域 st

目录

首先什么是约瑟夫环

约瑟夫环实现方式

一、创建结构体变量

二、初始化链表

三、构建循环链表

四、删除链表

五、完整代码及注释讲解


首先什么是约瑟夫环

约瑟夫环循环链表中的一个经典问题;题目描述:n 个人围成一圈,从第一个人开始报数,数到 m 的人出列,再由下一个人重新从 1 开始报数,数到 m 的人再出圈,依次类推,直到所有的人都出圈;

假设10个人围成一圈,依次编号1到10,按从小到大顺序报数,报到3的人出局,流程示意图如下

约瑟夫环实现方式

我个人倾向于循环链表

一、创建结构体变量

typedefstructNode{
intdata;  //数据域structNode*next;  //指针域}Node;

二、初始化链表

Node*Create(){
Node*head;
head= (Node*)malloc(sizeof(Node));
if (head==NULL) {
exit(1);
     }
head->next=NULL;
returnhead;
}

三、构建循环链表

创建一个临时结点tail,将头结点head赋予tail,插入新结点p,让tailnext指向pp的next指向head的下一个结点即结点p;同时让tail移到p的位置,为下个结点插入做准备;这样一个循环链表就初步完成了;

代码块

Node*Push(Node*head,Node*tail,int1){
p->data=i;
tail->next=p;
p->next=head->next;
tail=p;
returnhead;
}

四、删除链表

voidPrint(Node*head){
Node*q=head;
q->next=p->next;
printf("%d ", p->data);
free(p);
p=q->next;
}

五、完整代码及注释讲解

#include<stdio.h>#include<stdlib.h>typedefstructNode{
intdata;
structNode*next;
}Node;
//创建结构体变量Node*Create(){
Node*head;
head= (Node*)malloc(sizeof(Node));
if (head==NULL) {
exit(1);
     }
head->next=NULL;
returnhead;
}
intmain()
{
intn, m,i;
Node*tail, *p, *q;
Node*head=Create();
scanf("%d %d", &n, &m);  //输入n个人围一圈及报数m的人出局//判断如果插入的数据为0或者以报数0为出局,则结束操作if (n==0||m==0) {
return0;
     }
else {
tail=head;  //开始没有数据,故尾结点tail与头结点重合for ( i=0; i<n; i++) {
p= (Node*)malloc(sizeof(Node));  //插入新结点需,先申请动态内存if (p==NULL) {        //判断动态内存是否申请成功printf("申请失败!");
exit(1);
             }
p->data=i+1; //以下4步为插入新结点及数据的操作,具体分析请看上面构建循环链表tail->next=p;
p->next=head->next;
tail=p;
         }
     }
p=head->next;  //插入完数据后,将最后一个结点的临时结点移到第一个数据处q=tail;   //然后临时结点到尾结点处i=1;
while (p!=q) {      //首尾结点是否重合,重合则表示只剩一个数据,结束循环if (i==m) {     //对报数m的人进行出局操作q->next=p->next;  //以下四步为删除操作printf("%d ", p->data);
free(p);     //一定记得将删除链表处的内存释放,以免内存内存泄漏p=q->next;
i=1;     //删除后,重新从1开始报数         }
else {   //没有报数到m,则p,q结点都往后移一位q=p;  //先q移到p的位置p=q->next;  //然后p移到q的下一个位置i++;
         }
     }
printf("%d", p->data);  //打印最后一位出局的人的号数return0;
}


相关文章
|
6天前
|
搜索推荐 C语言
【C语言/数据结构】排序(归并排序|计数排序|排序算法复杂度)
【C语言/数据结构】排序(归并排序|计数排序|排序算法复杂度)
11 0
|
6天前
|
C语言
【C语言/数据结构】排序(快速排序及多种优化|递归及非递归版本)
【C语言/数据结构】排序(快速排序及多种优化|递归及非递归版本)
7 0
|
6天前
|
C语言
【C语言/数据结构】排序(选择排序,推排序,冒泡排序)
【C语言/数据结构】排序(选择排序,推排序,冒泡排序)
12 0
|
6天前
|
C语言
【C语言/数据结构】排序(直接插入排序|希尔排序)
【C语言/数据结构】排序(直接插入排序|希尔排序)
13 4
|
6天前
|
C语言
【C语言/数据结构】二叉树(层序遍历|判断完全二叉树|性质)
【C语言/数据结构】二叉树(层序遍历|判断完全二叉树|性质)
279 52
|
8天前
|
存储 缓存 算法
【C 言专栏】C 语言中的数据结构应用
【5月更文挑战第4天】本文探讨了C语言中的核心数据结构,包括数组、链表(单链表和双链表)、栈、队列、二叉树(如二叉搜索树和二叉堆)以及图结构。这些数据结构在程序设计中扮演着关键角色,如数组的快速访问、链表的动态管理、栈和队列的处理流程控制、树和图的复杂关系表示。理解并选择适当的数据结构可优化程序性能,而内存管理和算法优化则进一步提升效率。通过案例分析和展望未来发展趋势,本文旨在帮助读者深化对C语言数据结构的理解和应用。
【C 言专栏】C 语言中的数据结构应用
|
17天前
|
存储 算法 C语言
C语言进阶:顺序表(数据结构基础) (以通讯录项目为代码练习)
C语言进阶:顺序表(数据结构基础) (以通讯录项目为代码练习)
|
17天前
|
C语言
C语言:内存函数(memcpy memmove memset memcmp使用)
C语言:内存函数(memcpy memmove memset memcmp使用)
|
2天前
|
存储 编译器 C语言
C语言:字符函数 & 字符串函数 & 内存函数
C语言:字符函数 & 字符串函数 & 内存函数
11 2
|
11天前
|
缓存 安全 编译器
【C 言专栏】C 语言函数的高效编程技巧
【5月更文挑战第1天】本文探讨了C语言中函数的高效编程技巧,包括函数的定义与作用(如代码复用和提高可读性)、设计原则(单一职责和接口简洁)、参数传递方式(值传递、指针传递和引用传递)、返回值管理、调用约定、嵌套与递归调用,以及函数优化技巧和常见错误避免。掌握这些技巧能提升C语言代码的质量和效率。
【C 言专栏】C 语言函数的高效编程技巧