Description

上帝说,不要圆,要方,于是便有了这道题。由于我们应该方,而且最好能够尽量方,所以上帝派我们来找正方形
上帝把我们派到了一个有N行M列的方格图上,图上一共有(N+1)×(M+1)个格点,我们需要做的就是找出这些格点形
成了多少个正方形(换句话说,正方形的四个顶点都是格点)。但是这个问题对于我们来说太难了,因为点数太多
了,所以上帝删掉了这(N+1)×(M+1)中的K个点。既然点变少了,问题也就变简单了,那么这个时候这些格点组成
了多少个正方形呢?

Input

第一行三个整数 N, M, K, 代表棋盘的行数、 列数和不能选取的顶点个数。 保证 N, M >= 1, K <=(N + 1) ×
(M + 1)。约定每行的格点从上到下依次用整数 0 到 N 编号,每列的格点依次用 0到 M 编号。接下来 K 行,每
行两个整数 x,y 代表第 x 行第 y 列的格点被删掉了。保证 0 <=x <=N<=10^6, 0 <=y<=M<=10^6,K<=2*1000且不
会出现重复的格点。

Output

仅一行一个正整数, 代表正方形个数对 100000007( 10^8 + 7) 取模之后的值

容斥,不考虑删点的方案数可以O(min(n,m))枚举正方形在水平方向投影的长度算出

恰好删去2~4点的方案数可以O(k2)枚举其中的两个点,再看正方形另外两点是否也被删去

还有过定点的正方形数,可以列出8条不等式,求半平面交,然后因为一些特殊的性质,用pick定理可以算出半平面交内整点数(似乎有比这简洁的多的方法)

然后加加减减就得到答案了

#include<cstdio>
#include<algorithm>
int s[][];
int n,m,k;
typedef long long i64;
const int P=1e8+;
int min(int a,int b){return a<b?a:b;}
int max(int a,int b){return a>b?a:b;}
struct pos{int x,y;}ps[];
struct hp{int a,b,c;}hs[],ls[];
pos operator&(hp u,hp v){
int r=u.a*v.b-u.b*v.a;
if((u.c*v.b-u.b*v.c)%r||(u.a*v.c-u.c*v.a)%r)putchar('*');
return (pos){(u.c*v.b-u.b*v.c)/r,(u.a*v.c-u.c*v.a)/r};
}
bool operator==(pos a,pos b){return a.x==b.x&&a.y==b.y;}
i64 operator*(pos a,pos b){return i64(a.x)*b.y-i64(a.y)*b.x;}
pos operator-(pos a,pos b){return (pos){a.x-b.x,a.y-b.y};}
pos operator+(pos a,pos b){return (pos){a.x+b.x,a.y+b.y};}
bool in(pos a,hp b){return a.x*b.a+a.y*b.b<=b.c;}
bool chk(hp a,hp b,hp c){return !in(a&b,c);}
int gcd(int a,int b){return b?gcd(b,a%b):a;}
int abs(int x){return x>?x:-x;}
int count(pos w){return gcd(abs(w.x),abs(w.y));}
i64 cal(int x,int y){
int L=-x,R=n-x,U=-y,D=m-y;
int p=;
#define _(a,b,c) hs[p++]=(hp){a,b,c}
_(,,min(D,-L));
_(,,D);
_(,,min(R,D));
_(,-,R);
_(,-,-max(U,-R));
_(-,-,-U);
_(-,,-max(L,U));
_(-,,-L);
hs[p]=hs[];
int l=,r=;
for(int i=;i<p;++i){
while(r-l>=&&chk(ls[r-],ls[r-],hs[i]))--r;
while(r-l>=&&chk(ls[l+],ls[l],hs[i]))++l;
ls[r++]=hs[i];
}
while(r-l>=&&chk(ls[r-],ls[r-],ls[l]))--r;
ls[r]=ls[l];
for(int i=l;i<r;++i)ps[i]=ls[i+]&ls[i];
r=std::unique(ps+l,ps+r)-ps-l;
ps[r]=ps[l];
if(r-l==)return count(ps[l+]-ps[l]);
i64 ans=;
for(int i=l;i<r;++i){
ans+=count(ps[i+]-ps[i]);
ans+=ps[i+]*ps[i];
}
return ans/;
}
pos xs[],ht[];
bool hd[];
void ins(pos x){
int w=(x.x*+x.y*)%;
while(hd[w])w=(w+)%;
hd[w]=;ht[w]=x;
}
bool operator<(pos a,pos b){return a.x!=b.x?a.x<b.x:a.y<b.y;}
i64 d0,d1,d2,d3,d4;
bool exist(pos x){
int w=(x.x*+x.y*)%;
while(hd[w]){
if(ht[w]==x)return ;
w=(w+)%;
}
return ;
}
bool in(pos a){return a.x>=&&a.x<=n&&a.y>=&&a.y<=m;}
int main(){
scanf("%d%d%d",&n,&m,&k);
for(int i=;i<k;++i)scanf("%d%d",&xs[i].x,&xs[i].y),ins(xs[i]);
int s=min(n,m);
for(int i=;i<=s;++i)d0=(d0+i*i64(n+-i)*(m+-i))%P;
for(int i=;i<k;++i){
d1=(d1+cal(xs[i].x,xs[i].y))%P;
for(int j=i+;j<k;++j){
pos d=xs[j]-xs[i];
if(~d.x+d.y&){
pos a=(pos){d.x+d.y>>,-d.x+d.y>>};
pos p1=xs[i]+a,p2=xs[j]-a;
if(in(p1)&&in(p2)){
int e1=exist(p1);
int e2=exist(p2);
d4+=e1&e2;
d3+=e1^e2;
d2+=~(e1|e2)&;
}
}
d=(pos){d.y,-d.x};
pos p1=xs[i]+d,p2=xs[i]-d,p3=xs[j]+d,p4=xs[j]-d;
if(in(p1)&&in(p3)){
int e1=exist(p1);
int e3=exist(p3);
d4+=(e1&e3);
d3+=(e1^e3);
d2+=(~(e1|e3)&);
}
if(in(p2)&&in(p4)){
int e2=exist(p2);
int e4=exist(p4);
d4+=(e2&e4);
d3+=(e2^e4);
d2+=(~(e2|e4)&);
}
}
}
d4/=;d3/=;
d1-=d2+d3*+d4*;
d0-=d1;
printf("%lld",(d0+P)%P);
return ;
}

