[BZOJ5099][POI2018]Pionek(极角排序+two pointers)

几个不会严谨证明的结论:

1.将所有向量按极角排序,则答案集合一定是连续的一段。

当答案方向确定时,则一个向量会被选入答案集合当且仅当向量在答案方向上的投影一定都是正的

所以,两个选中向量中间隔着一个向量,则必然可以将后面所有选中向量均前移一位并使答案不劣。

2.答案集合中不存在两个向量的极角差超过$\pi$。

显然需要保证的是,答案集合中任意两个向量投影不为负值,否则一定可以通过删去其中一个使答案更优。

3.当选择的向量集合区间左端点增加时,右端点单调不减。

感性理解,答案向量的极角会在选择的向量区间中某两个相邻向量极角之间。当区间左端点增加时,答案向量的极角必然增加,故右端点不会减少。

于是,将所有向量极角排序后,对所有扫过的极角极差不超过$\pi$的向量区间更新答案即可。

 #include<cmath>
#include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=(l); i<=(r); i++)
typedef long long ll;
using namespace std; const int N=;
const double pi=acos(-.);
int n;
ll ans;
struct P{
int x,y; double s;
bool operator <(const P &a)const{ return s<a.s; }
P operator +(const P &a)const{ return (P){x+a.x,y+a.y}; }
P operator -(const P &a)const{ return (P){x-a.x,y-a.y}; }
ll len(){ return 1ll*x*x+1ll*y*y; }
}a[N]; int main(){
freopen("bzoj5099.in","r",stdin);
freopen("bzoj5099.out","w",stdout);
scanf("%d",&n);
rep(i,,n) scanf("%d%d",&a[i].x,&a[i].y),a[i].s=atan2(a[i].x,a[i].y);
sort(a+,a+n+); P cur=(P){,}; int x=;
rep(i,,n) a[i+n]=a[i],a[i+n].s+=*pi;
rep(i,,n){
while (x<n* && a[x+].s-a[i].s<=pi) cur=cur+a[++x],ans=max(ans,cur.len());
cur=cur-a[i]; ans=max(ans,cur.len());
}
printf("%lld\n",ans);
return ;
}

[BZOJ5100][POI2018]Plan metra(构造)

若1与n直接相连,则所有点到1与n的距离差的绝对值是相同的,根据距离选择与1还是n直接相连即可。

若不直接相连,则其余所有点到1与n的距离和的最小值就是1到n的距离。

找到所有在1到n的路径上的点,按与1的距离排序,然后相邻的两两连边。其余点算出应该挂在链上哪个点下面,注意判断各种不合法即可。

 #include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=(l); i<=(r); i++)
typedef long long ll;
using namespace std; const int N=;
int n,a[N],b[N],id[N];
struct P{ int x,i; }d[N];
bool operator <(const P &a,const P &b){ return a.x<b.x; } int main(){
freopen("bzoj5100.in","r",stdin);
freopen("bzoj5100.out","w",stdout);
scanf("%d",&n);
rep(i,,n-) scanf("%d",&a[i]);
rep(i,,n-) scanf("%d",&b[i]);
if (n==){ printf("TAK\n1 2 1\n"); return ; }
int len=abs(a[]-b[]);
rep(i,,n-) if (abs(a[i]-b[i])!=len){ len=; break; }
if (len){
puts("TAK"); printf("1 %d %d\n",n,len);
rep(i,,n-)
if (a[i]>b[i]) printf("%d %d %d\n",i,n,b[i]);
else printf("%d %d %d\n",i,,a[i]);
return ;
}
len=; int tot=;
rep(i,,n-) len=min(len,a[i]+b[i]);
rep(i,,n) if (a[i]+b[i]==len) d[++tot]=(P){a[i],i};
sort(d+,d+tot+); d[]=(P){,}; d[++tot]=(P){len,n};
rep(i,,tot) if (d[i].x==d[i-].x){ puts("NIE"); return ; }
rep(i,,tot) id[d[i].x]=d[i].i;
rep(i,,n-) if (a[i]+b[i]>len && (((a[i]+b[i]-len)&) || !id[a[i]-(a[i]+b[i]-len)/])){ puts("NIE"); return ; }
puts("TAK");
rep(i,,tot) printf("%d %d %d\n",d[i-].i,d[i].i,d[i].x-d[i-].x);
rep(i,,n-) if (a[i]+b[i]>len) printf("%d %d %d\n",id[a[i]-(a[i]+b[i]-len)/],i,(a[i]+b[i]-len)/);
return ;
}

