http://www.lydsy.com/JudgeOnline/problem.php?id=4541 (题目链接)

题意

  给出一个平面图,若干询问,每次询问一个凸多边形内小多边形面积的平方和与面积的和的比值。

Solution

  江哥的神题,右转题解→_→:http://blog.csdn.net/philipsweng/article/details/51201092

细节

  注意数组大小

代码

// bzoj4541
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<vector>
#include<cstdio>
#include<cmath>
#define LL long long
#define inf (1ll<<60)
#define HAS 40007
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std;
inline int gi() {
int x=0,f=1;char ch=getchar();
while (ch<'0' || ch>'9') {if (ch=='-') f=-1;ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0';ch=getchar();}
return x*f;
} const int maxn=200010,maxm=600010;
int head[maxm],last[maxn],next[maxm<<1],par[maxm],fa[maxm],N,n,m,q,s,cnt=1,ccc;
LL area[maxm],AREA[maxm],sum[maxm],SUM[maxm],P,Q;
struct point {int x,y;}a[maxn];
struct chain {LL w;int id,next;}line[maxm<<1];
struct edge {int u,v,next,id;}e[maxm<<1],d[maxm<<1];
vector<pair<double,int> >V[maxn]; inline LL gcd(LL a,LL b) {return b==0 ? a : gcd(b,a%b);}
inline int find(int x) {return fa[x]==x ? x : fa[x]=find(fa[x]);}
inline int cross(point a,point b) {return a.x*b.y-b.x*a.y;} inline void link(edge *e,int u,int v) {
e[++cnt]=(edge){u,v,head[u],0};head[u]=cnt;
e[++cnt]=(edge){v,u,head[v],0};head[v]=cnt;
}
inline void build(int x) {
for (int i=x;!d[i].id;i=next[i]) {
d[i].id=N;
int tmp=-cross(a[d[i].u],a[d[i].v]);
area[N]+=tmp;
}
}
inline void dfs(int x) {
sum[x]=area[x];SUM[x]=AREA[x];
for (int i=head[x];i;i=e[i].next) if (e[i].v!=par[x]) {
par[e[i].v]=x;
dfs(e[i].v);
sum[x]+=sum[e[i].v];SUM[x]+=SUM[e[i].v];
}
}
inline int edgeid(int u,int v) {
LL tmp=(LL)u*n+(LL)v-1;
int id=tmp%HAS;
for (int i=last[id];i;i=line[i].next)
if (line[i].w==tmp) return line[i].id;
}
int main() {
n=gi(),m=gi(),q=gi(),s=1;
for (int i=1;i<=n;++i) {
a[i].x=gi(),a[i].y=gi();
if (a[i].x<a[s].x || (a[i].x==a[s].x && a[i].y<a[s].y)) s=i;
}
for (int u,v,i=1;i<=m;++i) u=gi(),v=gi(),link(d,u,v);
for (int i=1;i<=n;++i) { //以i为起点的边极角排序
for (int j=head[i];j;j=d[j].next)
V[i].push_back(make_pair(atan2(a[d[j].v].y-a[i].y,a[d[j].v].x-a[i].x),j));
sort(V[i].begin(),V[i].end());
for (int j=0,k=V[i].size();j<k;++j) next[V[i][j].second^1]=V[i][(j+1)%k].second; //next[x]表示走完边x后该走的边
}
N=1;build(V[s][0].second);area[1]=AREA[1]=0; //先抠无穷域,面积赋为0
for (int i=2;i<=cnt;++i) if (!d[i].id) {++N;build(i);AREA[N]=area[N]*area[N],area[N]<<=1;}
for (int i=2;i<=cnt;++i) { //hash
LL tmp=(LL)d[i].u*n+(LL)d[i].v-1;
int id=tmp%HAS;
line[++ccc]=(chain){tmp,i,last[id]};last[id]=ccc;
}
memset(head,0,sizeof(head));
for (int i=1;i<=N;++i) fa[i]=i;
int tmp=cnt;cnt=0;
for (int i=2;i<=tmp;i+=2) { //随意构一棵域之间的生成树
int u=d[i].id,v=d[i^1].id;
int r1=find(u),r2=find(v);
if (r1!=r2) fa[r1]=r2,link(e,u,v),d[i].next=d[i^1].next=-1;
}
dfs(1); //dfs统计一下子树权值和以及父亲关系
while (q--) {
LL tmp=P,c=(gi()+tmp)%n+1;
s=(gi()+tmp)%n+1;
P=0,Q=0;
for (int u,v=s,i=1;i<=c;++i) {
u=v,v=i==c ? s : (gi()+tmp)%n+1;
int t=edgeid(u,v);
if (d[t].next!=-1) continue;
int x=d[t].id,y=d[t^1].id;
if (par[x]==y) P-=SUM[x],Q-=sum[x];
else if (par[y]==x) P+=SUM[y],Q+=sum[y];
}
LL G=gcd(P,Q);
P/=G,Q/=G;
printf("%lld %lld\n",P,Q);
}
return 0;
}

【bzoj4541】 Hnoi2016—矿区的更多相关文章

  1. BZOJ4541 HNOI2016矿区(平面图转对偶图)

    考虑先将平面图转化为对偶图.具体地,将无向边拆成两条有向边.每次考虑找到包围一个区域的所有边.对当前考虑的边,找到该边的反向边在该边终点的出边集中,按极角序排序的后继,这条后继边也是包围该区域的边.这 ...

  2. [BZOJ4541][HNOI2016]矿区(平面图转对偶图)

    https://www.cnblogs.com/ljh2000-jump/p/6423399.html #include<cmath> #include<vector> #in ...

  3. BZOJ4541 [Hnoi2016]矿区

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/ ...

  4. [HNOI2016]矿区

    [HNOI2016]矿区 平面图转对偶图 方法: 1.分成正反两个单向边,每个边属于一个面 2.每个点按照极角序sort出边 3.枚举每一个边,这个边的nxt就是反边的前一个(这样找到的是面的边逆时针 ...

  5. 【LG3249】[HNOI2016]矿区

    [LG3249][HNOI2016]矿区 题面 洛谷 题解 先平面图转对偶图, 建好了对偶图之后随意拿出一个生成树,以无边界的范围为根. 无边界的范围很好求,用叉积算出有向面积时,算出来是负数的就是无 ...

  6. BZOJ 4541: [Hnoi2016]矿区 平面图转对偶图+DFS树

    4541: [Hnoi2016]矿区 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 433  Solved: 182[Submit][Status][ ...

  7. 4541: [Hnoi2016]矿区

    学习了一下平面图剖分的姿势,orz cbh 每次只要随便选择一条边,然后不停尽量向左转就行 #include <bits/stdc++.h> #define N 1300000 #defi ...

  8. ●BZOJ 4541 [Hnoi2016]矿区

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4541 题解: 平面图的对偶图,dfs树 平面图的对偶图的求法: 把所有双向边拆为两条互为反向 ...

  9. bzoj 4541: [Hnoi2016]矿区【平面图转对偶图+生成树】

    首先平面图转对偶图,大概思路是每条边存正反,每个点存出边按极角排序,然后找每条边在它到达点的出边中极角排序的下一个,这样一定是这条边所属最小多边形的临边,然后根据next边找出所有多边形,用三角剖分计 ...

随机推荐

  1. 我是如何将页面加载时间从6S降到2S的?

    写在前面 生活在信息爆炸的今天,我们每天不得不面对和过滤海量的信息--无疑是焦躁和浮动的,这就意味着用户对你站点投入的时间可能是及其吝啬的(当然有一些刚需站点除外). 如何给用户提供迅速的响应就显得十 ...

  2. Controller - 压力机的设置 - 界面图表分析

    一. Controller- 压力机界面的一下设置讲解   2种测试场景的设计和压测策略 二. Controller- 压力机界面的图表分析

  3. 高可用Kubernetes集群-16. ansible快速部署

    说明 本文档指导采用二进制包的方式快速部署高可用kubernetes集群. 脚本托管:k8s-ansible(持续更新) 参考:高可用kubernetes集群 组件版本 组件 版本 备注 centos ...

  4. 【Docker】第三篇 Docker容器管理

    一.Docker容器概述: 简单理解容器是镜像的一个实例. 镜像是静态的只读文件,而容器的运行需要可写文件层. 二.创建容器 [root@web130 ~]# docker create -it ub ...

  5. NO.6:自学python之路------面向对象、内存持久化

    引言 虽然加速学习了,可是还是感觉进度不够快,担心.还得准备毕业论文,真是焦虑. 正文 面向对象 编程是程序员用特定语法+数据结构+算法组成的代码,告诉计算机如何执行任务的过程.对不同的编程方式的特点 ...

  6. TensorFlow中的卷积函数

    前言 最近尝试看TensorFlow中Slim模块的代码,看的比较郁闷,所以试着写点小的代码,动手验证相关的操作,以增加直观性. 卷积函数 slim模块的conv2d函数,是二维卷积接口,顺着源代码可 ...

  7. FFmpeg简单转码程序--视频剪辑

    学习了雷神的文章,慕斯人分享精神,感其英年而逝,不胜唏嘘.他有分享一个转码程序<最简单的基于FFMPEG的转码程序>其中使用了filter(参考了ffmpeg.c中的流程),他曾说想再编写 ...

  8. 转载----C/C++ 中 const 修饰符用法总结

    感谢原创作者,写的好详细.不忍错过,所以转载过来了... 原文地址: https://www.cnblogs.com/icemoon1987/p/3320326.html 在这篇文章中,我总结了一些C ...

  9. iOS 源代码混淆(初步混淆)

    我们可以用classdump对原程序进行 dump,像上篇文章( Class-dump 安装和使用记录(导出应用的头文件)),我们可以看到所有.h 文件全暴露了(如下图) 点击HWAccount.h后 ...

  10. Beta阶段基于NABCD评论作品

    组名:杨老师粉丝群 组长:乔静玉 组员:吴奕瑶  刘佳瑞  公冶令鑫  杨磊  刘欣  张宇  卢帝同 一.拉格朗日2018--<飞词> 1.1.NABCD分析 N(Need,需求):该小 ...