helloeverybody大家好,我又来了,今天发布爷的第二篇题解 (就是找优越感)
OK啊步入正题,上题目!
题目描述
有一个箱子容量为 V,同时有 n 个物品,每个物品有一个体积。
现在从 n 个物品中,任取若干个装入箱内(也可以不取),使箱子的剩余空间最小。输出这个最小值。
输入格式
第一行共一个整数 V,表示箱子容量。
第二行共一个整数 n,表示物品总数。
接下来 n 行,每行有一个正整数,表示第 i 个物品的体积。
输出格式
共一行一个整数,表示箱子最小剩余空间。
输入输出样例
输入 #1
24 6 8 3 12 7 9 7
输出 #1
0
说明/提示
对于 100% 数据,满足 0<n≤30,1≤V≤20000。
【题目来源】
NOIP 2001 普及组第四题(01年普及第四题啧嘛简单???)
说一下瓦滴思路:暴力必超时,所以用大根堆来存数据,用计数器把堆顶累加,直到它不比V小 danshi我提交之后Wonderf Answer了,错了一个测试点,n是3,i是4,8,5,那么8和(4+5),肯定是9大,所以也要考虑较小数加起来可能比较大数相加大的情况,因此我选择多用了一个小根堆
话不多说上代码!
#include <bits/stdc++.h>
using namespace std;
#define endl '\n' //快读甭管
int main(){
ios::sync_with_stdio(0); cin.tie(0); cout.tie(0); //快读甭管
priority_queue<int, vector<int>, less<int> > q; //大根堆
priority_queue<int, vector<int>, greater<int> > p; //小根堆
int i, n, v, a, c = 0, s = 0; //c、s:两个只因数器
cin >> v >> n;
for(i = 1;i <= n;++i){cin >> a; q.push(a); p.push(a);} //把每个数压到俩堆里
while(!q.empty()){ //堆非空(因为每次都pop掉堆顶,循环里也有条件,所以不用写`c < v`)
if(q.top() <= v - c) c += q.top(); //如果箱子还能装下它,则装进去物品的总体积继续累加
q.pop(); //弹出队首(堆顶)
}
//同上
while(!p.empty()){
if(p.top() <= v - s) s += p.top();
p.pop();
}
cout << min(v - c, v - s) << endl; //输出最小剩余体积
return 0;
}
白白,下次见!