[BZOJ5101][POI2018]Powódź(并查集)

类似于Kruskal重构树上DP。

相邻点之间连一条权为墙高的边,将边从小到大排序,每次合并一条边连接的两个连通块。

h[i]表示连通块i中最高的墙,g[i]表示所有水位都不超过h[i]的情况下的方案数。最后答案就是g[1]+H-h[1]。

当某处水位高于h[i]时,水就会往四周流淌,所以合并边权为w的边连接的两个连通块时,h=w,g=(g1+w-h1)*(g2+w-h2)。

意思就是,新的方案数,等于两个连通块在水位均不超过w时的方案数之积。一个连通块水位不超过w的方案数,等于水位不超过h1的方案数,加所有区域水位相等且都大于h1的方案数,即g1+w-h1。

 #include<cstdio>
#include<algorithm>
#define F(A,B) ((A-1)*m+B)
#define rep(i,l,r) for (int i=(l); i<=(r); i++)
typedef long long ll;
using namespace std; const int N=,mod=;
int n,m,H,x,tot,f[N],g[N],h[N];
struct E{ int u,v,w; }e[N<<];
bool operator <(const E &a,const E &b){ return a.w<b.w; }
int get(int x){ return (f[x]==x) ? x : f[x]=get(f[x]); } int main(){
freopen("bzoj5101.in","r",stdin);
freopen("bzoj5101.out","w",stdout);
scanf("%d%d%d",&n,&m,&H);
rep(i,,n) rep(j,,m-) scanf("%d",&x),e[++tot]=(E){F(i,j),F(i,j+),x};
rep(i,,n-) rep(j,,m) scanf("%d",&x),e[++tot]=(E){F(i,j),F(i+,j),x};
sort(e+,e+tot+);
rep(i,,n*m) f[i]=i,g[i]=,h[i]=;
rep(i,,tot){
int x=get(e[i].u),y=get(e[i].v);
if (x==y) continue;
g[y]=1ll*(g[x]+e[i].w-h[x])*(g[y]+e[i].w-h[y])%mod;
h[y]=e[i].w; f[x]=y;
}
printf("%d\n",(g[get()]+H-h[get()])%mod);
return ;
}

[BZOJ5102][POI2018]Prawnicy(堆)

区间排序后堆维护前k大,据说其它数据结构都被卡常了。

 #include<queue>
#include<cstdio>
#include<algorithm>
#define rep(i,l,r) for (int i=(l); i<=(r); i++)
typedef long long ll;
using namespace std; const int N=;
int n,m,ans,u,v;
struct P{ int l,r,id; }a[N];
bool operator <(const P &a,const P &b){ return a.r<b.r; }
priority_queue<int>Q; int main(){
freopen("bzoj5102.in","r",stdin);
freopen("bzoj5102.out","w",stdout);
scanf("%d%d",&n,&m);
rep(i,,n) scanf("%d%d",&a[i].l,&a[i].r),a[i].id=i;
sort(a+,a+n+);
for (int i=n; i; i--){
Q.push(a[i].l);
if (n-i+>=m){
int x=Q.top(); Q.pop();
if (a[i].r-x>ans) u=x,v=i,ans=a[i].r-x;
}
}
printf("%d\n",ans);
rep(i,v,n) if (a[i].l<=u) printf("%d ",a[i].id);
return ;
}

