【题解】SOFTWARE 二分+搜索/dp
题目描述
一个软件开发公司同时要开发两个软件,并且要同时交付给用户,现在公司为了尽快完成这一任务,将每个软件划分成m个模块,由公司里的技术人员分工完成,每个技术人员完成同一软件的不同模块的所用的天数是相同的,并且是已知的,但完成不同软件的一个模块的时间是不同的,每个技术人员在同一时刻只能做一个模块,一个模块只能由一个人独立完成而不能由多人协同完成。一个技术人员在整个开发期内完成一个模块以后可以接着做任一软件的任一模块。写一个程序,求出公司最早能在什么时候交付软件。
输入输出格式
输入格式:
输入文件第一行包含两个由空格隔开的整数n和m,接下来的n行每行包含两个用空格隔开的整数d1和d2,d1表示该技术人员完成第一个软件中的一个模块所需的天数,d2表示该技术人员完成第二个软件中的一个模块所需的天数。
输出格式:
输出文件仅有一行包含一个整数d,表示公司最早能于d天后交付软件。
输入输出样例
输入样例#1: 复制
3 20
1 1
2 4
1 6
输出样例#1: 复制
18
说明
1<=n<=100,1<=m<=100。 1<= d1,d2<=100。
思路
- 最大值最小(要求做得最慢的人越早做完),因此想到二分
dp
- 设mid天后交付;
- $f[i][j]$表示当前i个人共完成了j个模块一时,还能完成多少个模块二
- 设第i个人完成了k个模块一,则在剩下的时间内他还可以完成$(mid-k*d1[i])/d2[i]$个模块二
- 可以得到转移方程
$$f[i][j]=max(f[i][j],f[i-1][j-k]+((mid-k*d1[i])/d2[i]))$$
- 最后检验计较$f[n][m]$与m大小(当n个人完成了m个模块一时,能否完成m个模块二)
搜索
- 设mid天后交付;
- 按人员编号进行搜索,用lft[0],lft[1]表示还有几个模块一,模块二需要完成
- 枚举第i个人做了几个模块一,在通过$(mid-k*d1[i])/d2[i]$算出可以在剩下几天内在做几个模块二
- lft[0],lft[1]分别减去第i个人完成的模块一,二数量=>进行下一个人的搜索
- 当i=n时,自然剩下所有的未完成的lft[0],lft[1]都要他完成
- 计算此时$d1[i]*lft[0]+d2*lft[1]$的大小与mid的关系
这个二分的DFS检验其实是可以记忆化的,如果是成功的就直接退出了,但是不成功的没有退出,但是会重复计算,比如前3个人,模块一还剩下3个,模块二剩下4个,这个f(3,3,4)可能由很多的状态扩展过来,因此会有很多重复计算,不过要记得每次检验之前要清空f
- 可以用$f[cnt][lft[0]][lft[1]]$记下此方案是否可行,下次再搜到就直接
return false
代码
#include<cmath>
#include<cstdio>
#include<string>
#include<cstring>
#include<iostream>
#include<algorithm>
#define re register int
using namespace std;
inline int read(){
int x=0,w=1;
char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
if(ch=='-') w=-1,ch=getchar();
while(ch>='0'&&ch<='9') x=(x<<1)+(x<<3)+ch-48,ch=getchar();
return x*w;
}
int lft[2]; //还有多少模块要完成
int a1[1000],a2[1000];
int mid,n,m;
char f[110][110][110];
bool chek(int cnt) { //cnt--人
if (f[cnt][lft[1]][lft[2]]) return false; //表示前cnt个人 剩下lef[1] 和lef[2]时不可行
if (cnt==n) {
if (lft[1]*a1[cnt]+lft[2]*a2[cnt] <= mid)
return true;
else {
f[cnt][lft[1]][lft[2]]=true;
return false;
}
}
for (re i=0;i*a1[cnt]<=mid&&i<=lft[1];i++) {
int j=(mid-i*a1[cnt])/a2[cnt]; //可做的a2数
if (j>lft[2]) j=lft[2];
lft[1]-=i;
lft[2]-=j;
if (chek(cnt+1)) return true;
lft[1]+=i;
lft[2]+=j;
}
f[cnt][lft[1]][lft[2]] = true;
return false;
} int main() {
freopen("T21331.in","r",stdin);
freopen("T21331.out","w",stdout);
n=read(),m=read();
for (re i=1;i<n+1;i++){
a1[i]=read();
a2[i]=read();
}
if (n == 1) { printf("%d",m*a1[1]+m*a2[1]); return 0; }
int cnt=0;
int r=min(max(m*a1[2],m*a2[1]),max(m*a1[1],m*a2[2]));
int ans=0x7f7f7f7f;
while (cnt<=r) {
mid=(cnt+r)>>1;
lft[1]=m;
lft[2]=m;
memset(f,0,sizeof f);
if (chek(1)){
ans=min(mid,ans);
ans = mid;
r=mid-1;
}
else cnt=mid+1;
}
printf("%d",ans);
return 0;
}
搜索
#include<cmath>
#include<cstdio>
#include<string>
#include<cstring>
#include<iostream>
#include<algorithm>
#define re register int
using namespace std;
const int inf=2147483647;
const int N=110;
inline int read(){
int x=0,w=1;
char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
if(ch=='-') w=-1,ch=getchar();
while(ch>='0'&&ch<='9') x=(x<<1)+(x<<3)+ch-48,ch=getchar();
return x*w;
}
int f[N][N],d[N][2],n,m,t;
int main() {
freopen("T21331.in","r",stdin);
freopen("T21331.out","w",stdout);
re i,j,k;
n=read(),m=read();
for(i=1;i<=n;++i) d[i][0]=read(),d[i][1]=read();
int l=0,r=100000;
int mid;
while (l<r) {
mid=(l+r)>>1;
for(i=0;i<=n;++i) for(j=0;j<=m;++j) f[i][j]=-inf;
f[0][0]=0;
for(i=1;i<=n;++i)
for(j=0;j<=m;++j)
for(k=0;k<=min(j,mid/d[i][0]);++k)
f[i][j]=max(f[i][j],f[i-1][j-k]+(mid-d[i][0]*k)/d[i][1]);
if(f[n][m]>=m) r=mid;
else l=mid+1;
}
printf("%d",l);
return 0;
}
dp
【题解】SOFTWARE 二分+搜索/dp的更多相关文章
- Poj 1973 Software Company(二分+并行DP)
题意:软件公司接了两个项目,来自同一个合同,要一起交付.该公司有n个程序猿来做这两个项目A和B,每个项目都被分为m个子项目,给定每个程序猿做一个A中的子项目需要的时间Xi秒,和做B中的子项目所需时间Y ...
- 【题解】hdu 3586 Information Disturbing 二分 树形dp
题目描述 Information DisturbingTime Limit: 6000/3000 MS (Java/Others) Memory Limit: 131072/65536 K (Java ...
- (四连测)滑雪场的高度差题解---二分 + 搜索---DD(XYX)的博客
滑雪场的高度差 时间限制: 1 Sec 内存限制: 128 MB 题目描述 滑雪场可以看成M x N的网格状山地(1 <= M,N <= 500),每个网格是一个近似的平面,具有水平高度 ...
- 8VC Venture Cup 2016 - Final Round D. Preorder Test 二分 树形dp
Preorder Test 题目连接: http://www.codeforces.com/contest/627/problem/D Description For his computer sci ...
- POJ3208 Apocalypse Someday(二分 数位DP)
数位DP加二分 //数位dp,dfs记忆化搜索 #include<iostream> #include<cstdio> #include<cstring> usin ...
- CF360B Levko and Array (二分查找+DP)
链接:CF360B 题目: B. Levko and Array time limit per test 2 seconds memory limit per test 256 megabytes i ...
- HDU 5682 zxa and leaf 二分 树形dp
zxa and leaf 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5682 Description zxa have an unrooted t ...
- hihocoder #1301 : 筑地市场 二分+数位dp
#1301 : 筑地市场 题目连接: http://hihocoder.com/problemset/problem/1301 Description 筑地市场是位于日本东京都中央区筑地的公营批发市场 ...
- 【bzoj5174】[Jsoi2013]哈利波特与死亡圣器 二分+树形dp
题目描述 给你一棵以1为根的有根树,初始除了1号点为黑色外其余点均为白色.Bob初始在1号点.每次Alice将其中至多k个点染黑,然后Bob移动到任意一个相邻节点,重复这个过程.求最小的k,使得无论B ...
随机推荐
- JS求一个字符串在另一个字符串中出现的次数
参数说明: subString子字符串 originString母字符串 isIgnoreCap是否忽略大小写,默认忽略 function stringFre(subString, originStr ...
- CSS中常用的函数
一.CSS函数的用法 设置css的属性值时可以使用css函数,如果color:rgba(20,30,50,0.5),css中有很多这样的函数. 二.常用的CSS函数 a.calc()函数 这个函数 ...
- [DB] MapReduce
概述 大数据计算的核心思想:移动计算比移动数据更划算 MapReduce既是一个编程模型,又是一个计算框架 包含Map和Reduce两个过程 终极目标:用SQL语句分析大数据(Hive.SparkSQ ...
- 下载: www.bitmover.com/lmbench,最新版本3.0-a9
软件说明: lmbench是个用于评价系统综合性能的多平台开源benchmark,能够测试包括文档读写.内存操作.进程创建销毁开销.网络等性能,测试方法简单.Lmbench是个多平台软件,因此能够对同 ...
- Linux进阶之正则,shell三剑客(grep,awk,sed),cut,sort,uniq
一.正则表达式:Regular Expression 正则表达式:正则表达式使用单个字符串来描述.匹配一系列符合某个句法规则的字符串.在很多文本编辑器里,正则表达式通常被用来检索.替换那些符合某个模式 ...
- linux中级之ansible配置(roles)
一.roles介绍 什么情况下用到roles? 假如我们现在有3个被管理主机,第一个要配置成httpd,第二个要配置成php服务器,第三个要配置成MySQL服务器.我们如何来定义playbook? 第 ...
- Bootstrap Bootstrap3 与 Bootstrap4 的区别
Bootstrap3 与 Bootstrap4 官网地址 Bootstrap3 官网:https://v3.bootcss.com Bootstrap4 官网:https://v4.bootcss.c ...
- jenkins部署vue项目
一.新建自由风格的项目 二.配置项目 三.部分部署脚本 #!/bin/bashecho $PATHnpm config set proxy nullnpm config set https-proxy ...
- Java并发:ThreadLocal的简单介绍
作者:汤圆 个人博客:javalover.cc 前言 前面在线程的安全性中介绍过全局变量(成员变量)和局部变量(方法或代码块内的变量),前者在多线程中是不安全的,需要加锁等机制来确保安全,后者是线程安 ...
- Mui入门(Day_42)
开始体验Mui 1. 安装HbuilderX 下载地址:https://www.dcloud.io/hbuilderx.html 2. 新建Mui项目 3. 文件结构介绍 _ css : 样式表文件 ...