bzoj4558: [JLoi2016]方的更多相关文章

  1. bzoj4558[JLoi2016]方 容斥+count

    4558: [JLoi2016]方 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 452  Solved: 205[Submit][Status][D ...

  2. [BZOJ4558]:[JLoi2016]方(容斥+模拟)

    题目传送门 题目描述 上帝说,不要圆,要方,于是便有了这道题.由于我们应该方,而且最好能够尽量方,所以上帝派我们来找正方形上帝把我们派到了一个有N行M列的方格图上,图上一共有$(N+1)\times ...

  3. bzoj千题计划281:bzoj4558: [JLoi2016]方

    http://www.lydsy.com/JudgeOnline/problem.php?id=4558 容斥原理 全部的正方形-至少有一个点被删掉的+至少有两个点被删掉的-至少有3个点被删掉的+至少 ...

  4. 4558: [JLoi2016]方

    4558: [JLoi2016]方 https://lydsy.com/JudgeOnline/problem.php?id=4558 分析: 容斥原理+各种神奇的计数. 如果没有被删除了的点的话,直 ...

  5. 【BZOJ 4558】 4558: [JLoi2016]方 (计数、容斥原理)

    未经博主同意不能转载 4558: [JLoi2016]方 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 362  Solved: 162 Descri ...

  6. JLOI2016 方

    bzoj4558 真是一道非常excited的题目啊-JLOI有毒 题目大意:给一个(N+1)*(M+1)的网格图,格点坐标为(0~N,0~M),现在挖去了K个点,求剩下多少个正方形(需要注意的是正方 ...

  7. BZOJ.4558.[JLOI2016]方(计数 容斥)

    BZOJ 洛谷 图基本来自这儿. 看到这种计数问题考虑容斥.\(Ans=\) 没有限制的正方形个数 - 以\(i\)为顶点的正方形个数 + 以\(i,j\)为顶点的正方形个数 - 以\(i,j,k\) ...

  8. 【BZOJ】4558: [JLoi2016]方

    [题意]给定有(n+1)*(m+1)个点的网格图,其中指定k个点不合法,求合法的正方形个数(四顶点合法). [算法]计数 [题解]斜着的正方形很麻烦,所以考虑每个斜正方形其外一定有正的外接正方形. 也 ...

  9. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

随机推荐

  1. tfn2k工具使用介绍

    主要介绍tfn2k(Linux),因为它最著名嘛!主要分为使用说明 然后在说安装所以有点长 (注意:有的人拿VPS来做DDOS·问题是有的人发现了会关闭你VPS的·除非你认识他或者你自己有服务器而不是 ...

  2. MAC OSX Docker安装教程2018

    1.首先打开Docker商店 Docker商店地址:https://store.docker.com/editions/community/docker-ce-desktop-mac 2.打开后界面如 ...

  3. nodejs项目的model操作mongo

    想想以前学习hibernate的时候,学习各种表和表之间的映射关系等一对多,多对一,多对多,后来到了工作中,勇哥告诉我, 那时在学习的时候,公司中都直接用外键关联. 这里我们学习下,如何在Nodejs ...

  4. kue

    尝试着看英文的文档,你发现他其实并不难. https://www.npmjs.com/package/kue

  5. C++getline使用

    C++getline使用 一.心得 getline(cin,s);多去看函数的使用默认说明 二.使用 getline(istream &in, string &s) 从输入流读入一行到 ...

  6. 安装 android4.0 到 vmware的注意点

    我发现无论如何,wmware上是出不来声音的. 我放弃 android-x86-4.3-20130725.iso 在 声音上有问题 现在安装这个版本: android-x86-4.0-RC2-eeep ...

  7. 快速理解 FastCGI、PHP-CGI、PHP-FPM

    你(PHP)去和泰国人(web服务器,如 Apache.Nginx)谈生意 你说中文(PHP代码),他说泰语(C代码),互相听不懂,怎么办?那就都把各自说的话转换成英语(FastCGI 协议)吧. 怎 ...

  8. java并发编程:线程安全管理类--原子操作类--AtomicIntegerFieldUpdater<T>

    1.类 AtomicIntegerFieldUpdater<T> public abstract class AtomicIntegerFieldUpdater<T> exte ...

  9. Sublime Text 编辑器的设定

    # Sublime Text 编辑器的设定 > gloal setting ```{    "caret_style": "smooth",    &qu ...

  10. 苹果iPhone 5C和5S发布后,消费者如何选择?

    9月11日凌晨苹果新品发布会,笔者的朋友圈居然没有看直播的,真果粉太少了.笔者来阐述一些容易忽略的东西. iPhone5C和5S与5有什么不一样? 新品iPhone 5S 外观与iPhone5 相似度 ...