poj1821——Fence
题意:
一个栅栏一共有n(从1——n)个木板,我们找k个工人去粉刷它,li表示每个人有限制粉刷木板数量,pi表示粉刷一个木板得到的钱,si表示他开始在那个木板前面
如果一个工人要粉刷,那么他必须粉刷si这个木板,而且工人粉刷时必须是连续的木板
题解:
dp[i][j]表示有i个人粉刷j块木板所获得的最大利润
dp[i][j]=max(max(dp[i-1][j],dp[i][j-1]),dp[i][k]+(j-k)*p(i))
dp[i-1][j]表示i-1个人粉刷j块木板所获得的最大利润
dp[i][j-1]表示i个人粉刷j-1块木板所获得的最大利润
dp[i][k]+(j-k)*p(i) 这里面的k是枚举在第i个人可以粉刷木板的数量(因为题目要求第i个人必须粉刷si这块木板,那么粉刷区间肯定也包括它)
//m是一个结构体,里面包含一个工人的li,pi,si
for(int j = m[i].s;j <= m[i].s + m[i].l - 1;j ++)
{
for(int k = j - m[i].l;k <= m[i].s - 1;k ++)
if(k >= 0)
{
dp[i][j] = max(dp[i][j],dp[i - 1][k] + (j - k) * m[i].p);
}
}
话可以利用单调队列降低复杂度
通过上面的代码我们可以看出来j越大,那么k也就随之变大,这就符合单调队列的特性,单调队列(递减队列)里面放dp[i][k]-k*p(i),为什么放这个是因为我们后面枚举j的时候
直接可以通过 (队列头)+j*p(i) 来找求最优解(这个(j*p(i))与前面的抵消了一部分,剩下的就是第i个人粉刷的部分),因为递减队列头部放的值肯定是最大的,所以
队列头部就是最好的dp转移位置,为了防止粉刷区间大于l(i),我们要每次先对单调队列中的数据进行处理
代码:
1 #include<stdio.h>
2 #include<string.h>
3 #include<iostream>
4 #include<algorithm>
5 #include<queue>
6 #include<deque>
7 using namespace std;
8 typedef long long ll;
9 const int maxn=2e5+10; //数组开到2e4都不行,
10 const int mod = 998244353;
11 const int INF=0x3f3f3f3f;
12 struct shudui
13 {
14 int l,p,s;
15 }m[maxn];
16 bool mmp(shudui x,shudui y)
17 {
18 return x.s<y.s;
19 }
20 int dp[110][maxn];
21 struct jihe
22 {
23 int k,x;
24 }str1;
25 deque<jihe>r;
26 int main()
27 {
28 int n,k;
29 scanf("%d%d",&k,&n);
30 for(int i = 1;i <= n;i ++)
31 scanf("%d%d%d",&m[i].l,&m[i].p,&m[i].s);
32 sort(m + 1,m + 1 + n,mmp); //这个要加上
33
34 for(int i=1;i<=n;++i)
35 {
36 for(int j = 1;j <= k;j ++)
37 dp[i][j] = max(dp[i - 1][j],dp[i][j - 1]);
38
39 while(r.size()) r.pop_back();
40 for(int kk = max(m[i].s - m[i].l,0);kk <= m[i].s - 1;kk ++)
41 {
42 int tmp = dp[i - 1][kk] - kk * m[i].p;
43
44 while(r.size() && r.back().x < tmp) r.pop_back();
45 str1.k=kk;
46 str1.x=tmp;
47 r.push_back(str1);
48 }
49 for(int j = m[i].s;j <= m[i].s + m[i].l - 1;j ++)
50 {
51 while(r.size() && r.front().k < j - m[i].l) r.pop_front();
52 dp[i][j] = max(dp[i][j],r.front().x + j * m[i].p);
53 }
54 }
55 int ans = 0;
56 for(int i = 1;i <= k;i ++)
57 ans = max(ans,dp[n][i]);
58 printf("%d\n",ans);
59 return 0;
60 }
poj1821——Fence的更多相关文章
- [POJ1821]Fence(单调队列优化dp)
[poj1821]Fence 有 N 块木板从左至右排成一行,有 M 个工匠对这些木板进行粉刷,每块木板至多被粉刷一次.第 i 个工匠要么不粉刷,要么粉刷包含木板 Si 的,长度不超过Li 的连续一段 ...
- POJ1821 Fence
题意 Language:Default Fence Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 6478 Accepted: ...
- poj1821 Fence【队列优化线性DP】
Fence Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 6122 Accepted: 1972 Description ...
- POJ1821 Fence 题解报告
传送门 1 题目描述 A team of $k (1 <= K <= 100) $workers should paint a fence which contains \(N (1 &l ...
- poj1821 Fence(单调队列优化dp)
地址 一排N个木板,M个工匠站在不同位置$S_i$,每个人可以粉刷覆盖他位置的.最长长度为$L_i$木板段,每刷一个有$P_i$报酬.同一木板只刷一次.求最大报酬. 根据每个人的位置dp,设$f[i] ...
- $Poj1821\ Fence\ $单调队列优化$DP$
Poj Acwing Description 有N块木板等待被M个工匠粉刷,每块木板至多被刷一次.第i个工匠要么不粉刷,要么粉刷包含木块Si的,长度不超过Li的连续的一段木板,每粉刷一块可以得到P ...
- poj1821 Fence(dp,单调队列优化)
题意: 由k(1 <= K <= 100)个工人组成的团队应油漆围墙,其中包含N(1 <= N <= 16 000)个从左到右从1到N编号的木板.每个工人i(1 <= i ...
- 单调队列与DP
算是一个总结吧! 先来一个模板: TYVJ 1305 最大子序和 题目描述 输入一个长度为n的整数序列,从中找出一段不超过M的连续子序列,使得整个序列的和最大. 例如 1,-3,5,1,-2,3 当m ...
- Fence(poj1821)
Fence Time Limit: 1000MS Memory Limit: 30000K Total Submissions: 4705 Accepted: 1489 Description ...
随机推荐
- MySQL常用的一些(就几个)聚合函数
聚合函数 (常用) 函数名称 描述 CONUT() 记数 SUM() 求和 AVG() 平均值 MAX() 最大值 MIN() 最小值 -- ================= 聚合函数 ====== ...
- 更改mysql的密码
mysql> set password for 'root'@'localhost' =PASSWORD('');Query OK, 0 rows affected (0.17 sec) mys ...
- Java高并发与多线程(二)-----线程的实现方式
今天,我们开始Java高并发与多线程的第二篇,线程的实现方式. 通常来讲,线程有三种基础实现方式,一种是继承Thread类,一种是实现Runnable接口,还有一种是实现Callable接口,当然,如 ...
- 【十天自制软渲染器】DAY 02:画一条直线(DDA 算法 & Bresenham’s 算法)
推荐关注公众号「卤蛋实验室」或访问博客原文,更新更及时,阅读体验更佳 第一天我们搭建了 C++ 的运行环境并画了一个点,根据 点 → 线 → 面 的顺序,今天我们讲讲如何画一条直线. 本文主要讲解直线 ...
- Linux下Too many open files问题排查与解决
作者: Grey 原文地址: Github 语雀 博客园 Too many open files是Linux系统中常见的错误,从字面意思上看就是说程序打开的文件数过多,不过这里的files不单是文件的 ...
- Spring Boot Scheduled定时任务特性
SpringBoot中的Scheduled定时任务是Spring Boot中非常常用的特性,用来执行一些比如日切或者日终对账这种定时任务 下面说说使用时要注意的Scheduled的几个特性 Sched ...
- 开发中经常使用到的Xcode快捷键
工欲善其事必先利其器. 有了这些快捷键加持,你写代码不仅很6而且还很好看. 这些快捷键都是平时使用频率非常高的,今天整理出来分享给大家了. 左缩进:Cmd + [ 右缩进:Cmd + ] 代码格式化/ ...
- 【Python】中国有哪些同名的省市县?
这道题适合写个脚本来解. 首先从百度地图API下载一份行政区划数据. 开发资源 | 百度地图API SDK 然后做一个简单的数据统计就可以啦~ 行政区划同一级同名的: import pandas as ...
- 细数JS中实用且强大的操作符&运算符
目录 1,前言 2,代码+应用 2.1,短路运算符 || 2.2,短路运算符 && 2.3,零合并操作符 ?? 2.4,可选链操作符 ?. 2.5,位运算符 & 和 | 2.6 ...
- python-列表包字典-根据字典的某一个键的值来进行排序
python-列表包字典-根据字典的某一个键的值来进行排序 列表包字典的数据结构 要实现按照字典中的某一个键所对应的值进行排序 有两种办法 方法一,使用列表的sort方法 由小到大排 列表.sort( ...