zoukankan      html  css  js  c++  java
  • 【BZOJ 4558】 4558: [JLoi2016]方 (计数、容斥原理)

       未经博主同意不能转载

    4558: [JLoi2016]方

    Time Limit: 20 Sec  Memory Limit: 256 MB
    Submit: 362  Solved: 162

    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) 取模之后的值

    Sample Input

    2 2 4
    1 0
    1 2
    0 1
    2 1

    Sample Output

    1

    HINT

    Source

    【分析】

      终于A了,昨晚做了一晚上了。

      容斥真的很容易搞错。

      注意这题的正方形可以是【斜着的】

      于是怎么算呢?首先假设没有删除点,答案=总方案-至少有一个删除点的方案数

      1、总方案怎么算?

      你会觉得斜着的正方形很难算,但是每个正方形都有外接正方形,对于边长为a的正的正方形来说,他是a个正方形的外接正方形,于是枚举外接正方形边长即可。

      2、至少有一个删除点的方案数?

      这个的答案=SIGEMA(枚举每一个删除点然后算以他为顶点的正方形的方案数)-至少两个删除点的方案数+至少三个删除点的方案数-4个删除点的方案数

      3、枚举每一个删除点然后算以他为顶点的正方形的方案数?

      【这里的正方形还是有斜的】,但是!这个删除点一定在这个以它的顶点的正方形的外接正方形的某条边长上(或角上),所以只需计算正的正方形且某条边长覆盖这个点的方案数。

      【这个貌似是最难算的,因为看不懂题解于是我自己Y了一个计算方法】:

      

      分成四部分来做(还有一部分是删除点是顶点,这个比较容易就不说了,所以现在计算的答案是不包含删除点是顶点的,只包含删除点在四边上)

      四部分是左半、右半、上半、下半,图示蓝框是左半。

      左半部分就是说删除点在正方形右边长上。

      【还是把正方形画一画吧】

      

       根据蓝框的横宽可以知道正方形边长的一个极限,于是可以把图形降维,变成一维上的问题。

      

      问题变成:0~n数轴上有一点x,问你长度为[1,r]且为整数的,并覆盖了x点的区间有多少个(端点覆盖不算覆盖)。

      之前做的一道概率题已让我对此问题有所体会,这里再用一个小容斥最好算了:总区间-不覆盖x的区间。

      即(0~n)中长度[1,r]的整区间-(0~x)中长度[1,r]的整区间-(x~n)中长度[1,r]的整区间。

      这样就求出来了这个答案,记为a1。【记得要把删除点为正方形角的加上

      4、问题还没有结束,对于2个、3个、4个删除点,我们枚举两个删除点可以确定这个正方形(有三种形态)

      然后这三个答案一起算。记为a2,a3,a4

      最后答案=总-a1+a2-a3/3+a4/6【因为你枚举两个点,所以同一个正方形有可能算多遍,于是需要除,a3加的时候是加TMP,具体看代码】

      【说一下这个为什么?我其实我己都没想清楚这个,也是看别人这样写的】

      首先a2计算是正确的,你假设两个未删除点,枚举的也是两个未删除点。

      a3的话,你应该枚举三个未删除的点,但是你只枚举两个,第三个是计算出来的。

      假设真的只有三个点,对于同一个正方形,你枚举了$C_{3}^{2}$次,实际上应该枚举$C_{3}^{3}$次,是3倍,所以要除以3。

      不过至少是3的话还有可能是4,你枚举了$C_{4}^{2}$次,实际上应该枚举$C_{4}^{3}$次,所以要*2/3

      a4的话,你枚举了$C_{4}^{2}$次,实际是$C_{4}^{4}$次,所以要除以6。

      确定正方形的三种形态的话,我也是自己Y了一下,经验就是用向量!,向量是解决几何问题的最好工具!【不懂可以看看代码怎么打

      

      【这篇写得够认真了吧??】

      1 #include<cstdio>
      2 #include<cstdlib>
      3 #include<cstring>
      4 #include<iostream>
      5 #include<algorithm>
      6 using namespace std;
      7 #define Maxn 2010
      8 #define Maxm 1000010
      9 #define Mod 100000007
     10 #define LL long long
     11 
     12 int mymin(int x,int y) {return x<y?x:y;}
     13 
     14 int nx[Maxn],ny[Maxn],rx[Maxm],ry[Maxm];
     15 bool vis[Maxn][Maxn];
     16 
     17 struct node
     18 {
     19     int x,id;
     20 }t[Maxn];
     21 bool cmp(node x,node y) {return x.x<y.x;}
     22 
     23 int check(int x,int y)
     24 {
     25     int ret=0;
     26     if(vis[nx[x]][ny[x]]) ret++;
     27     if(vis[nx[y]][ny[y]]) ret++;
     28     if(vis[nx[x]][ny[y]]) ret++;
     29     if(vis[nx[y]][ny[x]]) ret++;
     30     return ret;
     31 }
     32 
     33 int n,m,k;
     34 void init()
     35 {
     36     for(int i=1;i<=k;i++) t[i].x=nx[i],t[i].id=i;sort(t+1,t+1+k,cmp);
     37     int p=1;rx[t[1].x]=p;
     38     for(int i=2;i<=k;i++)
     39     {
     40         if(t[i].x!=t[i-1].x) p++,rx[t[i].x]=p;
     41         // nx[t[i].id]=p;
     42     }
     43     for(int i=1;i<=k;i++) t[i].x=ny[i],t[i].id=i;sort(t+1,t+1+k,cmp);
     44     p=1;ry[t[1].x]=p;
     45     for(int i=2;i<=k;i++)
     46     {
     47         if(t[i].x!=t[i-1].x) p++,ry[t[i].x]=p;
     48         // ny[t[i].id]=p;
     49     }
     50     memset(vis,0,sizeof(vis));
     51     for(int i=1;i<=k;i++) vis[rx[nx[i]]][ry[ny[i]]]=1;
     52 }
     53 
     54 int cal(int x,int y)
     55 {
     56     if(!x) return 0;
     57     y=min(y,x);
     58     int l=x-y+1,r=x;
     59     return (1LL*(l+r)*(r-l+1)/2)%Mod;
     60 }
     61 
     62 int a1,a2,a3,a4;
     63 void get_ans(int x1,int y1,int x2,int y2)
     64 {
     65     if(x1<0||x1>n||y1<0||y1>m||x2<0||x2>n||y2<0||y2>m) return;
     66     int tmp=0;
     67     if(rx[x1]&&ry[y1]&&vis[rx[x1]][ry[y1]]) tmp++;
     68     if(rx[x2]&&ry[y2]&&vis[rx[x2]][ry[y2]]) tmp++;
     69     a2++;if(tmp) a3+=tmp;if(tmp>1) a4++;
     70 }
     71 
     72 void solve(int x1,int y1,int x2,int y2)
     73 {
     74     int dx=x1-x2,dy=y1-y2;
     75     get_ans(x1+dy,y1-dx,x2+dy,y2-dx);
     76     get_ans(x1-dy,y1+dx,x2-dy,y2+dx);
     77     if(abs(dx+dy)&1) return;
     78     get_ans((x1+x2+dy)/2,(y1+y2-dx)/2,(x1+x2-dy)/2,(y1+y2+dx)/2);
     79 }
     80 
     81 int main()
     82 {
     83     scanf("%d%d%d",&n,&m,&k);
     84     int ans=0;
     85     for(int i=1;i<=n&&i<=m;i++) ans=(ans+1LL*(n+1-i)*(m+1-i)%Mod*i%Mod)%Mod;
     86     for(int i=1;i<=k;i++)
     87     {
     88         scanf("%d%d",&nx[i],&ny[i]);
     89     }
     90     for(int i=1;i<=k;i++)
     91     {
     92         a1+=cal(n,ny[i])-cal(nx[i],ny[i])-cal(n-nx[i],ny[i]);a1%=Mod;
     93         a1+=cal(n,m-ny[i])-cal(nx[i],m-ny[i])-cal(n-nx[i],m-ny[i]);a1%=Mod;
     94         a1+=cal(m,nx[i])-cal(ny[i],nx[i])-cal(m-ny[i],nx[i]);a1%=Mod;
     95         a1+=cal(m,n-nx[i])-cal(ny[i],n-nx[i])-cal(m-ny[i],n-nx[i]);a1%=Mod;
     96         a1+=min(nx[i],ny[i])+min(n-nx[i],m-ny[i])+min(n-nx[i],ny[i])+min(nx[i],m-ny[i]);a1%=Mod;
     97     }
     98     init();
     99     for(int i=1;i<=k;i++)
    100      for(int j=i+1;j<=k;j++)
    101      {
    102         solve(nx[i],ny[i],nx[j],ny[j]);
    103      }
    104     // printf("%d
    ",a3);
    105     printf("%d
    ",((ans-a1+a2-a3/3+a4/6)%Mod+Mod)%Mod); 
    106     return 0;
    107 }
    View Code

    2017-04-21 09:25:43

  • 相关阅读:
    JavaScript条件判断和循环
    JavaScript数据类型详解
    Dockerfile使用
    让ie8、ie9支持媒体查询
    事件穿透
    判断是苹果手机还是安卓手机
    ES6中字符串的扩展
    ES6数组的扩展运算符
    let和const
    ES6中函数的扩展
  • 原文地址:https://www.cnblogs.com/Konjakmoyu/p/6741896.html
Copyright © 2011-2022 走看看