Description

传送门

Solution

算法1 32pts

爆搜,复杂度\(O((m+1)^n)\)

算法2 84pts

裸的dp,复杂度\(O(n^3m)\)

首先有一个显然的性质要知道:

最多只有一种主要食材出现在超过一半的主要食材里。

接下来考虑如果只有前两个限制条件的情况,那么答案就是

\[\Pi_{i=1}^{n} (sum_i+1) - 1
\]

其中\(sum_i = \sum \limits_{j=1}^m a_{i,j}\),\(+1\)是因为对于每一行只有选一道菜或者不选这些选择,\(-1\)是因为要去除一道菜都不选的情况。

对于第3个限制条件,发现直接做不太好做,考虑容斥,即用总方案数,也就是上面的式子,减去不合法的方案数。

由最开始的那个性质可以得到一个做法:

枚举不合法的那一种主要食材,然后进行\(dp\)。发现我们并不需要知道每一种主要食材具体用在了多少道菜上,只需要知道当前枚举的食材用在了多少道菜,其它的并不影响方案。那么设\(f_{i,j,k}\)表示前\(i\)中烹饪方式,选了\(j\)道菜,其中\(k\)道的主要食材是枚举的不合法食材。转移分三种情况:令\(s\)表示当前枚举的不合法食材,

  1. 不在这一种烹饪方式中进行选择:\(f_{i,j,k}=f_{i-1,j,k}\)

  2. 在这种烹饪方式中选择了合法的食材:\(f_{i,j,k}=(sum_i-a_{i,s}) \times f_{i,j-1,k}\)

  3. 在这种烹饪方式中选择了不合法的食材:\(f_{i,j,k}=a_{i,s}\times f_{i,j-1,k-1}\)

那么不合法的方案数就是

\[\sum \limits_{k\ge \lfloor \frac{j}{2} \rfloor} f_{n,j,k}
\]

code

#include <bits/stdc++.h>
using namespace std; typedef long long ll;
const ll mod = 998244353;
const int _ = 100 + 10;
const int __ = 2000 + 10;
int n, m, A[_][__];
ll sum[_], f[_][_ << 1], tmp, ans = 1; int main() {
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
scanf("%d", &A[i][j]);
sum[i] = (sum[i] + A[i][j]) % mod;
}
ans = ans * (sum[i] + 1) % mod;
}
ans = (ans - 1 + mod) % mod; for (int k = 1; k <= m; ++k) {
memset(f, 0, sizeof(f));
f[0][n] = 1;
for (int i = 1; i <= n; ++i) {
for (int j = -i + n; j <= i + n; ++j) {
f[i][j] = (f[i - 1][j] + f[i - 1][j - 1] * A[i][k] % mod + f[i - 1][j + 1] * (sum[i] - A[i][k]) % mod) % mod;
if (i == n && j > n) tmp = (tmp + f[i][j]) % mod;
}
}
} ans = (ans - tmp + mod) % mod;
printf("%lld\n", ans);
return 0;
}

算法三 100pts

考虑如何对算法二的\(dp\)进行优化,减少不必要的状态。对限制三进行转化,限制三即为

\[x\le \lfloor \frac{k}{2} \rfloor
\]

\[2x \le k
\]

\[2x - k \le 0
\]

\[x-(k-x)\le 0
\]

发现并不需要关心使用了食材的菜的具体数量,只需要关心合法与不合法的菜的差值即可,即这个差值与原来差值相同的状态的集合是对应的,那么我们就可以以此为状态进行dp,转移与上面是类似的。

唯一要注意的一点是可能出现负数,要加上一个偏移量\(n\)

  1. 不在这一种烹饪方式中进行选择:\(f_{i,j}=f_{i-1,j}\)

  2. 在这种烹饪方式中选择了合法的食材:\(f_{i,j}=(sum_i-a_{i,s}) \times f_{i,j+1}\)

  3. 在这种烹饪方式中选择了不合法的食材:\(f_{i,j}=a_{i,s}\times f_{i,j-1}\)

code

#include <bits/stdc++.h>
using namespace std; typedef long long ll;
const ll mod = 998244353;
const int _ = 100 + 10;
const int __ = 2000 + 10;
int n, m, A[_][__];
ll sum[_], f[_][_ << 1], tmp, ans = 1; int main() {
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; ++i) {
for (int j = 1; j <= m; ++j) {
scanf("%d", &A[i][j]);
sum[i] = (sum[i] + A[i][j]) % mod;
}
ans = ans * (sum[i] + 1) % mod;
}
ans = (ans - 1 + mod) % mod; for (int k = 1; k <= m; ++k) {
memset(f, 0, sizeof(f));
f[0][n] = 1;
for (int i = 1; i <= n; ++i) {
for (int j = -i + n; j <= i + n; ++j) {
f[i][j] = (f[i - 1][j] + f[i - 1][j - 1] * A[i][k] % mod + f[i - 1][j + 1] * (sum[i] - A[i][k]) % mod) % mod;
if (i == n && j > n) tmp = (tmp + f[i][j]) % mod;
}
}
} ans = (ans - tmp + mod) % mod;
printf("%lld\n", ans);
return 0;
}

