小孩报数问题

Description

有N个小孩围成一圈,给他们从1开始依次编号,现指定从第W个开始报数,报到第S个时,该小孩出列,然后从下一个小孩开始报数,仍是报到S个出列,如此重复下去,直到所有的小孩都出列(总人数不足S个时将循环报数),求小孩出列的顺序。

Input

第一行输入小孩的人数N(N<=64)
接下来每行输入一个小孩的名字(人名不超过15个字符)
最后一行输入W,S (W < N),用逗号","间隔

Output

按人名输出小孩按顺序出列的顺序,每行输出一个人名

Sample Input

5
Xiaoming
Xiaohua
Xiaowang
Zhangsan
Lisi
2,3

Sample Output

Zhangsan
Xiaohua
Xiaoming
Xiaowang
Lisi


普通版:
#include<iostream>
#include<cstring>
using namespace std;
int main() {
    char name[56][16];
    int n, w, s;
    cin >> n;
    for (int i = 1; i <= n; i++) {
        cin >> name[i];
    }
    scanf("%d,%d", &w, &s);
    int count = 0;                            //记录经过未标记的名字几次
    int k = 0;                                //记录输出的次数
    for (int i = w; k!=n ; i++) {            //输出 n 次,跳出
        if (i > n) i = i % n;                //循环 1~n~1~n
        if (strcmp(name[i], "1")) count++;    //经过未标记的名字时 count++
        if (count == s) {                    //经过s次未被标记的名字时
            cout << name[i] << endl;
            k++;                            //输出一次 k++
            count = 0;                        //输出以后count重新记数
            w = i;                            //输出之后,w 重新从 i 开始
            strcpy(name[i], "1");            //已经输出的名字标记为 1 
        }
    }
    return 0;
}

线性表:
#include<iostream>
#include<string>
using namespace std;

class SeqList {
public:
    SeqList(string str[], int n);            //构造函数,创建长度为n 的线性表
    string Delete(int i);                    //输出并删除结点
    int getLength() { return length; }        //获取长度
private:
    int length;
    string name[65];
};

SeqList::SeqList(string str[], int n) {        //实现构造函数
    for (int i = 0; i < n; i++) {
        name[i] = str[i];
    }
    length = n;
}

string SeqList::Delete(int i) {
    string x = name[i - 1];                    //定义一个x接收被删除的字符串
    for (int j = i; j < length; j++) {        //从被删除的结点开始,后面的向前移动,覆盖被删除的结点
        name[j - 1] = name[j];                        
    }
    length--;                                //删除一个结点,长度减 1
    return x;
}

int main() {
    string str[65];
    int n, w, s;
    cin >> n;
    for (int i = 0; i < n; i++) {
        cin >> str[i];
    }
    scanf("%d,%d", &w, &s);
    SeqList S(str, n);
    while(n--) {
        int ans = (w + s - 1);                 //ans 表示要输出的名字的序号
        if ( ans==S.getLength() ) {
            w = ans;
            cout << S.Delete(w) << endl;
        }
        else {
            if (ans % S.getLength() != 0) {
                w = ans % S.getLength();
                cout << S.Delete(w) << endl;
            }
            else {
                cout << S.Delete(w--) << endl;
            }
            
        }

    }
    return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
小孩报数问题可以使用循环链表来解决,具体实现步骤如下: 1. 定义一个结构体表示链表节点,包含一个数据域和一个指向下一个节点的指针。 ```c typedef struct node { int data; // 数据域 struct node* next; // 指向下一个节点的指针 } Node; ``` 2. 创建一个循环链表,首先需要输入小孩的总数和每次报数的数字。然后循环创建节点,并将节点链接起来形成循环链表。 ```c int n, m; printf("请输入小孩的总数n和每次报数的数字m:"); scanf("%d%d", &n, &m); Node* head = (Node*)malloc(sizeof(Node)); // 创建头结点 head->data = 1; // 头结点数据域存储编号为1的小孩 head->next = head; // 头结点指向自身 Node* tail = head; // 尾指针指向头结点,用于添加新节点 for (int i = 2; i <= n; i++) { Node* newNode = (Node*)malloc(sizeof(Node)); // 创建新节点 newNode->data = i; // 新节点数据域存储小孩的编号 newNode->next = head; // 新节点指向头结点 tail->next = newNode; // 尾节点指向新节点 tail = newNode; // 尾指针指向新节点 } ``` 3. 实现小孩报数并删除的过程。从头结点开始遍历链表,每次遍历到第m个节点,将该节点从链表中删除。重复该过程,直到链表只剩下一个节点为止。 ```c Node* p = head; // 指针p指向头结点 while (p->next != p) { // 当链表不为空 for (int i = 1; i < m-1; i++) { // 找到第m个节点的前一个节点 p = p->next; } Node* q = p->next; // 指针q指向第m个节点 p->next = q->next; // 删除第m个节点 printf("%d ", q->data); // 输出被删除的小孩编号 free(q); // 释放节点空间 p = p->next; // 指针p指向下一个节点 } printf("%d", p->data); // 输出最后剩下的小孩编号 ``` 完整代码如下:

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值