zoukankan      html  css  js  c++  java
  • [NOI 2015]寿司晚宴

    Description

    题库链接

    给定 (2sim n) 一共 (n-1) 个数字,第一个人选择一些数字,第二个人选择一些数字,要求第一个人选的任意一个数字和第二个人选择的任意一个数字都互质,求方案数。

    (2leq nleq 500)

    Solution

    做的时候想偏了...正解做法比较神...

    我们考虑对一个数质因数分解,容易发现对于 (geq sqrt{500}) 的质因数一定最多一个。

    我们可以拿 (geq sqrt{500}) 的质因数为依据分组。对于 (leq sqrt{500}) 的质因数一共只有 (8) 个,我们拿来状压。

    如果一个数没有 (geq sqrt{500}) 的质因数,那么它单独成一组。

    显然的是同一组的数不能同一个人拿,因为同一组共同拥有一个 (geq sqrt{500}) 的质因数(或没有)。

    所以我们可以按组来做。

    (f_{i,j}) 表示第一个人选 (leq sqrt{500}) 的质因数的状态为 (i) ,第二个人为 (j) 的方案数,显然 (icap j=0)

    那么考虑组内 ( ext{DP}) 。记 (f_{0/1,i,j}) 表示第一/二个人选这一组(或是不选)第一个人选 (leq sqrt{500}) 的质因数的状态为 (i) ,第二个人为 (j) 的方案数。

    首先先将 (f) 分别拷一份给 (g_{0},g_{1})

    组内 ( ext{DP}) 后再将 (f'=g_{0}+g_{1}-f) ,因为都不选的方案算了两次。

    最后统计答案即可。

    Code

    #include <bits/stdc++.h>
    #define ll long long
    #define pii pair<int, int>
    using namespace std;
    const int prime[8] = {2, 3, 5, 7, 11, 13, 17, 19};
    const int N = 505, SZ = (1<<8)+5;
    
    int n, bin[10];
    ll p, f[SZ][SZ], g[2][SZ][SZ];
    pii a[N];
    
    void work() {
        scanf("%d%lld", &n, &p);
        bin[0] = 1;
        for (int i = 1; i <= 8; i++) bin[i] = (bin[i-1]<<1);
        for (int i = 2; i <= n; i++) {
            int x = i;
            for (int j = 0; j < 8; j++) {
                if (x%prime[j] == 0) a[i].second |= bin[j];
                while (x%prime[j] == 0) x /= prime[j];
            }
            a[i].first = x;
        }
        sort(a+2, a+n+1); f[0][0] = 1;
        for (int i = 2; i <= n; i++) {
            if (a[i].first == 1 || a[i].first != a[i-1].first)
                memcpy(g[0], f, sizeof(g[0])), memcpy(g[1], f, sizeof(g[1]));
            for (int j = bin[8]-1; ~j; j--)
                for (int k = bin[8]-1; ~k; k--) {
                    if ((a[i].second&k) == 0)
                        (g[0][j|a[i].second][k] += g[0][j][k]) %= p;
                    if ((a[i].second&j) == 0)
                        (g[1][j][k|a[i].second] += g[1][j][k]) %= p;
                }
            if (a[i].first == 1 || a[i].first != a[i+1].first) {
                for (int j = 0; j < bin[8]; j++)
                    for (int k = 0; k < bin[8]; k++)
                        f[j][k] = (g[0][j][k]+g[1][j][k]-f[j][k])%p;
            }
        }
        ll ans = 0;
        for (int j = 0; j < bin[8]; j++)
            for (int k = 0; k < bin[8]; k++)
                (ans += f[j][k]) %= p;
        printf("%lld
    ", (ans+p)%p);
    }
    int main() {work(); return 0; }
  • 相关阅读:
    小公司的程序员,老想跳槽怎么办?
    阿里出品的最新版 Java 开发手册,嵩山版,扫地僧
    程序员的“三十而已”
    应届毕业生,只会抄代码,该怎么办?
    可笑,你竟然不知道 Java 如何生成 UUID
    因为不知道Java的CopyOnWriteArrayList,面试官让我回去等通知
    保姆级教程,如何发现 GitHub 上的优质项目?
    给我半首歌的时间,给你说明白Immutable List
    十分钟色彩科学:LUT 的前世今生
    无源与有源元件的区别
  • 原文地址:https://www.cnblogs.com/NaVi-Awson/p/9253780.html
Copyright © 2011-2022 走看看