【CSP-S 2019】D2T1 Emiya 家今天的饭的更多相关文章

  1. 【NOIP/CSP2019】D2T1 Emiya 家今天的饭

    这个D2T1有点难度啊 原题: 花了我一下午的时间,作为D2T1的确反常 条件很奇怪,感觉不太直观,于是看数据范围先写了个暴力 写暴力的时候我就注意到了之前没有仔细想过的点,烹饪方式必须不同 虽然a很 ...

  2. 「CSP-S 2019」Emiya 家今天的饭

    description loj 3211 solution 看到题目中要求每种主要食材至多在一半的菜中被使用,容易想到补集转换. 即\(ans=\)总方案数-存在某一种食材在一半以上的菜中被使用的方案 ...

  3. [CSP-S 2019 Day2]Emiya家今天的饭

    思路: 这种题目就考我们首先想到一个性质.这题其实容易想到:超限的菜最多只有一个,再加上这题有容斥那味,就枚举超限的菜然后dp就做完了. 推式子能力还是不行,要看题解. 式子还需要一个优化,就是废除冗 ...

  4. 洛谷P5664 Emiya 家今天的饭 问题分析

    首先来看一道我编的题: 安娜写宋词 题目背景 洛谷P5664 Emiya 家今天的饭[民间数据] 的简化版本. 题目描述 安娜准备去参加宋词大赛,她一共掌握 \(n\) 个 词牌名 ,并且她的宋词总共 ...

  5. 洛谷P5664 Emiya 家今天的饭 题解 动态规划

    首先来看一道题题: 安娜写宋词 题目背景 洛谷P5664 Emiya 家今天的饭[民间数据] 的简化版本. 题目描述 安娜准备去参加宋词大赛,她一共掌握 \(n\) 个 词牌名 ,并且她的宋词总共有 ...

  6. [CSP-S2019]Emiya 家今天的饭 题解

    CSP-S2 2019 D2T1 很不错的一题DP,通过这道题学到了很多. 身为一个对DP一窍不通的蒟蒻,在考场上还挣扎了1h来推式子,居然还有几次几乎推出正解,然而最后还是只能打个32分的暴搜滚粗 ...

  7. 【CSP-S 2019】【洛谷P5664】Emiya 家今天的饭【dp】

    题目 题目链接:https://www.luogu.org/problem/P5664 Emiya 是个擅长做菜的高中生,他共掌握 \(n\) 种烹饪方法,且会使用 \(m\) 种主要食材做菜.为了方 ...

  8. Emiya家今天的饭 NOIP2019 (CSP?) 类DP好题 luoguP5664

    luogu题目传送门! 首先,硬求可行方案数并不现实,因为不好求(去年考场就这么挂的,虽然那时候比现在更蒟). 在硬搞可行方案数不行之后,对题目要求的目标进行转换: 可行方案数 = 总方案数 - 不合 ...

  9. CSP-S 2019 Emiya 家今天的饭

    64 pts 类似 乌龟棋 的思想,由于 \(64pts\) 的 \(m <= 3\), 非常小. 我们可以设一个 \(dp\),建立 \(m\) 个维度存下每种物品选了几次: \(f[i][A ...

随机推荐

  1. 7.SourceTree 的使用

    SourceTree 是 Windows 和Mac OS X 下免费的 Git 和 Hg 客户端管理工具,同时也是Mn版本控制系统工具.支持创建.克隆.提交.push.pull 和合并等操作. 下载路 ...

  2. 201871010112-梁丽珍《面向对象程序设计(java)》第十六周学习总结

    项目 内容 这个作业属于哪个课程 https://www.cnblogs.com/nwnu-daizh/ 这个作业的要求在哪里 https://www.cnblogs.com/nwnu-daizh/p ...

  3. C语言快速入门一:win10系统环境搭建

    0.搭建环境:WIN10 64位 1.下载minGW.zip编译器 2.解决上述文件,配置环境变量 3.配置变成后验证:打开cmd命令行,输入gcc -v 提示以下内容,说明编译器安装成功 D:\mm ...

  4. acwing 116. 飞行员兄弟

    地址  https://www.acwing.com/problem/content/118/ “飞行员兄弟”这个游戏,需要玩家顺利的打开一个拥有16个把手的冰箱. 已知每个把手可以处于以下两种状态之 ...

  5. Acwing 14. 不修改数组找出重复的数字

    题目地址  https://www.acwing.com/problem/content/description/15/ 来源:剑指Offer 给定一个长度为 n+1n+1 的数组nums,数组中所有 ...

  6. WPF 委托 事件 B窗体调用A窗体方法

    原文:WPF 委托 事件 B窗体调用A窗体方法 具体实现 A窗体 中加载B窗体  B窗体触发A窗体里的方法 当点击B窗体确定Button事件   给A窗体俩个TextBox赋值 并关闭B窗体 B窗体 ...

  7. 给用户提供就医帮助的安卓APP

    经过我们的小组的成员讨论,我们确定了我们小组的项目,即是一款给用户提供就医帮助的安卓APP. 项目计划及功能:计划两个月内团队成员共同开发完成此款APP,此款APP提供预约挂号,名医名院咨询, 就医导 ...

  8. 基于UDP协议的socket套接字编程

    目录 一.UDP套接字简单示例 1.1 服务端 二.客户端 三.UPD套接字无粘包问题 3.1 服务端 3.2 客户端 四.qq聊天 4.1 服务端 4.2 客户端1 4.3 客户端2 4.4 运行结 ...

  9. js a 标签 通过download 实现下载功能

    download 属性规定被下载的超链接目标. 在 <a> 标签中必须设置 href 属性. 该属性也可以设置一个值来规定下载文件的名称.所允许的值没有限制,浏览器将自动检测正确的文件扩展 ...

  10. 用OC实现双向链表:构造链表、插入节点、删除节点、遍历节点

    一.介绍 双向链表:每一个节点前后指针域都和它的上一个节点互相指向,尾节点的next指向空,首节点的pre指向空. 二.使用 注:跟单链表差不多,简单写常用的.循环链表无法形象化打印,后面也暂不实现了 ...