BZOJ4380 : [POI2015]Myjnie

将$c$离散化,设:

$f[i][j][k]$为区间$[i,j]$最小值为$k$的最大收益。

$g[i][j][k]$为$\max(g[i][j][k..m])$。

$h[i][j]$为对于当前DP区间,经过$i$点的,费用限制$\geq j$的人数。

然后直接DP即可,时间复杂度$O(n^3m)$。

#include<cstdio>
#include<algorithm>
const int N=52,M=4002;
int n,m,i,j,k,x,y,t,a[M],b[M],c[M],v[M],h[N][M];
char f[N][N][M];int g[N][N][M];short p[N][N][M];
inline int lower(int x){
  int l=1,r=m,mid,t;
  while(l<=r)if(v[mid=(l+r)>>1]<=x)l=(t=mid)+1;else r=mid-1;
  return t;
}
void dfs(int l,int r,int k){
  if(l>r)return;
  int x=f[l][r][k=p[l][r][k]];
  a[x]=v[k],dfs(l,x-1,k),dfs(x+1,r,k);
}
int main(){
  scanf("%d%d",&n,&m);
  for(i=1;i<=m;i++)scanf("%d%d%d",&a[i],&b[i],&c[i]),v[i]=c[i];
  for(std::sort(v+1,v+m+1),i=1;i<=m;i++)c[i]=lower(c[i]);
  for(i=n;i;i--)for(j=i;j<=n;j++){
    for(k=i;k<=j;k++)for(x=1;x<=m;x++)h[k][x]=0;
    for(k=1;k<=m;k++)if(i<=a[k]&&b[k]<=j)for(x=a[k];x<=b[k];x++)h[x][c[k]]++;
    for(k=i;k<=j;k++)for(x=m-1;x;x--)h[k][x]+=h[k][x+1];
    for(k=m;k;k--){
      for(y=0,x=i;x<=j;x++){
        t=g[i][x-1][k]+g[x+1][j][k]+v[k]*h[x][k];
        if(t>=y)y=t,f[i][j][k]=x;
      }
      if(y>=g[i][j][k+1])g[i][j][k]=y,p[i][j][k]=k;
      else g[i][j][k]=g[i][j][k+1],p[i][j][k]=p[i][j][k+1];
    }
  }
  dfs(1,n,1);
  for(printf("%d\n",g[1][n][1]),i=1;i<=n;i++)printf("%d ",a[i]);
  return 0;
}

  

时间: 2024-11-01 00:57:26

BZOJ4380 : [POI2015]Myjnie的相关文章

bzoj [POI2015]Myjnie

[POI2015]Myjnie Time Limit: 40 Sec Memory Limit: 256 MBSec Special Judge Description 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i]. 有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗车店,且会选择这些店中最便宜的一个进行一次消费.但是如果这个最便宜的价格大于c[i],那么这个人就不洗车了. 请给每家店指定一个价格,使得所有人花的钱的总和最大. Input 第一行包含两个正整数

[POI2015]Myjnie

[POI2015]Myjnie 题目大意: 有\(n(n\le50)\)家洗车店从左往右排成一排,每家店都有一个正整数价格\(d_i\). 有\(m(m\le4000)\)个人要来消费,第\(i\)个人会选择\(a_i\sim b_i\)这些店中最便宜的一个进行一次消费.但是如果这个最便宜的价格大于\(c_i\),那么这个人就不洗车了. 请给每家店指定一个价格,使得所有人花的钱的总和最大. 思路: 将\(c\)离散化后进行区间DP. 用\(f_{i,j,k}\)表示区间\([i,j]\)最小值为

BZOJ 4380 [POI2015]Myjnie | DP

链接 BZOJ 4380 题面 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i]. 有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗车店,且会选择这些店中最便宜的一个进行一次消费.但是如果这个最便宜的价格大于c[i],那么这个人就不洗车了. 请给每家店指定一个价格,使得所有人花的钱的总和最大. Input 第一行包含两个正整数n,m(1<=n<=50,1<=m<=4000). 接下来m行,每行包含三个正整数a[i],b[i],ci Output 第

@bzoj - [email&#160;protected] [POI2015] Myjnie

目录 @[email protected] @[email protected] @accepted [email protected] @[email protected] @[email protected] 有 n 家洗车店从左往右排成一排,每家店都有一个正整数价格 p[i]. 有 m 个人要来消费,第 i 个人会驶过第 a[i] 个开始一直到第 b[i] 个洗车店,且会选择这些店中最便宜的一个进行一次消费.但是如果这个最便宜的价格大于 c[i],那么这个人就不洗车了. 请给每家店指定一个

[Poi2015]

[POI2015]?asuchy 一看以为是sb题 简单来说就是每个人获得热量要尽量多 不能找别人 首先这道题好像我自己找不到NIE的情况 很容易想到一个优化 如果一个数/2>另一个数 那么一定选这个数 然后我想着其他的话就随便分配一个 然后会得出下一个 其实这样做是错的 因为你选完之后不知道下一个会不会是来降低我当前选的那一个的热量使得我当前的原来最优变成不是最优 然后这样子 怎么办呢??? 废话 膜题解 膜拜Claris 我们既然不知道下一个会不会来降低热量 不妨把每个食物的状态都定下来 让

bzoj 4385: [POI2015]Wilcze do?y

4385: [POI2015]Wilcze do?y Description 给定一个长度为n的序列,你有一次机会选中一段连续的长度不超过d的区间,将里面所有数字全部修改为0.请找到最长的一段连续区间,使得该区间内所有数字之和不超过p. Input 第一行包含三个整数n,p,d(1<=d<=n<=2000000,0<=p<=10^16).第二行包含n个正整数,依次表示序列中每个数w[i](1<=w[i]<=10^9). Output 包含一行一个正整数,即修改后能

BZOJ 3747: [POI2015]Kinoman( 线段树 )

线段树... 我们可以枚举左端点 , 然后用线段树找出所有右端点中的最大值 . ----------------------------------------------------------------------------------------- #include<cstdio> #include<algorithm> #include<cstring> #include<iostream> #define rep( i , n ) for( i

poi2015 bzoj4377-4386训练

就按时间顺序写吧 完成度:8/10 3.30 bzoj4385 首先一定是删去连续d个数,然后枚举终点,起点显然有单调性,用单调队列乱搞搞就可以啦 bzoj4378 首先才结论:可行当且仅当把所有大于s的数全变成s然后看所有的数的和大于等于c*s,然后两个树状数组分别维护<=s的和及个数即可,注意需要离散化 3.31 bzoj4377 设一段的起点处的数为x,则m个限制条件就可以转化为x在若干个区间(或两个区间的并)里面,然后把这些区间交起来就得到了x的范围,算出个数然后减去最后m-1个数(没有

【BZOJ 3747】 3747: [POI2015]Kinoman (线段树)

3747: [POI2015]Kinoman Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 830  Solved: 338 Description 共有m部电影,编号为1~m,第i部电影的好看值为w[i]. 在n天之中(从1~n编号)每天会放映一部电影,第i天放映的是第f[i]部. 你可以选择l,r(1<=l<=r<=n),并观看第l,l+1,-,r天内所有的电影.如果同一部电影你观看多于一次,你会感到无聊,于是无法获得这部电影的好看值