A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 行我福 中级黑马   /  2015-1-27 12:58  /  1155 人查看  /  2 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

数学有棵“数”,叫“奥数”,我想很多人,都在这个“数”上,羁绊过。今天就和大家分享下用算法解决奥数的一些小问题,请看题目:     一根长度为L厘米的木棍上有n只蚂蚁,每只蚂蚁要么朝左爬,要么朝右爬,速度为1厘米/秒。当两只蚂蚁相撞时,二者同时掉头(掉头时间忽略不计)。给出每只蚂蚁的初始位置和朝向,计算T秒之后每只蚂蚁的位置。

别着急找纸笔,且看如何运用电脑去计算。为方便说明,我这样设定下~
【输入格式】
输入的第一行为数据组数。每组数据的第一行为3个正整数L, T,n(0≤n≤10 000);以下n行每行描述一只蚂蚁的初始位置,其中,整数x为蚂蚁距离木棍左端的距离(单位:厘米),字母表示初始朝向(L表示朝左,R表示朝右)。
【输出格式】
对于每组数据,输出n行,按输入顺序输出每只蚂蚁的位置和朝向(Turning表示正在碰撞)。在第T秒之前已经掉下木棍的蚂蚁(正好爬到木棍边缘的不算)输出Fell off。
【样例输入】

2
10 1 4
1 R
5 R
3 L
10 R
10 2 3
4 R
5 L
8 R

【样例输出】

Case #1:
2 Turning
6 R
2 Turning
Fell off

Case #2:
3 L
6 R
10 R
【分析】
假设你在远处观察这些蚂蚁的运动,会看到什么?一群密密麻麻的小黑点在移动。由于黑点太小,所以当蚂蚁因碰撞而掉头时,看上去和两个点“对穿而过”没有任何区别,换句话说,如果把蚂蚁看成是没有区别的小点,那么只需独立计算出每只蚂蚁在T时刻的位置即可。比如,有3只蚂蚁,蚂蚁1=(1, R),蚂蚁2= (3, L),蚂蚁3=(4, L),则两秒钟之后,3只蚂蚁分别为(3,R)、(1,L)和(2,L)。
注意,虽然从整体上讲,“掉头”等价于“对穿而过”,但对于每只蚂蚁而言并不是这样。蚂蚁1的初始状态为(1,R),因此一定有一只蚂蚁在两秒钟之后处于(3,R)的状态,但这只蚂蚁却不一定是蚂蚁1。换句话说,我们需要搞清楚目标状态中“谁是谁”。
也许读者已经发现了其中的奥妙:所有蚂蚁的相对顺序是保持不变的,因此把所有目标位置从小到大排序,则从左到右的每个位置对应于初始状态下从左到右的每只蚂蚁。由于原题中蚂蚁不一定按照从左到右的顺序输入,还需要预处理计算出输入中的第i只蚂蚁的序号order[i]。完整代码如下。
  1. #include<cstdio>
  2. #include<algorithm>
  3. using namespace std;

  4. const int maxn = 10000 + 5;

  5. struct Ant {
  6.   int id;         //输入顺序
  7.   int p;         //位置
  8.   int d;         //朝向。 -1: 左; 0:转身中; 1:右
  9.   bool operator < (const Ant& a) const {
  10.     return p < a.p;
  11.   }
  12. } before[maxn], after[maxn];

  13. const char dirName[][10] = {"L", "Turning", "R"};

  14. int order[maxn]; //输入的第i只蚂蚁是终态中的左数第order[i]只蚂蚁

  15. int main() {
  16.   int K;
  17.   scanf("%d", &K);
  18.   for(int kase = 1; kase <= K; kase++) {
  19.     int L, T, n;
  20.     printf("Case #%d:\n", kase);
  21.     scanf("%d%d%d", &L, &T, &n);
  22.     for(int i = 0; i < n; i++) {
  23.       int p, d;
  24.       char c;
  25.       scanf("%d %c", &p, &c);
  26.       d = (c == 'L' ? -1 : 1);
  27.       before[i] = (Ant){i, p, d};
  28.       after[i] = (Ant){0, p+T*d, d};         //这里的id是未知的
  29.     }

  30.     //计算order数组
  31.     sort(before, before+n);
  32.     for(int i = 0; i < n; i++)
  33.       order[before[i].id] = i;

  34.     //计算终态
  35.     sort(after, after+n);   
  36.     for(int i = 0; i < n-1; i++)                 //修改碰撞中的蚂蚁的方向
  37.       if(after[i].p == after[i+1].p) after[i].d = after[i+1].d = 0;

  38.     //输出结果
  39.     for(int i = 0; i < n; i++) {
  40.       int a = order[i];
  41.       if(after[a].p < 0 || after[a].p > L) printf("Fell off\n");
  42.       else printf("%d %s\n", after[a].p, dirName[after[a].d+1]);
  43.     }
  44.     printf("\n");
  45.   }
  46.   return 0;
  47. }
复制代码
运行效果:


[


2 个回复

倒序浏览
完全看不懂 :lol。。。

点评

那就不妨用直接用数学方法做下,在看算法,就好理解啦  发表于 2015-1-28 09:39
回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马