咕咕机app官网_咕咕小哨

咕咕机app官网_咕咕小哨P4996 咕咕咕

大家好,又见面了,我是你们的朋友全栈君。

贡献法+组合数学

讲道理一看到辣么小的\(n\),我就想到了状压dp。

枚举子集?哦,\(O(3^n)\)的复杂度。但是我没有注意到\(n=20\)的时候是过不了的。

接下来我就在想70pts的状压dp要怎么写?

没有清楚定义状态的我连样例2都过不了,只能够手动枚举拿30pts。

所以最简单的月赛也爆炸了


70pts的状压

下面的做法是借鉴luogu题解的,不是抄袭。(你信吗?)

dp[i]表示从全0转移到i状态的所有歉意值之和。

那么会有

\[dp[i]=\sum{dp[j] + qy[j] \times num[j]}, j \in i and j \neq \emptyset\]

其中num[j]表示从全0转移到j的方案数,qy[j]表示当前这个状态的歉意值。

num[j]自然也是可以递推的,可以这么递推:

\[num[j] = \sum{num[k]}, k \in j and k \neq \emptyset\]

用枚举子集的优化技巧就可以做到\(O(3^n)\)的搞出来了。

不给代码,题解里面是有的。

我的错误之处:没有意识到方案数是对答案有影响的。

满分做法

使用贡献法,那么每一个歉意值就会对答案产生贡献了。

产生的贡献是多少?是这个状态在所有方案中出现的次数 乘以 这个方案的歉意值。

还能够发现:其实一个状态只跟它的01个数有关,跟她们的顺序是无关的。

所以设一个\(num[i]\)表示填\(i\)个1的方案数,那么就可以递推出来了。

递推式是这样子的:

\[num[i] = \sum_{j=1}^n{num[i-j] \times C_i^j}\]

那么最后的答案就是该方案0的个数 乘以 该方案1的个数 乘以 这个方案的歉意值。

对了:注意取膜!否则你就只有80pts啦。

代码:

#include<cstdio> #include<cstring> #define ll long long const int maxn = 21, maxN = 1048580; const int MOD = 998244353; ll dp[maxN]; ll qy[maxN]; ll num[maxn]; ll C[maxn][maxn]; int n, m; void init() { for(int i = 0; i <= 20; i++) { C[i][0] = C[i][i] = 1; } for(int i = 1; i <= 20; i++) { for(int j = 1; j < i; j++) { C[i][j] = (C[i - 1][j - 1] + C[i - 1][j]) % MOD; } } } int popcount(char *ch) { int len = strlen(ch); int res = 0; for(int i = len - 1; i >= 0; i--) { if(ch[i] == '1') res++; } return res; } int main() { init(); scanf("%d%d", &n, &m); num[0] = 1; for(int i = 1; i <= n; i++) { for(int j = 1; j <= i; j++) { num[i] = (num[i] + C[i][i - j] * num[i - j] % MOD) % MOD; } } ll ans = 0; for(int i = 1; i <= m; i++) { char ch[25]; int x; scanf("%s%d", ch, &x); int temp = popcount(ch), len = strlen(ch); ans = (ans + x * num[temp] % MOD * num[len - temp] % MOD) % MOD; } printf("%lld\n", ans); return 0; }

转载于:https://www.cnblogs.com/Garen-Wang/p/9913101.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/107210.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • Qt 之 QThread(深入理解)

    Qt 之 QThread(深入理解)简述前面,我们介绍了QThread常用的两种方式:worker-object子类化QThread下面,我们首先来看看子类化QThread在日常中的应用。简述子类化QThread在主线程中更新UI正常结束线程更多参考一般情况下,QThread进行耗时操作的同时会与UI进行交互,比如:显示进度、旋转等待。。。进行友好型的交互,让用户知道当前的操作。子类化QThread我们以更新进度条为例,

    2022年5月28日
    44
  • curl源码编译安装

    curl源码编译安装平台:Ubuntu20方法一:apt-get使用内置的apt下载工具进行安装,sudoapt-getinstallcurl方法二:从官网下载压缩包在官网可以找到curl的多个版本,http://curl.haxx.se/download/wgethttps://curl.haxx.se/download/curl-7.55.1.tar.gztar-xzvfcurl-7.55.1.tar.gzcdcurl-7.55.1./configurema

    2022年7月20日
    12
  • prototype.js的系列文章——关于prototype.js

    prototype.js的系列文章——关于prototype.js 很早就知道prototype.js是一个javascript的工具函数库,平时的开发中使用频率也非常的高,但是,由于工作时间问题,一直都没有静下心来研究学习一下,最近又萌发了系统学习prototype.js的念头,刚好手头比较闲,就决定边学习边将学习心得记录下来,以和更多的同仁交流分享。关于prototype.js如果你曾经使用过prototype.js,那么,本系列文章希望能够给你提供

    2022年7月23日
    11
  • ubuntu 查看 CPU 核数

    ubuntu 查看 CPU 核数物理 CPU 的个数物理核心就是计算机上实际配置的 CPU 个数 wc l 是统计行数 cat proc cpuinfo grep physicalid sort u wc l 每个 CPU 的核数指 CPU 上集成的处理数据的 CPU 核心个数 单核指 CPU 核心数一个 双核则指的是两个 uniq 可以去重连续出现的相同记录 cat proc cpuinfo grep cpucores uniq 逻辑处理器数量操作系统可以使用逻辑 CPU

    2025年12月4日
    3
  • visifire  柱状图控件

    visifire  柱状图控件最近使用到一个柱状图控件visifire用起来还是比较高级的不过会有水印商业用途需要购买正版效果还是很好的还有动画效果能够识别最大高度创建之前需要引用http://note.youdao.com/noteshare?id=4a8d01bd0bfef2cdc86c5752aad3156…

    2022年7月21日
    14
  • 中国石化测试面试题_中石化面试一般问什么

    中国石化测试面试题_中石化面试一般问什么面试过程:首先,上午进行面试人员签到,大约100人左右。一共要2个人。下午1点半开始统一面试。人员较多,所以每个人只有3分钟时间,一共最少8位面试官。过程中,他们很少提问题,如果你的技术比较新颖,会问你一些。例如SSH或SSM框架就没意思了。面试官问的面试题:以下都是对我当时的提问及个人回答。1.你都擅长哪些RPC技术。答:webservice或者restFul或者ICE微服务。2.你用过微服务…

    2022年10月15日
    1

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号