VIJOS1392拼拼图的小杉

2024-02-03 03:08
文章标签 拼拼 vijos1392 小杉

本文主要是介绍VIJOS1392拼拼图的小杉,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目大意

给定n个数,将这n个数中的一些依次放进m个集合,每个集合中所有数的和不能超过T。集合包含的元素不能交叉,也就是说如果第1个数和第3个数放入了集合1,那么第2个数要么放入集合1,要么不放入任何一个集合。
求m个集合中最多能包含多少个数

1<=n,m,T<=1000

分析

来源于题解中小岛的想法

这题如果状态直接定义为f[i,j]=x表示前i个数放入j个集合最多能包含x个数
那么操作起来会比较困难,所有应换一种状态的定义。

记a[i]为第i个数,f[i,j]=(t,s)表示前i个数选了j个数需要t个集合,最后一个集合所有数之和为s,那么
f[i,j]=min{f[i-1,j],f[i-1,j-1]+a[i]}
f[0,0]=(1,0)
那么就是一个0-1背包问题啦
最后的答案是使得f[n,i]的t<=m的最大的i

代码

#include<cstdio>
const int maxn=1000;
int n,m,T,a[maxn+10];
struct node{int t,s;bool operator < (const node d) const{if (t<d.t)return 1;if (t>d.t)return 0;return s<d.s;}node operator + (const int d) const{if (s+d<=T)return (node){t,s+d};elsereturn (node){t+1,d};}
};
node f[maxn+10];
int main(){scanf("%d%d%d",&n,&m,&T);node te;for (int i=1;i<=n;i++)scanf("%d",&a[i]);f[0].t=1;for (int i=1;i<=n;i++){f[i]=f[i-1]+a[i];for (int j=i-1;j>=1;j--){te=f[j-1]+a[i];if (te<f[j])f[j]=te;}}for (int i=n;i>=0;i--)if (f[i].t<=m){printf("%d\n",i);return 0;}
}

这篇关于VIJOS1392拼拼图的小杉的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/672812

相关文章

从小程序升级成独立APP,“小鹅拼拼”如何帮鹅厂“拼”未来?

去年的4月29日,小鹅拼拼小程序和公众号同时上线,分别承载着卖货和招商与推广功能。经历了一年的沉浮之后, 5月17日,小鹅拼拼已经被单独分离出来,成为独立的APP。 与小程序时期的不同之处在于,那时的Slogan为“没有什么事情比买到好东西更快乐!”。而在独立成App之后,Slgan已经更新为了“享受每一分!”。小鹅拼拼,已经独立成为了一款综合性的社交电商平台。 那么,小鹅拼拼独立成

基于Vue实现一个有点意思的拼拼乐小游戏

笔者去年曾写过一个类似的拼拼乐小游戏,技术栈采用自己的Xuery框架和原生javascript实现的,脚手架采用gulp来实现,为了满足对vue的需求,笔者再次使用vue生态将其重构,脚手架采用比较火的vue-cli。 前言 为了加深大家对vue的了解和vue项目实战,笔者采用vue生态来重构此项目,方便大家学习和探索。技术栈如下: vue-cli4 基于vue的脚手架Xuery 笔者基于原生