洛谷 P1336 最佳课题选择 题解

这篇具有很好参考价值的文章主要介绍了洛谷 P1336 最佳课题选择 题解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

P1336 最佳课题选择 题解

题目链接

题目分析

状态:考虑 \(f_{i,j}\) 表示前 \(i\) 种论文里面,一共写了 \(j\) 篇,的最少花费时间。

转移策略:我们一次考虑每一种论文写多少篇。假设写 \(k\) 篇,\(k \in [0,j] \cap \mathbb{Z}\) ,有转移方程:

\[f_{i,j} = \min(f_{i-1,j-k} + \text{cost}(i,k)), k \in [0,j] \cap \mathbb{Z} \]

其中

\[\text{cost}(i,k) = a_i \times k ^ {b_i} \]

可以做记忆化优化,因为多次访问同一个 \(cost(i,k)\)

初始化:我们考虑选0篇无论怎样花费为0,选0种论文但多于0篇一定是不可能的,花费为Inf,再加上转移中要取min的缘故,我们初始化为:

\[f_{i,j}= \begin{cases} 0 & ,j=0\\ + \inf &, j \neq 0\\ \end{cases} \]

相关变量范围见代码。

最后,我们就可以开滚动数组,或者倒序\(j\)优化掉一维。

代码在文末。


和背包问题对比

这里和01背包及无限背包做个对比。

01背包的转移选择策略,是一个一个选(同时也是一种一种选,因为一种里面只有一个)。转移方程:

\[f_{i,j} = \max(f_{i-1,j}, f_{i-1,j-c_i} + v_i) \]

无限背包的转移选择策略,是在一种里面一个个选。转移方程:

\[f_{i,j} = \max(f_{i-1,j}, f_{i,j-c_i} + v_i) \]

可以看到两者细微的差别在,选物品的时候,01背包访问 \(i-1\),无限背包访问 \(i\),所以可以把其中一个维度压缩掉之后,01反着做,无限正着做。

本题目有两点不同:

第一,本题相当于“选择超过某价值的物品,使得容量需要量最小”,把状态的内外对换了一对。

第二,本题目中,如果选了前面一个,那么后面再选的时候,贡献会不一样,因为时间贡献是幂级的而非线性的。类比到无限背包中就是“选的越多,价值越少”,已经选了多少会影响到后面选的东西。选同一个东西的贡献,在不同情况下不一样。

于是,按照无限背包“一个一个”选的策略,就会涉及到之前的选择情况,我们尝试把第 \(i\) 种选择数目 \(k\) 加入到状态中,有

\[f_{i,j,k(k \leq j)}= \begin{cases} f_{i,j-1,k-1} - \text{cost}(i,k-1) + \text{cost}(i,k) & ,k \neq 0\\ \min(f_{i-1,j,l}), l \in [0,j] \cap \mathbb{Z} &, k = 0\\ \end{cases} \]

这样是正确的,但是我们发现大多数的状态被浪费了,因为 \(k \neq 0\) 的时候只有一种选择。

还有一种方式是把 \(f_{i,j}\) 中选择了多少个第i种论文,另存为 \(g_{i,j} = k\),于是我们有转移方程(错误的):

\[f_{i,j}=\min\left( \begin{array}{} f_{i,j-1} - \text{cost}(i,g_{i,j-1}) + \text{cost}(i,g_{i,j}), g_{i,j} = g_{i,j-1} + 1 ,\\ f_{i-1,j}, g(i,j) = 0 \end{array} \right) \]

这个转移方程看上去好像是对的,因为就是按照无限背包的思路改的,但实际上是错误的,提交后只能有10分。因为理论上这种转移方程只能处理线性的 \(\text{cost}(i,k)\),即 \(b_{i} = 1\)\(b_{i} = 0\)

我们和无限背包进行对比:假设有 \(f_{1,1} = 12\)\(f_{1,2} = 24\)\(f_{2,1} = 2\),那么对于无限背包问题,有:

\[\begin{array}{} f_{2,2} = \max(f_{1,2},f_{2,1} + v_{2})\\ \because f_{2,1} = \max(f_{1,1},f_{2,0} + v_{2}) \neq f_{1,1}\\ \therefore f_{2,1} \geq f_{1,1}\\ \therefore f_{2,1} + v_{2} \geq f_{1,1} + v_{2} \end{array} \]

因此 \(f_{1,1} + v_{2}\) 不会影响 \(f_{2,1} + v_{2}\)\(f_{2,2}\) 可能的贡献,也就是不可能贡献(被max掉了)。

