牛客2020跨年场 D 衔尾蛇

衔尾蛇

https://ac.nowcoder.com/acm/contest/9854/D

我们发现题目中有若干旋转置换。
而对于一共有n条蛇的环而言,本题对应的置换群由一个单位置换和n-1个旋转置换构成。
根据burnside引理
图片说明
我们可以对于每个置换求出不变的颜色数进而求出答案。
对于单次确定三种蛇使用个数的情况,即代码中的cal(a, b, c)
复杂度是O((a+b+c)log(a+b+c))
当然,我们能发现gcd只有log个,能够通过反演进一步优化到亚线性。

#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;

typedef long long ll;

ll cc[20][20];

ll gcd(ll a, ll b) {
    if (b == 0) return a;
    return gcd(b, a % b);
}

ll cal(ll a, ll b, ll c) {
        if (a + b + c == 0) return 0;
    ll res = cc[a + b + c][a] * cc[b + c][b]; //处理单位置换
    for (int i = 1; i < a + b + c; i++) { //处理旋转i个位置的旋转置换
        int g = gcd(i, a + b + c);
        int d = (a + b + c) / g;
        if (a % d == 0 && b % d == 0 && c % d == 0) {
            res += cc[g][a / d] * cc[g - a / d][b / d];
        }
    }
    return res / (a + b + c);
}

int main() {
    //freopen("0.txt", "r", stdin);
    cc[0][0] = 1;
    for (int i = 1; i < 20; i++) {
        cc[i][0] = 1;
        for (int j = 1; j < 20; j++) cc[i][j] = cc[i - 1][j - 1] + cc[i - 1][j];
    }
    ll a, b, c;
    scanf("%lld%lld%lld", &a, &b, &c);
    ll ans = 0;
    for (int i = 0; i <= a; i++) {
        for (int j = 0; j <= b; j++) {
            for (int k = 0; k <= c; k++) {
                ans += cal(i, j, k);
            }
        }
    }
    printf("%lld\n", ans);
    return 0;
}
全部评论
学长tql
1 回复 分享
发布于 2021-01-01 21:14

相关推荐

测试糕手手:社会第一课,随便吹牛逼,直接说四个月,别老实。老实人只会被欺负
点赞 评论 收藏
分享
小浪_Coding:找硬件测试,也可兼顾软测欧, 简历还可以的 ,注意排版,项目写的有条理一点, 然后个人技能多加点, 润色好简历之后就开始沟通海投了,深圳,东莞这边做硬件相关的公司还不少, 医疗类,仪器类的都可以尝试
点赞 评论 收藏
分享
不愿透露姓名的神秘牛友
07-03 18:22
投了几百份简历,专业和方向完全对口,都已读不回。尝试改了一下学校,果然有奇效。
steelhead:这不是很正常嘛,BOSS好的是即便是你学院本可能都会和聊几句,牛客上学院本机会很少了
点赞 评论 收藏
分享
评论
8
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务