【题解】洛谷P1776 宝物筛选 [单调队列优化多重背包]

二进制优化 还是不够快,如果我们想时间复杂度为 ,还得找新的方法。

(W 为背包最大可承载量,N 为物品种类数)

例题:P1776 宝物筛选 - 洛谷

原来的转移式很普通:

注意到对于每个 ,有一个特定的取值范围

而且答案是取该范围内的极值(最大或最小)

最重要的,对于每个最优决策点 j - k * w ,具有单调性, 随着 i 的增长是单调递增的。

这种情况下可以用单调队列优化

队首 保存最优决策点 ,每次将**不符合条件(超出范围)**的队首弹出。

上面那个式子,可以化为:

整体:

而对于单种物体

取值范围:

其实就是把原来式子里的 换成了

之所以要写成这一坨,

是要让 格式一样 ,方便单调队列

但是注意到 ,而不是

这是为什么呢?不会越界吗?

这是因为背包有个特点, 占背包的空间 不一定就是 ,但一定

所以 不一定就真的放了 个当前物品,只是长这个样子。

所以上面这整个式子,真正当前物品被放进去的个数是

把转移式化成这样,其实已经很快了。

但还能更快,我们知道 只跟 有关系。

也就是我们枚举 ,而 余数

讲了这么多,看看代码吧:

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;

typedef long long LL;
const int N = 4e4 + 10;

struct node {
	LL x;   //f[j + k * w] - k * v
	int k;   // k值
} q[N];

LL f[N];

int main() {
	ios::sync_with_stdio(false);
	cin.tie(0);

	int n; LL W;
	cin >> n >> W;

	LL sum = 0, ans = 0;
	for (int i = 1; i <= n; i++) {
		LL v, w; int m;
		cin >> v >> w >> m;

		if (w == 0) {      //如果这个宝物重量为 0,那就直接加上
			sum += v;
			continue;
		}

		int K = W / w;   //最大可选数量
		m = min(m, K);
		for (int j = 0; j < w; j++) {    //枚举余数 j
			int head, tail;    
			head = 1; tail = 0;    //队头队尾初始化
			LL r = (W - j) / w;    // k 的上限

			for (int k = 0; k <= r; k++) {
				while(head <= tail && f[j + k * w] - k * v >= q[tail].x) {
                    tail--;     //当前 k 比队尾优而且比队尾后,踢队尾
				}
				tail ++;
				q[tail].k = k;                    // 记录物品数量
                q[tail].x = f[j + k * w] - k * v;    // 记录对应的 f 值

				while(head <= tail && k - q[head].k > m) {     //队头在不在可选域
                    head++;
				}
                if (head <= tail) {
					f[j + k * w] = max(f[j + k * w], q[head].x + k * v);    //更新 f
					ans = max(ans, f[j + k * w]);   //找最大的 f
				}
			}
		}
	}
	// f[W] + sum 也一样
	cout << sum + ans << "\n";
	return 0;
}
相关推荐
KingRumn6 小时前
Linux信号之标准信号与实时信号
linux·算法
LXS_3579 小时前
Day 18 C++提高 之 STL常用容器(string、vector、deque)
开发语言·c++·笔记·学习方法·改行学it
源代码•宸9 小时前
Leetcode—620. 有趣的电影&&Q3. 有趣的电影【简单】
数据库·后端·mysql·算法·leetcode·职场和发展
2301_8002561110 小时前
地理空间数据库中的CPU 和 I/O 开销
数据库·算法·oracle
deng-c-f10 小时前
Linux C/C++ 学习日记(53):原子操作(二):实现shared_ptr
开发语言·c++·学习
一个不知名程序员www10 小时前
算法学习入门---结构体和类(C++)
c++·算法
墨雪不会编程11 小时前
C++ string 详解:STL 字符串容器的使用技巧
java·开发语言·c++
yangpipi-12 小时前
《C++并发编程实战》第5章 C++内存模型和原子操作
android·java·c++
SunkingYang12 小时前
MFC进程间消息通信深度解析:SendMessage、PostMessage与SendNotifyMessage的底层实现与实战指南
c++·mfc·共享内存·通信·postmessage·sendmessage·进程间