蚂蚁感冒
长 100 厘米的细长直杆子上有 n 只蚂蚁。
它们的头有的朝左,有的朝右。
每只蚂蚁都只能沿着杆子向前爬,速度是 1 厘米/秒。
当两只蚂蚁碰面时,它们会同时掉头往相反的方向爬行。
这些蚂蚁中,有 1 只蚂蚁感冒了。
并且在和其它蚂蚁碰面时,会把感冒传染给碰到的蚂蚁。
请你计算,当所有蚂蚁都爬离杆子时,有多少只蚂蚁患上了感冒。
输入格式
第一行输入一个整数 n, 表示蚂蚁的总数。
接着的一行是 n 个用空格分开的整数 Xi, Xi 的绝对值表示蚂蚁离开杆子左边端点的距离。
正值表示头朝右,负值表示头朝左,数据中不会出现 0 值,也不会出现两只蚂蚁占用同一位置。
其中,第一个数据代表的蚂蚁感冒了。
输出格式
输出1个整数,表示最后感冒蚂蚁的数目。
数据范围
1<n<50,
0<|Xi|<100
输入样例1:
3
5 -2 8
输出样例1:
1
输入样例2:
5
-10 8 -20 12 25
输出样例2:
3
难度:简单
时/空限制:1s / 64MB
总通过数:8499
总尝试数:16408
来源:第五届蓝桥杯省赛C++A/B组
算法分析
分析
首先我们必须要明白两只蚂蚁相撞掉头可以看作时一只蚂蚁穿过了另一只蚂蚁,因为相撞之后两只蚂蚁都感冒了,掉不掉头其实无所谓,毕竟都感冒了,这样的话这题就简单多了。我们先不考虑特殊情况,先来看看一般情况:
第一只蚂蚁不管方向朝哪里,只要它右边的蚂蚁向左走就可能碰撞感染,同样,第一只蚂蚁左边的蚂蚁只要朝右边走也可能被感染,这样就很容易得到ans=right+left+1。这里left表示左边蚂蚁向右走的数量,right表示右边蚂蚁向左走的数量,1是指第一只蚂蚁本身。
还有一种特殊情况,就是当第一只蚂蚁向左走的时候,如果第一只蚂蚁左边没有向右爬行的蚂蚁,由于爬行速度相同,所以不管第一只蚂蚁右边有多少向左爬行的,其右边的蚂蚁永远不可能被感染。同理,当第一只蚂蚁向右走的时候,如果第一只蚂蚁右边没有向左爬行的蚂蚁,其左边也永远不可能感染。
提交代码:
c++
#include<bits/stdc++.h> using namespace std; int n; int x[110]; int main() { int left = 0, right = 0; // 分别表示在x[0]左边 然后从左往右走的蚂蚁数量, 在x[0]右边 然后从右往左走的蚂蚁数量 // left 的蚂蚁 和 right的蚂蚁最终会相互感染 这个就是这个题的核心 // 这里的一个核心是 就是在计算left和right的时候不要考虑第一只 // 蚂蚁的方向 不然这个题就做不出来了 cin >> n; for (int i = 0; i < n; ++ i) cin >> x[i]; for (int i = 1; i < n; ++ i) { if (x[i] > 0 && abs(x[0]) > abs(x[i]) ) left ++; else if (x[i] < 0 && abs(x[0]) < abs(x[i]) ) right ++; } if (x[0] > 0 && right == 0 || x[0] < 0 && left == 0) cout << 1 << endl; // 在这里的时候 才考虑第一只蚂蚁的 // 方向 也就是如果蚂蚁向右走 如果蚂蚁的右边没有 // 从右往左的蚂蚁 那么就只有自己感染 // 另一种情况同理 else cout << left + right + 1 << endl; return 0; }
Java
import java.util.*; import java.io.*; public class Main { static int left, right; public static void main(String [] args) throws IOException { BufferedReader reader = new BufferedReader(new InputStreamReader(System.in)); int n = Integer.parseInt(reader.readLine()); String [] strs = reader.readLine().split(" "); int [] x = new int [100]; for (int i = 0; i < n; ++ i) x[i] = Integer.parseInt(strs[i]); for (int i = 1; i < n; ++ i) { if (x[i] > 0 && Math.abs(x[0]) > Math.abs(x[i])) left ++; if (x[i] < 0 && Math.abs(x[0]) < Math.abs(x[i])) right ++; } if (x[0] > 0 && right == 0 || x[0] < 0 && left == 0) System.out.println("1"); else System.out.println(left + right + 1); } }