但是在这个问题中,我们令\(w_{i,k} = \text{cost}(i,k) - \text{cost}(i,k-1)\),有:

\[\begin{array}{} f_{2,2} = \min(f_{1,2},f_{2,1} + w_{2,1})\\ \because f_{2,1} = \min(f_{1,1},f_{2,0} + w_{2,2}) \neq f_{1,1}\\ \therefore f_{2,1} \leq f_{1,1}\\ \therefore f_{2,1} + w_{2,2} \quad? \quad f_{1,1} + w_{2,1} \end{array} \]

可以发现我们无法判断 \(f_{1,1} + w_{2,1}\) 对于\(f_{2,2}\)是否有贡献,其根源在于对于同一个 \(i\)\(w_{i,k}\) 是可能不同的,因此不等式失效,即在这种状态和转移方程下,不满足最优子结构。因此DP失效。

(这也说明,一个可DP的问题是存在“最优子结构”“无后效性”,只有特定的方式去“成立”这些性质才能DP,不是随便搞一搞就一定可以DP的)

也就是说,在无限背包问题中,我们把一个物品拿走,从规模更小的状态中转移,这个物品的贡献是确定的,决定“拿走”这一选择中的最优解,就只能是承接之前的最优解。

然而在这个问题中,如果我们把一个物品拿走,从规模更小的状态中转移时,这个物品的贡献是不确定的,也就是说除了最优解以外,次有解也有可能贡献出更好的结果,因此“最优子结构失效了”。

因此,我们在考虑状态和状态转移方程时,可以灵活更改状态、转移方式和策略、记录信息等等,同时也要按照普适状况考虑转移的合法性和具体转移方法。文章来源地址https://www.toymoban.com/news/detail-630796.html


标准AC代码:

#include <bits/stdc++.h>

#define N (int)(205)
#define M (int)(25)

using namespace std;

typedef long long LL;

int n,m;
LL f[N];
LL a[M],b[M];
LL p[M][N];

LL cost(int i, int k)
{
	if(p[i][k] == -1)
		return p[i][k] = a[i] * pow(k,b[i]);
	else
		return p[i][k];
}

int main()
{
	memset(p,-1,sizeof(p));
	memset(f,0x7f,sizeof(f));
	ios::sync_with_stdio(false);
	cin.tie(0);cout.tie(0);
	cin >> n >> m;
	for(int i = 1;i <= m;i++) cin >> a[i] >> b[i];
	f[0] = 0;
	for(int i = 1;i <= m;i++)
	{
		for(int j = n;j >= 1;j--)
		{
			for(int k = 0;k <= j;k++)
			{
				f[j] = min(f[j],f[j-k] + cost(i,k));
			}
		}
	}
	cout << f[n];
	return 0;
}

另一种AC的代码:

#include <bits/stdc++.h>

#define N (int)(205)
#define M (int)(25)

using namespace std;

typedef long long LL;

int n,m;
LL f[M][N][N];
LL a[M],b[M];
LL p[M][N];

LL cost(int i, int k)
{
	if(p[i][k] == -1)
		return p[i][k] = a[i] * pow(k,b[i]);
	else
		return p[i][k];
}

int main()
{
	memset(p,-1,sizeof(p));
	memset(f,0x7f,sizeof(f));
	ios::sync_with_stdio(false);
	cin.tie(0);cout.tie(0);
	cin >> n >> m;
	for(int i = 1;i <= m;i++) cin >> a[i] >> b[i];
	for(int i = 0;i <= m;i++) f[i][0][0] = 0;
	for(int i = 1;i <= m;i++)
	{
		for(int j = 1;j <= n;j++)
		{
			for(int l = 0;l <= j;l++)
			{
				f[i][j][0] = min(f[i][j][0],f[i-1][j][l]);
			}
			for(int k = 1;k <= j;k++)
			{
				f[i][j][k] = f[i][j-1][k-1] - cost(i,k-1) + cost(i,k);
			}
		}
	}

	LL ans = 1e18;
	for(int k = 0;k <= n;k++)
		ans = min(ans,f[m][n][k]);
	cout << ans;
	return 0;
}