POI2018的更多相关文章

  1. bzoj5100 [POI2018]Plan metra 构造

    5100: [POI2018]Plan metra Time Limit: 40 Sec  Memory Limit: 128 MBSec  Special JudgeSubmit: 189  Sol ...

  2. [POI2018]Pionek

    [POI2018]Pionek 题目大意: 在无限大的二维平面的原点放置着一个棋子.你有\(n(n\le2\times10^5)\)条可用的移动指令,每条指令可以用一个二维整数向量表示.请你选取若干条 ...

  3. bzoj千题计划249:bzoj5100: [POI2018]Plan metra

    http://www.lydsy.com/JudgeOnline/problem.php?id=5100 1.找到d1[i]+dn[i] 最小的点,作为1到n链上的点 2.令链长为D,若abs(d1[ ...

  4. 【BZOJ5102】[POI2018]Prawnicy 堆

    [BZOJ5102][POI2018]Prawnicy Description 定义一个区间(l,r)的长度为r-l,空区间的长度为0. 给定数轴上n个区间,请选择其中恰好k个区间,使得交集的长度最大 ...

  5. 【BZOJ5099】[POI2018]Pionek 几何+双指针

    [BZOJ5099][POI2018]Pionek Description 在无限大的二维平面的原点(0,0)放置着一个棋子.你有n条可用的移动指令,每条指令可以用一个二维整数向量表示.每条指令最多只 ...

  6. 【BZOJ5100】[POI2018]Plan metra 构造

    [BZOJ5100][POI2018]Plan metra Description 有一棵n个点的无根树,每条边有一个正整数权值,表示长度,定义两点距离为在树上的最短路径的长度. 已知2到n-1每个点 ...

  7. 【BZOJ5101】[POI2018]Powód 并查集

    [BZOJ5101][POI2018]Powód Description 在地面上有一个水箱,它的俯视图被划分成了n行m列个方格,相邻两个方格之间有一堵厚度可以忽略不计的墙,水箱与外界之间有一堵高度无 ...

  8. bzoj 5099 [POI2018]Pionek 计算几何 极角排序

    [POI2018]Pionek Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 269  Solved: 80[Submit][Status][Disc ...

  9. [bzoj5101][POI2018]Powódź_并查集

    Powódź bzoj-5101 POI-2018 题目大意:在地面上有一个水箱,它的俯视图被划分成了$n$行$m$列个方格,相邻两个方格之间有一堵厚度可以忽略不计的墙,水箱与外界之间有一堵高度无穷大 ...

随机推荐

  1. Spring4笔记7--AspectJ 对 AOP 的实现

    AspectJ 对 AOP 的实现: 对于 AOP 这种编程思想,很多框架都进行了实现.Spring 就是其中之一,可以完成面向切面编程.然而,AspectJ 也实现了 AOP 的功能,且其实现方式更 ...

  2. 【codeforces】【比赛题解】#920 Educational CF Round 37

    [A]浇花 题意: 一个线段上每个整点都有花,有的点有自动浇花的喷水器,有问几秒能浇完所有的花. 题解: 大模拟 #include<cstdio> #include<cstring& ...

  3. CentOS6.6中安装telnet

    一.查看本机是否安装telnet rpm -qa | grep telnet 如果什么都不显示.说明你没有安装telnet 二.开始安装 yum install xinetd yum install ...

  4. 项目中遇到的问题:Gradle传递性依赖冲突

    问题描述: 在调用别人接口时,由于他们接口做了拦截处理在使用RestTemplate调用时必须要使用@Qualifier("他们封装好的类"),需要导入jar包 gradle方式导 ...

  5. poj1102

    模拟 #include <iostream> #include <string> using namespace std; ][][] = { { ' ', '-', ' ', ...

  6. JS可以监控手机的返回键吗?

    html5的话 一进页面就pushState,然后监控onpopstate不过好像没有办法知道是前进还是后退我的奇淫巧计是,一个数字变量,pushState一个锚,锚是这个数字,前进一个页面数字+1, ...

  7. 链家2018春招C/C++开发实习生在线考试编程题

    题目一 题解:该题目意思就是让你输入n组数据,然后求并集,利用STL容器set集合的特性:元素不重复存储,我们可以很轻易得出答案 #include <iostream> #include ...

  8. java jps命令使用解析

    在linux环境下显示一个进程的信息大家可能一直都在使用ps命令,比如用以下命令来显示当前系统执行的java进程: ps -ef | grep java 针对java的进程,jdk1.5以后提供了一个 ...

  9. 如何在VS2013创建WebService并在IIS中发布

    第一步:打开VS2013,选择文件->新建->项目. 第二步:选择[ASP.net 空web应用程序],将其命名为自己想的工程名称. 第三步:然后右键点击工程,添加->web服务.然 ...

  10. maven:missing artifact jdk.tools:jar:1.7

    http://stackoverflow.com/questions/11118070/buiding-hadoop-with-eclipse-maven-missing-artifact-jdk-t ...