题目:

代码1(数组实现):

//
// main.cpp
// DS-击鼓传花
//
// Created by wasdns on 16/11/9.
// Copyright © 2016年 wasdns. All rights reserved.
// #include <iostream>
#include <cstdio>
#include <cstdlib>
#include <string.h>
#include <string>
using namespace std; int killman[1000]; //killer and alive man int main()
{
int n, m; cin >> n >> m; memset(killman, 0, sizeof(killman)); int killt = 1; //死亡情况,如果killt=总人数,结束游戏
killman[1] = -1; //第一位发言差,根据题目要求首杀 int rcd = 1, turn = 1; //rcd记录存活者,turn为指针
int cnt = 0; //cnt计数器,一旦为m,杀掉此时指针指向者 while (1)
{
if (turn > n) { //指针越界
turn %= n;
} if (killman[turn] == -1) { //此时指向的人已经死亡
turn++; continue;
} cnt++; //指向的人还活着,更新计数器 if (cnt == m) { //计数器计数为m,同时指针指向的人还活着
killman[turn] = -1; //杀死 killt++; //总死亡人数加一 rcd = turn; //rcd记录截止目前最后一位阵亡的同学 cnt = 0; //计数器置0 if (killt == n) break; //游戏结束
} turn++; //游戏还没有结束,指针后移
} cout << rcd << endl; //最后一位同学假死,存活 return 0;
}

结果:

代码2(指针实现):

//
// main2.cpp
// DS-击鼓传花
//
// Created by wasdns on 16/11/9.
// Copyright © 2016年 wasdns. All rights reserved.
// #include <cstdio>
#include <string.h>
#include <iostream>
#include <algorithm>
#include <string>
using namespace std; struct killman {
int num;
killman *next;
}; /*
建立循环链表:
*/ killman* CreatCircle(int n) { killman *k; k = new killman;
k -> next = NULL;
k -> num = 1; killman *p1, *p2; p1 = p2 = k; for (int i = 2; i <= n; i++)
{
p1 = new killman; p1 -> next = NULL;
p1 -> num = i; p2 -> next = p1;
p2 = p1;
} p1 -> next = k; return k;
} /*
根据题目要求,找到开始节点的前一个节点。
目的是为了删除第一个节点。
*/ killman* FindKpre(killman *k)
{
killman *p;
p = k; while (p -> next != k) {
p = p -> next;
} return p;
} /*
杀人游戏主体:
*/ int killgame(killman *k, int m)
{
int cnt = m; //计数器cnt,根据题目要求,初始置m killman *p1, *p2; //p1 为指向当前节点的指针;
//p2 为指向前一个节点的指针。
p1 = k;
p2 = FindKpre(k); //找到开始节点的前一个节点,进行删除操作 while (1)
{
if (cnt == m) { //计数器达到阈值时,删除当前节点 p1 = p1 -> next;
p2 -> next = p1; cnt = 1; //注意:删除之后,本质上进行了前移;
//计数器置1.
continue;
} if (p2 -> next == p2) break; //当出现回环(loop)的时候:
//说明只剩下当前节点,游戏结束。 cnt++; //计数器尚未溢出,指针前移,更新计数器。 p2 -> next = p1;
p2 = p1; p1 = p1 -> next;
} return p1 -> num; //返回游戏的赢家
} /*
Debug,输出环形链表:
*/ void KPrint(killman *k) { killman *p;
p = k -> next; while (p != k) {
cout << "p " << p -> num << endl;
p = p -> next;
} } int main()
{
int n, m; cin >> n >> m; killman *k; k = CreatCircle(n); //KPrint(k); int rcd = 1; rcd = killgame(k, m); cout << rcd << endl; return 0;
}

结果:

小结:

此题是经典的约瑟夫问题,采用ADT表,有两种实现方法:(1)指针实现 (2)数组实现。

需要注意的点是:

  • 题目把第一个人直接出局
  • 计数器和指针更新时的位置(更新指针之后马上更新计数器!)
  • 循环结束条件

可以看到,使用环形链表的性能更优,但是实现相比数组而言更加复杂。

2016/11/9