到了这里,关于洛谷 P1336 最佳课题选择 题解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 洛谷 P1122 最大子树和 题解

    一道入门的树形DP。 首先我们对于数据进行有序化处理,这便于我们利用数据结构特点(可排序性)来发觉数据性质(有序、单调、子问题等等性质),以便于后续的转化、推理和处理。 有序化可以“转化和创造”性质 首先将视角从无根树切换为有根树,这样我们就可以得

    2024年02月17日
    浏览(37)
  • 洛谷 U123017 机器人题解

    机器人会按照输入的指令进行移动,命令包括’E’,‘S’,‘W’,\\\'N’四种,分别对应东南西北。执行某个命令时它会消耗1秒钟向对应的方向移动一格单位。 在0时刻机器人位于(0,0)。 如果遇到’E’命令,向右一个单位,即到达(1,0)。如果遇到’S’命令,向下一个单位,即到达

    2024年03月21日
    浏览(43)
  • 【洛谷题解】B2010 带余除法

    题目链接:带余除法 - 洛谷 题目难度:入门 涉及知识点:除法,计算余数 题意: 分析:计算商后再用a/商*b得余数 AC代码: 总结:计算商后再用a/商*b得余数

    2024年02月19日
    浏览(37)
  • 洛谷 P1387 最大正方形 题解

    通过二维前缀和判定矩形内是否全为1,计算和等于长度的平方就判断为是 复杂度 (Theta (n^2log{n})) 设状态 (f_{i,j}) 为以第 (i) 行 (j) 列为右下角的最大正方形的边长, (a_{i,j}) 表示输入矩阵中的数值,有转移方程: [f_{i,j} = (min(f_{i-1,j},f_{i,j-1},f_{i-1,j-1}) + 1) * a_{i,j}] 解释:

    2024年02月16日
    浏览(45)
  • 洛谷 P3304 [SDOI2013] 直径 题解

    题目链接 第一部分好说,求直径,dfs或者DP都可以。 第二部分,有一个定理,就是所有直径中点重叠。 那么有两种情况 一种是中点在一个节点上,那么显然这个点是每条直径的终点,也就是说直径的一半相等。从这个点出发dfs,找出所有最远点。如果只有两条,输出depth之

    2024年02月14日
    浏览(40)
  • 洛谷 P1379 八数码难题 A* 题解

    刚做完一道模板A*,看到这题我直接小脑萎缩了... 阿米诺斯!这怎么用A*?!——刚开题的我 beeeeeeeeee like 甚至比模板简单(这是绿的...) 其实会是会但是纸张的是这玩意我不会搞估价函数我草! 然后突然想到能不能把这个状态下有多少个数字不在目标位置作为估价函数?

    2024年03月22日
    浏览(63)
  • 【洛谷 P2084】进制转换 题解(模拟+字符串)

    无 今天小明学会了进制转换,比如(10101)2 ,那么它的十进制表示的式子就是 : 1*2 4+0*2 3+1*2 2+0*2 1+1*2^0, 那么请你编程实现,将一个M进制的数N转换成十进制表示的式子。 注意:当系数为0时,该单项式要省略。 两个数,M和N,中间用空格隔开。 共一行,一个十进制表示的式

    2024年01月20日
    浏览(37)
  • 【洛谷 P1106】删数问题 题解(贪心+字符串)

    键盘输入一个高精度的正整数 N N N (不超过 250 250 250 位),去掉其中任意 k k k 个数字后剩下的数字按原左右次序将组成一个新的非负整数。编程对给定的 N N N 和 k k k ,寻找一种方案使得剩下的数字组成的新数最小。 输入两行正整数。 第一行输入一个高精度的正整数 n n

    2024年02月07日
    浏览(46)
  • 【洛谷 P1996】约瑟夫问题 题解(数组+模拟+循环)

    n n n 个人围成一圈,从第一个人开始报数,数到 m m m 的人出列,再由下一个人重新从 1 1 1 开始报数,数到 m m m 的人再出圈,依次类推,直到所有的人都出圈,请输出依次出圈人的编号。 注意:本题和《深入浅出-基础篇》上例题的表述稍有不同。书上表述是给出淘汰 n − 1

    2024年02月07日
    浏览(36)
  • 【洛谷 P1181】数列分段 Section I 题解(贪心算法)

    对于给定的一个长度为 N N N 的正整数数列 A i A_i A i ​ ,现要将其分成 连续 的若干段,并且每段和不超过 M M M (可以等于 M M M ),问最少能将其分成多少段使得满足要求。 第1行包含两个正整数 N , M N,M N , M ,表示了数列 A i A_i A i ​ 的长度与每段和的最大值,第 2 2 2 行包

    2024年02月08日
    浏览(51)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包