zoukankan      html  css  js  c++  java
  • [Sdoi2013]spring

    [Sdoi2013]spring

    题目

    INPUT

    OUTPUT

    SAMPLE

    INPUT

    3 3

    1 2 3 4 5 6

    1 2 3 0 0 0

    0 0 0 4 5 6

    OUTPUT

    2

    解题报告

    $hash$加容斥

    我们一看到恰有$K$个相等,很容易就能想到容斥原理,所以,我们需要枚举每种对应相等的情况,也就是枚举$2^{6}$种对应相等的情况,然后$C_{num}^{2}$求出总对数

    问题在于如何处理对应相等的情况,$O(n^{2})$的暴力是很容易想出来的,然而在并没有的数据范围中,$n$是$10^{5}$级别的,意味着$O(n^{2})$的暴力可以说再见了,那么我们就很容易想到$hash$,用对应相等的$hash$值来判断对应位置的相等,就可以做到$O(n)$判等了

    接着就是最重要的一部分了:容斥原理

    我们考虑,我们只取出了$a$位对应位来判断是否相等,但是显然$a+1$位对应相等的一对城市也会被包含进去,所以我们要应用容斥原理,从$k$个对应相等开始枚举$k+1$位对应相等,直到$n$个对应相等,奇偶性与$k$相同的加上,不同的减去

    (不懂原理的自行百度容斥原理)

    重要的是,我们不能简单的加减总对数,我们考虑,从$k+a$位对应相等的一对城市中取出$k$位对应相等,共有$C_{k+a}^{k}$种取法,(因为我们是按位取,按位比较的),所以我们还要对应的乘上他们的组合数

     1 #include <iostream>
     2 #include <cstdio>
     3 #include <cstring>
     4 #include <algorithm>
     5 #include <vector>
     6 using namespace std;
     7 #define mod 1000007
     8 #define P 2333333333333LL
     9 int read() {
    10     int s=0,f=1;
    11     char ch=getchar();
    12     for( ; ch<'0'||ch>'9'; f=(ch=='-')?(-1):(f),ch=getchar()) ;
    13     for( ; ch>='0'&&ch<='9'; s=(s<<1)+(s<<3)+(ch^48),ch=getchar()) ;
    14     return s*f;
    15 }
    16 int n,k,sta[7],top,f[100005][7],fac[7],C[7][7];
    17 unsigned long long num[1000010],inv[1000010],tmp,biao[100005];
    18 void clear() {
    19     top=tmp=0;
    20     memset(biao,0,sizeof(biao));
    21     memset(num,0,sizeof(num));
    22     memset(inv,0,sizeof(inv));
    23 }
    24 void init() {
    25  
    26     fac[0]=fac[1]=1;
    27     for(int i=2; i<=6; ++i) {
    28         fac[i]=fac[i-1]*i;
    29     }
    30     for(int i=0; i<=6; ++i) {
    31         for(int j=0; j<=i; ++j) {
    32             C[i][j]=fac[i]/(fac[j]*fac[i-j]);
    33         }
    34     }
    35 }
    36 void get(int x) {
    37     clear();
    38     for(int i=1; i<=6; ++i) {
    39         if((1<<(i-1))&x) {
    40             sta[++top]=i;
    41         }
    42     }
    43     for(int i=1; i<=n; ++i) {
    44         for(int j=1; j<=top; ++j) {
    45             biao[i]=biao[i]*P+f[i][sta[j]];
    46         }
    47     }
    48     for(int i=1; i<=n; ++i) {
    49         int ss=biao[i]%mod;
    50         while(inv[ss]!=biao[i]) {
    51             if(!num[ss]) {
    52                 break;
    53             }
    54             ++ss;
    55         }
    56         inv[ss]=biao[i];
    57         tmp+=num[ss];
    58         ++num[ss];
    59     }
    60 }
    61 int main() {
    62     n=read(),k=read();
    63     for(int i=1; i<=n; ++i) {
    64         for(int j=1; j<=6; ++j) {
    65             f[i][j]=read();
    66         }
    67     }
    68     init();
    69     long long ans=0;
    70     for(int i=0; i<(1<<6); ++i) {
    71         get(i);
    72         if(top>=k) {
    73             if((top-k)&1) {
    74                 ans-=tmp*C[top][k];
    75             }
    76             if(!((top-k)&1)) {
    77                 ans+=tmp*C[top][k];
    78             }
    79         }
    80     }
    81     cout<<ans;
    82     return 0;
    83 }
    View Code

    好久不写题解,都快不会写了= =

  • 相关阅读:
    DNS服务器详解
    numpy学习
    test_pandas
    1.爬虫基本介绍
    数据分析介绍及软件使用 01
    3.解析库beautifulsoup
    jQuery UI vs EasyUI
    "file:///" file 协议
    Display:Block
    前端响应式设计中@media等的相关运用
  • 原文地址:https://www.cnblogs.com/hzoi-mafia/p/7495741.html
Copyright © 2011-2022 走看看