题解 P5403 【[CTS2019]田野】
liuzhangfeiabc · · 题解
题目大意:给定平面上一堆线段,求一些凸包完整覆盖这些线段,求凸包周长和的最小值。
一道绝好的乱搞题。
场上对着这个题一脸懵逼,打了40分之后觉得不能就这么弃疗,就打了两个贪心拼在一起,当时想能多拿10分就算撞大运了,出来结果一看65真是激动得不行。
后来才知道我的乱搞还不算厉害,有位大神直接随机化+贪心拿到了95分……在此深受我一拜。
闲话少说进入正题。
一开始我们有n个凸包,我们的任务就是恰当地合并凸包。
我们定义某次合并的价值为:新凸包的周长-原凸包的周长和。显然这个值越小越优。
自然地产生几个问题:
1、如果某次合并的价值是负数,那这次合并是否一定要进行?
2、如果某次合并的价值是正数,那这次合并是否一定不进行?
一次合并多个凸包过于复杂,我们先从2个开始研究。
考虑这样一种贪心策略:每次选择2个凸包,如果合并的价值为负就合并,直到不存在合并的价值为负为止。
第一个问题是,在此过程中的每次合并是否一定最优?
可以证明是这样的。不过由于较(wo)为(bu)繁(hui)琐(zheng),在此略去证明。
但是这样跑出的答案就是最终答案吗?我们发现过不了最后一个样例。
我们惊奇地发现,可能出现单独合并任意两个凸包的价值都为正,但是一次性合并若干集合的价值为负的情况!
此时有一种玄学乱搞:每次随机两个凸包合并(即使价值为正),并用新集合再进行贪心合并,到不能贪心合并时继续随机,并在过程中随时更新最优答案。
多随机几次,恭喜你获得了95分的好成绩!
其实这主要是因为强数据很难造。在绝大多数测试点中,不优的合并都只需要进行1~2次,多次随机很容易就能随机出来。然而唯一没过的点需要的不优合并很多,我用我的随机程序需要跑好几分钟才能随到最优解。
所以有什么靠谱做法吗?
同样不加证明(还是因为我不会证)地给出一个引理:如果某次合并的价值为负,而从中选出任何一个子集来单独合并的价值均不为负,则这次合并一定会进行。更强的结论是,如果某次合并的价值为负,且任意子集合并的价值都大于当前集合,且不存在两个不交的子集的合并价值都为负,那么这次合并一定会进行。
注意这并不是现场讲题时ppt上给出的结论(如果任意子集合并的价值都大于当前集合就一定合并),后者可以很容易地构造数据hack掉。
因此现场讲题时给出的n^5和n^4的“每次找出全局价值最小的集合进行合并”的做法是错的!实测会在第10个点wa掉。
没错我就是一开始先打算写写n^4做法,发现各种花式wa之后强行手玩数据发现了这一点……
但是有意思的是,std的n^3做法却是正确的。实际上,std的做法就是依赖于上文给出的结论。
考虑一个对凸包的dp。
将所有点按照x第一关键字,y第二关键字排序,并从右上到左下枚举一个点。我们尝试选一个以这个点为左下角,且价值最小的凸包。
设f(i,j,k)为以点i为起点,所选的最后一条边为j~k时的最优答案。每次枚举下一个点进行转移。
但是具体怎么计算价值呢?或者换句话说,我们怎么才能知道我们新的凸包会包住哪些旧凸包呢?
有一个经典的套路叫“射线法”。对于每个凸包,我们选一个点(一般是最左下的点),向某个方向做一条射线。在dp时按逆时针顺序进行,则每选一条逆时针方向穿越某条射线的边就减去对应凸包的周长,顺时针方向穿越就加上对应周长。这样转一圈回来,所有被包进去的凸包都会被算1次贡献,而外面的凸包则全抵消掉不会算贡献。
当然这里的边界特判非常多。比如,我们不能把原先已有的凸包分割开,因此选一条边时要与所有原凸包求交判合法性;再比如,如果新凸包恰好过原来某个凸包拉射线的点,怎样才能避免这个凸包的贡献不被重算/漏算?再比如,如果拉的射线恰好过其他点怎么办……这些细节不写不知道,一写吓一跳。具体实现见代码吧。
更进一步,我们有一个喜闻乐见的优化:为什么要在dp过程中时刻保持凸性呢?显然一个不凸的方案一定不优啊。因此我们就不需要记最后一条边而是直接记最后一个点,为了防止循环转移按照极角序dp即可。
可是等等……你刚才不是说这种dp价值最小的凸包的算法不对吗?
这里非常奇妙的一点是:一旦我们找到了一个价值为负的集合,就立即将其合并!然后再从这个点开始进行新的一轮dp。如果从当前点开始没有价值为负的区间,那么我们后面的dp中就不会再考虑以它为起点的情况了!
如此一来,当我们某一次合并一个集合的时候,一定不会包含一个不以当前点为左下角的价值为负的子集(因为如果有这样的子集,它早在之前的dp中就已经被合并了),而这个集合又是以当前点为左下角的价值最小的集合,根据引理,它一定会被合并。
而为什么无法合并就可以跳过一个点呢?因为接下来的合并都一定会涉及到更靠左下的点,这自然无法使得任何一个以当前点为左下角的集合价值变小。
如此dp一次的复杂度是n^2。而每次dp要么会合并集合,要么会直接跳过一个点,这两者都只有On次,总复杂度就是n^3的了。
最后说一句,如果有人会了文中结论的证明,或者找到了把结论/std叉掉的数据,请尽快联系我直接发到uoj群里吧我实在被这个题折磨够了。
由于代码是从n^4强转过来的,所以又长又丑,大家将就着看吧……
#include<bits/stdc++.h>
using namespace std;
#define gc getchar()
#define pc putchar
#define li long long
inline li read(){
li x = 0,y = 0,c = gc;
while(!isdigit(c)) y = c,c = gc;
while(isdigit(c)) x = (x << 1) + (x << 3) + (c ^ '0'),c = gc;
return y == '-' ? -x : x;
}
#define ldb long double
int cnt,tong[510];
struct node{
li x,y;
node(li _x = 0,li _y = 0){x = _x;y = _y;}
}p[510];
inline bool operator < (node q,node w){
return q.x == w.x ? q.y < w.y : q.x < w.x;
}
inline bool cmp(int q,int w){
return p[q].x == p[w].x ? p[q].y < p[w].y : p[q].x < p[w].x;
}
inline node operator + (node q,node w){
return node(q.x + w.x,q.y + w.y);
}
inline node operator - (node q,node w){
return node(q.x - w.x,q.y - w.y);
}
inline li operator * (node q,node w){
return q.x * w.y - q.y * w.x;
}
inline ldb dis(node q,node w){
return sqrtl((q.x - w.x) * (q.x - w.x) + (q.y - w.y) * (q.y - w.y));
}
ldb ans;
struct tb{
vector<int> a;
ldb as;
inline int& operator [] (int x){return a[x];}
}a[260],b[260];
int n,nw[1010],tot;
ldb f[510][510],h[510][510],ds[510][510];
int g[510][510];
int jj[510][510];
bool ff[510][510];
node nwd;
inline bool cpm(int q,int w){
if(p[q].x > nwd.x && p[w].x < nwd.x) return 1;
if(p[q].x < nwd.x && p[w].x > nwd.x) return 0;
if(p[q].x == nwd.x && p[q].y < nwd.y) return 0;
if(p[w].x == nwd.x && p[w].y < nwd.y) return 1;
return (p[q] - nwd) * (p[w] - nwd) > 0;
}
inline bool jiao(node a,node b,node c,node d){
node p1 = a - c,p2 = d - c,p3 = b - c;
if(1.0l * (p1 * p2) * (p3 * p2) >= 0) return 0;
p2 = b - a;p3 = d - a;
return 1.0l * (p1 * p2) * (p3 * p2) > 0;
}
bool shan[260];
int pm[510];
int sx[260];
inline bool cp(int q,int w){
return q > w;
}
bool intb[510];
int st1[1010],ft1,st2[1010],ft2;
inline ldb wk(){
int i;
ft1 = ft2 = 0;
sort(nw + 1,nw + tot + 1);
for(i = 1;i <= tot;++i){
while(ft1 > 1 && ((p[st1[ft1]] - p[st1[ft1 - 1]]) * (p[st1[ft1]] - p[nw[i]]) >= 0)) --ft1;
st1[++ft1] = nw[i];
}
for(i = tot;i;--i){
while(ft2 > 1 && ((p[st2[ft2]] - p[st2[ft2 - 1]]) * (p[st2[ft2]] - p[nw[i]]) >= 0)) --ft2;
st2[++ft2] = nw[i];
}
ldb as = 0;
for(i = 1;i < ft1;++i) as += dis(p[st1[i]],p[st1[i + 1]]);
for(i = 1;i < ft2;++i) as += dis(p[st2[i]],p[st2[i + 1]]);
return as;
}
int main(){
int i,j,k,l,u,v;
n = read();
for(i = 1;i <= n;++i){
a[i].a.resize(2);
p[++cnt].x = read();p[cnt].y = read();tong[cnt] = cnt;
p[++cnt].x = read();p[cnt].y = read();tong[cnt] = cnt;
a[i].as = 2 * dis(p[cnt - 1],p[cnt]);
ans += a[i].as;
}
if(n == 1){printf("%.10lf\n",(double)ans);return 0;}
sort(tong + 1,tong + cnt + 1,cmp);
for(i = 1;i <= cnt;++i) a[tong[i] + 1 >> 1][tong[i] + 1 & 1] = i;
for(i = 1;i <= n;++i) if(a[i][0] > a[i][1]) swap(a[i][0],a[i][1]);
memset(tong,0,sizeof(tong));
sort(p + 1,p + cnt + 1);
for(i = 1;i <= cnt;++i){
int tt = 0;
for(j = 1;j <= cnt;++j) if(i != j) jj[i][++tt] = j;
nwd = p[i];
sort(jj[i] + 1,jj[i] + tt + 1,cpm);
}
for(i = 1;i <= cnt;++i) for(j = i;j <= cnt;++j) ds[i][j] = ds[j][i] = dis(p[i],p[j]);
bool lss = 0,fg = 1;
while(fg){
fg = 0;
for(i = 1;i < n;++i) for(j = i + 1;j <= n;++j){
tot = 0;
for(k = 0;k < a[i].a.size();++k) nw[++tot] = a[i][k];
for(k = 0;k < a[j].a.size();++k) nw[++tot] = a[j][k];
ldb nxt = wk();
if(nxt <= a[i].as + a[j].as){
ans += nxt - a[i].as - a[j].as;
a[i].a.clear();a[i].a.resize(ft1 + ft2 - 2);
for(k = 1;k < ft1;++k) a[i][k - 1] = st1[k];
for(k = 1;k < ft2;++k) a[i][k + ft1 - 2] = st2[k];
a[i].as = nxt;swap(a[j],a[n]);--n;fg = 1;
goto qwq;
}
}
qwq:;
}
for(i = 1;i <= n;++i) for(j = 0;j < a[i].a.size();++j) for(k = 1;k <= n;++k) for(l = 0;l < a[k].a.size();++l) if(i != k || j != l) ff[a[i][j]][a[k][l]] = 1;
while(1){
memset(g,0,sizeof(g));
memset(intb,0,sizeof(intb));
for(i = 1;i <= n;++i) for(j = 0;j < a[i].a.size();++j) intb[a[i][j]] = 1;
for(i = 1;i <= cnt;++i) for(j = 1;j <= cnt;++j) if(!intb[i] || !intb[j]) ff[i][j] = 0;
for(i = 1;i <= cnt;++i) for(j = 1;j <= cnt;++j) f[i][j] = ans;
for(i = 1;i <= n;++i) for(j = 0;j < a[i].a.size();++j) for(k = 1;k <= n;++k) for(l = 0;l < a[k].a.size();++l) if(i != k || j != l){
int p1 = a[i][j],p2 = a[k][l];
if(!ff[p1][p2]) continue;
if((p[p2] - p[p1]) * ((j != a[i].a.size() - 1 ? p[a[i][j + 1]] : p[a[i][0]]) - p[p1]) < 0){
ff[p1][p2] = 0;continue;
}
if((p[p1] - p[p2]) * ((l ? p[a[k][l - 1]] : p[a[k][a[k].a.size() - 1]]) - p[p2]) > 0){
ff[p1][p2] = 0;continue;
}
for(u = 1;u <= n;++u) if(u == n || !lss){
if((i < k || (i == k && j < l)) || !ff[p2][p1]){
for(v = 0;v < a[u].a.size();++v){
if(jiao(p[p1],p[p2],p[a[u][v]],v ? p[a[u][v - 1]] : p[a[u][a[u].a.size() - 1]])){
ff[p1][p2] = ff[p2][p1] = 0;break;
}
}
}
if(!ff[p1][p2]) break;
if(p1 == a[u][0]){
if((p[p2] - p[p1]) * node(1,-1234567890 - p[p1].y) < 0) h[p1][p2] -= a[u].as;
}
else{
nwd = node(p[a[u][0]].x + 1,-1234567890);
li q1 = (p[p1] - p[a[u][0]]) * node(1,-1234567890 - p[a[u][0]].y),q2 = (p[p2] - p[a[u][0]]) * node(1,-1234567890 - p[a[u][0]].y);
if(q1 > 0 && q2 < 0){
if(1.0l * ((nwd - p[p1]) * (p[p2] - p[p1])) * ((p[a[u][0]] - p[p1]) * (p[p2] - p[p1])) < 0) h[p1][p2] -= a[u].as;
}
else if(q1 < 0 && q2 > 0){
if(1.0l * ((nwd - p[p1]) * (p[p2] - p[p1])) * ((p[a[u][0]] - p[p1]) * (p[p2] - p[p1])) < 0) h[p1][p2] += a[u].as;
}
}
}
}
ldb nwa = ans;int wz = 0;
for(i = 1;i <= n;++i) sx[i] = a[i][0];
sort(sx + 1,sx + n + 1,cp);
for(i = 1;i <= n;++i){
int p1 = sx[i],p2;
if(lss && p[a[n][0]] < p[p1]) continue;
pm[p1] = cnt;
for(j = 1;j < cnt;++j) pm[jj[p1][j]] = j;
for(j = 1;j < cnt && p[jj[p1][j]].x >= p[p1].x;++j){
p2 = jj[p1][j];
if(p[p2].x == p[p1].x && p[p2].y < p[p1].y) break;
if(!ff[p1][p2]) continue;
if(f[p1][p2] > ds[p1][p2] + h[p1][p2]){
f[p1][p2] = ds[p1][p2] + h[p1][p2];
g[p1][p2] = p1;
}
}
for(j = 1;j < cnt && p[jj[p1][j]].x >= p[p1].x;++j){
p2 = jj[p1][j];
if(p[p2].x == p[p1].x && p[p2].y < p[p1].y) break;
for(k = 1;k <= cnt;++k) if(ff[p2][k] && pm[k] > pm[p2]){
if(f[p1][k] > f[p1][p2] + ds[p2][k] + h[p2][k]){
f[p1][k] = f[p1][p2] + ds[p2][k] + h[p2][k];
g[p1][k] = p2;
}
}
}
if(f[p1][p1] < -1e-8){
nwa = f[p1][p1];wz = p1;break;
}
}
if(nwa >= -1e-8) break;
ans += nwa;
nw[tot = 1] = wz;
for(i = g[wz][wz];i != wz;i = g[wz][i]) nw[++tot] = i;
nw[++tot] = wz;
int nxtn = 0;
for(i = 1;i <= n;++i){
nwd = node(p[a[i][0]].x + 1,-1234567890);
bool inn = 0;
for(j = 1;j < tot;++j){
if(p[a[i][0]].x == p[nw[j]].x && p[a[i][0]].y == p[nw[j]].y){
inn = 1;break;
}
if(jiao(p[nw[j]],p[nw[j + 1]],p[a[i][0]],nwd)) inn ^= 1;
}
if(inn){
nwa += a[i].as;u = i;
for(int p1 = 1;p1 <= cnt;++p1) for(int p2 = 1;p2 <= cnt;++p2) if(ff[p1][p2]){
if(p1 == a[u][0]){
if((p[p2] - p[p1]) * node(1,-1234567890 - p[p1].y) < 0) h[p1][p2] += a[u].as;
}
else{
nwd = node(p[a[u][0]].x + 1,-1234567890);
li q1 = (p[p1] - p[a[u][0]]) * node(1,-1234567890 - p[a[u][0]].y),q2 = (p[p2] - p[a[u][0]]) * node(1,-1234567890 - p[a[u][0]].y);
if(q1 > 0 && q2 < 0){
if(1.0l * ((nwd - p[p1]) * (p[p2] - p[p1])) * ((p[a[u][0]] - p[p1]) * (p[p2] - p[p1])) < 0) h[p1][p2] += a[u].as;
}
else if(q1 < 0 && q2 > 0){
if(1.0l * ((nwd - p[p1]) * (p[p2] - p[p1])) * ((p[a[u][0]] - p[p1]) * (p[p2] - p[p1])) < 0) h[p1][p2] -= a[u].as;
}
}
}
}
else b[++nxtn] = a[i];
}
for(i = 1;i <= nxtn;++i) a[i] = b[i];
n = nxtn + 1;
a[n].as = nwa;a[n].a.clear();a[n].a.resize(tot - 1);
for(i = tot;i > 1;--i) a[n][tot - i] = nw[i];lss = 1;
}
printf("%.10lf\n",(double)ans);
return 0;
}