UVa 1606 两亲性分子

https://vjudge.net/problem/UVA-1606

题意:平面上有n个点,每个点为白点或者黑点。现在需放置一条隔板,使得隔板一侧的白点数加上另一侧的黑点数总数最大。隔板上的点可以看做是在任意一侧。

思路:每次可以选取两个点作为隔板,所以我们可以先枚举一个基准点,然后算出其他点关于基准点的相对坐标和关于坐标系的极角(可以用atan2函数来计算,返回与x坐标值的角度),剩下的点依次与基准点形成分隔线,然后将一条直线绕基准点旋转,每当直线扫过一个点,就可以动态修改两侧的点数。

本题还有一个优化的方法,那就是如果该点是黑点,就可以将它的坐标关于基准点旋转180°,这样扫描时就只需要计算一侧的白点数了。

动态维护的情况如下:

首先,红色的为分割线,扫描线从红线开始逆时针扫描,当扫描线逆时针旋转至黄线时,由于旋转角度已大于180°,所以点数为3个。此时需要移动分隔线,即分隔线变为了蓝线,由于分隔线改变,原来的1号点到了分隔线的右边,此时总的点数需要减去1。此时扫描线与分隔线小于180°,所以扫描线还可以继续逆时针旋转直到y负轴,由于经过了2,3两点,总的点数需要加上2。此时总点数为4。之后分隔线继续旋转,重复上述步骤,直到所有分隔线都模拟完毕。

 1 #include<iostream>
 2 #include<algorithm>
 3 #include<cstring>
 4 #include<cmath>
 5 using namespace std;
 6
 7 const int maxn = 1000+5;
 8
 9 int n;
10 int ans;
11
12 struct node
13 {
14     int x, y;
15     int color;
16     double rad;
17     bool operator < (const node&rhs) const
18     {
19         return rad < rhs.rad;
20     }
21 }op[maxn],p[maxn];
22
23
24 bool left(node a, node b)
25 {
26     return a.x*b.y - a.y*b.x >= 0;
27 }
28
29 void solve()
30 {
31     if (n <= 3)  { ans = n; return; }
32     ans = 0;
33     for (int i = 0; i < n; i++)  //枚举选择基点,一共有n个基点可以选择
34     {
35         int k = 0;
36         for (int j = 0; j < n; j++)    //计算出各个点关于基点的相对坐标
37         {
38             if (i == j)  continue;
39             p[k].x = op[j].x - op[i].x;
40             p[k].y = op[j].y - op[i].y;
41             if (op[j].color)    //如果为黑点,则可以将它旋转180之后变为白点来计数
42             {
43                 p[k].x = -p[k].x; p[k].y = -p[k].y;
44             }
45             p[k].rad = atan2(p[k].y, p[k].x);   //求极角,返回角度值
46             k++;
47         }
48         sort(p, p + k);
49         //基准点-p[L]为分割线,基准点-p[R]为扫描线
50         int    L = 0, R = 0, cnt = 2;   //初始点数设为2,即分隔线上的两个点
51         while (L < k)   //每个点都尝试与基点成为分割线
52         {
53             if (R == L)  { R = (R + 1) % k; cnt++; }  //空区域,数量+1,后面还会减去的
54             while (R != L && left(p[L], p[R]))   //R不等于L并且在180度之内
55             {
56                 R = (R + 1) % k;
57                 cnt++;
58             }
59             cnt--; //分隔线旋转,原本在分隔线上的点到了右边,所以要减去
60             L++;  //分隔线旋转
61             ans = max(ans, cnt);
62         }
63     }
64 }
65
66 int main()
67 {
68     //freopen("D:\\txt.txt", "r", stdin);
69     while (cin >> n && n)
70     {
71         for (int i = 0; i < n; i++)
72             cin >> op[i].x >> op[i].y >> op[i].color;
73         solve();
74         cout << ans << endl;
75     }
76     return 0;
77 }
时间: 2024-07-30 01:33:35

UVa 1606 两亲性分子的相关文章

UVA 1606 Amphiphilic Carbon Molecules 两亲性分子 (极角排序或叉积,扫描法)

任意线可以贪心移动到两点上.直接枚举O(n^3),会TLE. 所以采取扫描法,选基准点,然后根据极角或者两两做叉积比较进行排排序,然后扫一遍就好了.旋转的时候在O(1)时间推出下一种情况,总复杂度为O(n^2logN)就可以过了. 另外,本题有个很巧妙的技巧,就是一点等效与相反坐标的相反颜色的点. 第一次写,细节还是蛮多的,花了好久才搞清所有细节... 极角排序版,比较容易理解,932ms. #include<bits/stdc++.h> using namespace std; const

【极角排序、扫描线】UVa 1606 - Amphiphilic Carbon Molecules(两亲性分子)

