首页 > 代码库 > Hdu 1156
Hdu 1156
题目链接
Brownie Points II
Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 207 Accepted Submission(s): 77Problem DescriptionStan and Ollie play the game of Odd Brownie Points. Some brownie points are located in the plane, at integer coordinates. Stan plays first and places a vertical line in the plane. The line must go through a brownie point and may cross many (with the same x-coordinate). Then Ollie places a horizontal line that must cross a brownie point already crossed by the vertical line.
Those lines divide the plane into four quadrants. The quadrant containing points with arbitrarily large positive coordinates is the top-right quadrant.
The players score according to the number of brownie points in the quadrants. If a brownie point is crossed by a line, it doesn‘t count. Stan gets a point for each (uncrossed) brownie point in the top-right and bottom-left quadrants. Ollie gets a point for each (uncrossed) brownie point in the top-left and bottom-right quadrants.
Stan and Ollie each try to maximize his own score. When Stan plays, he considers the responses, and chooses a line which maximizes his smallest-possible score.
InputInput contains a number of test cases. The data of each test case appear on a sequence of input lines. The first line of each test case contains a positive odd integer 1 < n < 200000 which is the number of brownie points. Each of the following n lines contains two integers, the horizontal (x) and vertical (y) coordinates of a brownie point. No two brownie points occupy the same place. The input ends with a line containing 0 (instead of the n of a test).
OutputFor each input test, print a line of output in the format shown below. The first number is the largest score which Stan can assure for himself. The remaining numbers are the possible (high) scores of Ollie, in increasing order.
Sample Input113 23 33 43 62 -21 -30 0-3 -3-3 -2-3 -43 -70
Sample OutputStan: 7; Ollie: 2 3;
Accepted Code:
1 /************************************************************************* 2 > File Name: A.cpp 3 > Author: Stomach_ache 4 > Mail: sudaweitong@gmail.com 5 > Created Time: 2014年07月27日 星期日 14时45分32秒 6 > Propose: 7 ************************************************************************/ 8 9 #include <cmath> 10 #include <string> 11 #include <vector> 12 #include <cstdio> 13 #include <fstream> 14 #include <cstring> 15 #include <iostream> 16 #include <algorithm> 17 using namespace std; 18 19 const int maxn = 200002; 20 //l维护垂直线左侧的点,r维护垂直线右侧的点 21 int l[maxn], r[maxn]; 22 //每一条垂直于x轴的直线信息 23 struct Line { 24 int x, y; 25 friend bool operator < (Line a, Line b) { 26 return a.x < b.x; 27 } 28 }line[maxn]; 29 //保存所有y轴坐标 30 int y[maxn]; 31 int n, w; 32 33 //BIT 34 int lowbit(int x) { 35 return x & -x; 36 } 37 38 void add(int t[], int x, int v) { 39 while (x <= w) { 40 t[x] += v; 41 x += lowbit(x); 42 } 43 } 44 45 int sum(int t[], int x) { 46 int res = 0; 47 while (x > 0) { 48 res += t[x]; 49 x -= lowbit(x); 50 } 51 return res; 52 } 53 54 int main(void) { 55 #ifndef ONLINE_JUDGE 56 freopen("in.txt", "r", stdin); 57 #endif 58 while (~scanf("%d", &n) && n) { 59 for (int i = 0; i < n; i++) { 60 scanf("%d %d", &line[i].x, &line[i].y); 61 y[i] = line[i].y; 62 } 63 //y轴坐标离散化 64 sort(y, y + n); 65 w = unique(y, y + n) - y; 66 //按x轴坐标从小到大排序 67 sort(line, line + n); 68 //初始化BIT数组 69 memset(l, 0, sizeof(l)); 70 memset(r, 0, sizeof(r)); 71 //把所有点加入右侧的BIT 72 for (int i = 0; i < n; i++) add(r, lower_bound(y, y + w, line[i].y)+1-y, 1); 73 //Stan是其可以获得的最大的最小值 74 //st保存重复x坐标出现的起点 75 int Stan = -1, st = 0; 76 //保存Ollie可能的结果 77 vector<int> Ollie; 78 for (int i = 1; i <= n; i++) { 79 if (i == n || line[i].x != line[i-1].x) { 80 //把重复的点从右侧BIT中删除 81 for (int j = st; j < i; j++) add(r, lower_bound(y, y + w, line[j].y)+1-y, -1); 82 int stan = -1, ollie = -1; 83 //扫描x坐标重复的点,枚举平行于x轴的直线 84 for (int j = st; j < i; j++) { 85 int f = lower_bound(y, y + w, line[j].y) + 1 - y; 86 int s = sum(l, f-1) + sum(r, w) - sum(r, f); 87 int o = sum(l, w) - sum(l, f) + sum(r, f-1); 88 //为了使ollie最大 89 if (o > ollie) { 90 ollie = o; 91 stan = s; 92 } else if (o == ollie) { 93 stan = min(stan, s); 94 } 95 } 96 //更新最大的最小值 97 if (stan > Stan) { 98 Stan = stan; 99 Ollie.clear();100 Ollie.push_back(ollie);101 } else if (stan == Stan) {102 Ollie.push_back(ollie);103 }104 //把重复的点加入左侧的BIT105 for (int j = st; j < i; j++) add(l, lower_bound(y, y + w, line[j].y)+1-y, 1);106 st = i;107 }108 }109 //注意要将Ollie的结果去重110 sort(Ollie.begin(), Ollie.end());111 int len = unique(Ollie.begin(), Ollie.end()) - Ollie.begin();112 printf("Stan: %d; Ollie:", Stan);113 for (int i = 0; i < len; i++) printf(" %d", Ollie[i]);114 puts(";");115 }116 117 return 0;118 }
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。