DS实验题 击鼓传花的更多相关文章

  1. DS实验题 融合软泥怪-2 Heap实现

    题目和STL实现:DS实验题 融合软泥怪-1 用堆实现优先队列 引言和堆的介绍摘自:Priority Queue(Heaps)--优先队列(堆) 引言: 优先队列是一个至少能够提供插入(Insert) ...

  2. DS实验题 Old_Driver UnionFindSet结构 指针实现邻接表存储

    题目见前文:DS实验题 Old_Driver UnionFindSet结构 这里使用邻接表存储敌人之间的关系,邻接表用指针实现: // // main.cpp // Old_Driver3 // // ...

  3. 洛谷 P2529 [SHOI2001]击鼓传花 解题报告

    P2529 [SHOI2001]击鼓传花 题意:求出\(n!\)末尾最后一位非0数字 数据范围:\(n<=10^{100}\) 我们从简单的开始考虑 1.显然,\(n!\)可以被这么表示 \(n ...

  4. FZU 1962 新击鼓传花游戏

    新击鼓传花游戏 Time Limit: 1000ms Memory Limit: 32768KB This problem will be judged on FZU. Original ID: 19 ...

  5. 原生JS实现队结构及利用队列模拟‘击鼓传花’游戏

    1. 前言 队列,是一种遵从先进先出(FIFO,First-In-First-Out)原则的有序集合.队列在尾部添加新元素,并从顶部移除元素,最新添加的元素必须排在队列的末尾. 2.功能说明 enqu ...

  6. 击鼓传花联想到了Java设计模式:责任链模式

    目录 应用场景 简单示例 责任链模式 定义 意图 主要解决问题 何时使用 优缺点 击鼓传花的故事 应用场景 http web请求处理,请求过来后将经过转码.解析.参数封装.鉴权等一系列的处理(责任), ...

  7. DS实验题 Dijkstra算法

    参考:Dijkstra算法 数据结构来到了图论这一章节,网络中的路由算法基本都和图论相关.于是在拿到DS的实验题的时候,决定看下久负盛名的Dijkstra算法. Dijkstra的经典应用是开放最短路 ...

  8. DS实验题 sights

    算法与数据结构实验题 6.3 sights ★实验任务 美丽的小风姑娘打算去旅游散心,她走进了一座山,发现这座山有 n 个景点, 由于山路难修,所以施工队只修了最少条的路,来保证 n 个景点联通,娇弱 ...

  9. DS实验题 order

    算法与数据结构 实验题 6.4 order ★实验任务 给出一棵二叉树的中序遍历和每个节点的父节点,求这棵二叉树的先序和后序遍历. ★数据输入 输入第一行为一个正整数n表示二叉树的节点数目,节点编号从 ...

随机推荐

  1. jQuery工具函数

    要点:1.字符串操作2.数组和对象操作3.测试操作4.URL 操作5.浏览器检测6.其他操作 工具函数是指直接依附于 jQuery 对象,针对 jQuery 对象本身定义的方法,即全局性的函数.它的作 ...

  2. LinuxC语言读取文件,分割字符串,存入链表,放入另一个文件

    //file_op.c #include <string.h> #include <stdio.h> #include <stdlib.h> struct info ...

  3. 为什么C++类定义中,数据成员不能被指定为自身类型,但可以是指向自身类型的指针或引用?为什么在类体内可以定义将静态成员声明为其所属类的类型呢 ?

    static的成员变量,不是存储在Bar实例之中的,因而不会有递归定义的问题. 类声明: class Screen: //Screen类的声明 1 类定义: class Screen{ //Scree ...

  4. jquery easy ui 1.3.4 布局layout(4)

    4.1.easyui布局-layout 在easyui里面只有一种布局方式,layout(东.南.西.北.中)的布局方式,创建layout布局的方式如下: <div id="cc&qu ...

  5. Sql server之路 (四)添加本地数据库MDF文件

    安装环境 VS2008  Vs2008 Sp1 安装系统 Win8 1.创建窗体 右键添加新项 上一步 上一步 点击确定 双击Database1.mdf文件 在列名出填写字段名 保存 Ctrl+S 点 ...

  6. PHP生成token防止表单重复提交

    .提交按钮置disabled 当用户提交后,立即把按钮置为不可用状态.这种用js来实现. 提交前代码如下: $()  {  $exec="insert into student (user_ ...

  7. 如何用Linux的命令正确识别cpu的个数和核数【转】

    判断依据: 1.具有相同core id的cpu是同一个core的超线程. 2.具有相同physical id的cpu是同一颗cpu封装的线程或者cores. 英文版: 1.Physical id an ...

  8. zookeeper 4 letter 描述与实践

    命令示例描述 Conf echo conf | nc localhost 2181 (New in 3.3.0)输出相关服务配置的详细信息.比如端口.zk数据及日志配置路径.最大连接数,session ...

  9. Intent的七大属性

    1.Action Action属性代表系统要执行的动作 系统提供如下常用的Action属性 *ACTION_MAIN:应用程序入口点 *ACTION_VIEW:显示指定数据 *ACTION_EDIT: ...

  10. 用DTD约束XML详解及示例

    文档类型定义(DTD)可定义合法的XML文档构建模块.它使用一系列合法的元素来定义文档的结构. dtd的三种引入方式    (1)引入外部的dtd文件 <!DOCTYPE 根元素名称 SYSTE ...