Shanghai Hypercomputers, the world's largest computer chip manufacturer, has invented a new class of nanoparticles called Amphiphilic Carbon Molecules (ACMs). ACMs are semiconductors. It means that they can be either conductors or insulators of elect

UVa 1606 (极角排序) Amphiphilic Carbon Molecules

如果,没有紫书上的翻译的话,我觉得我可能读不懂这道题.=_=|| 题意: 平面上有n个点,不是白点就是黑点.现在要放一条直线,使得直线一侧的白点与另一侧的黑点加起来数目最多.直线上的点可以看作位于直线的任意一侧. 分析: 首先假设直线经过两个点,否则可以移动直线使其经过两个点,并且总数不会减少. 所以,我们可以先枚举一个基点,然后以这个点为中心. 围绕基点将其他点按照极角排序,将直线旋转,统计符合要求的点的个数. 小技巧: 如果将所有的黑点以基点为中心做一个中心对称,则符合要求的点的个数就变成了

uva 1606 amphiphilic carbon molecules【把缩写写出来,有惊喜】(滑动窗口)——yhx

Shanghai Hypercomputers, the world's largest computer chip manufacturer, has invented a new classof nanoparticles called Amphiphilic Carbon Molecules (ACMs). ACMs are semiconductors. It meansthat they can be either conductors or insulators of electro

UVa 1606 Amphiphilic Carbon Molecules (扫描法+极角排序)

题意:平面上有 n 个点,每个点不是黑的就是白的,现在要放一个隔板,把它们分成两部分,使得一侧的白点数加上另一侧的黑点数最多. 析:这个题很容易想到的就是暴力,不妨假设隔板至少经过两个点,即使不经过也可以通过平移使它经过,然后每次枚举两个点,当作隔板,枚举量是n*n, 然后计算是 n,那么时间复杂度就是 n3 ,一定会超时的,我产可以这样想,先枚举一个点,然后绕这个点旋转,每扫过一个点,就动态修改两侧的点数, 在转一周过程中,每个点至多扫到两次,这个过程复杂是 n,扫描前进行极角,时间复杂度是

uva 1606 (计算几何)

这是我人生中的第一个计算几何题,好好收藏一下. 1.当一个点关于远点对称之后,除了两点之间的连线外,其他的任何直线都会讲两点分成两个部分 2.极角排序 3.线的旋转 4.叉积求 sin #include <iostream> #include <cstdio> #include <cmath> #include <algorithm> using namespace std; class Node { public: int x,y,color; doubl

UVa 1606 Amphiphilic Carbon Molecules 题解

难度:β 建议用时:40 min 实际用时:1 h 题目:?? 代码:?? 这题我看了刘汝佳大神的代码,在上面改了几个变量的名称,方便理解. 这是一道简单的几何题(不是真正的集合题,更可以说是一个坐标题),所以长话短说了. 题目告诉我们有一些随机分布的点,要我们用一个隔板把平面分开,然后往两边分别倒入水和丙酮来溶解掉这些点. 然而这些点有的只能溶解在水里,有的只能溶解到丙酮里. 任务是找到最大的溶解的点的数量. 这题建模就是按照刘汝佳大神的方法,一个隔板分开平面,要使左侧黑点数加右侧白点数的和最

UVA - 1606 Amphiphilic Carbon Molecules 极角扫描法

题目:点击查看题目 思路:这道题的解决思路是极角扫描法.极角扫描法的思想主要是先选择一个点作为基准点,然后求出各点对于该点的相对坐标,同时求出该坐标系下的极角,按照极角对点进行排序.然后选取点与基准点形成的线对点进行扫描,基准线为遍历选取,扫描线扫过的点,减去基准线扫过的点即为所要求的点的数量.同时注意到我们要求的是线两边的两种点的数量,于是一种点比如黑点可以旋转180度,然后之考察这180度内的百点数量即可.本题的基准点选取复杂度为O(n),极角排序复杂度O(nlogn),扫描复杂度O(n),

UVA - 1606 Amphiphilic Carbon Molecules (计算几何,扫描法)

平面上给你一些具有黑或白颜色的点,让你设置一个隔板,使得隔板一侧的黑点加上另一侧的白点数最多.隔板上的点可视作任意一侧. 易知一定存在一个隔板穿过两个点且最优,因此可以先固定以一个点为原点,将其他点中的黑点移到对称的位置,并将所有点按极角排序,然后双指针遍历其他点,利用尺取法维护一个角度不超过180°的区间(算角度是否大于180°可以用叉积).对每个点都算一遍,取最大区间长度即为最终答案. 区间尺取部分网上的代码基本都的lr表示法,比较冗长,我这里给出了lw表示法(w代表区间长度),代码简洁了许