【动态规划】bzoj1044: [HAOI2008]木棍分割
需要滚动优化或者short int卡空间
Description
有n根木棍, 第i根木棍的长度为Li,n根木棍依次连结了一起, 总共有n-1个连接处. 现在允许你最多砍断m个连
接处, 砍完后n根木棍被分成了很多段,要求满足总长度最大的一段长度最小, 并且输出有多少种砍的方法使得总长
度最大的一段长度最小. 并将结果mod 10007。。。
Input
输入文件第一行有2个数n,m.接下来n行每行一个正整数Li,表示第i根木棍的长度.n<=50000,0<=m<=min(n-1,10
00),1<=Li<=1000.
Output
输出有2个数, 第一个数是总长度最大的一段的长度最小值, 第二个数是有多少种砍的方法使得满足条件.
Sample Input
1
1
10
Sample Output
HINT
两种砍的方法: (1)(1)(10)和(1 1)(10)
题目分析
昨天做到这道题的稍微强化版:允许空集存在。我那题想法是,强制没有空集,最后再用组合数统计答案。状态$f[i][j]$表示前$i$个数分为$j$个非空集合的方案数,$t[i]$表示最小能够与$i$合并的位置。于是$f[i][j]=\sum_{x=t[i]}^{i}{f[i][j-1]}$。这里有一个$\sum_{i}{f[i][j]}$的形式,自然用前缀和优化。
组合数溢出调了好久……
#include<bits/stdc++.h>
const int maxn = ;
const int MO = ; int n,m,mxBound,mnBound,ans;
int a[maxn],s[maxn],t[maxn],g[maxn][maxn];
int f[maxn][maxn];
int C[maxn][maxn]; int read()
{
char ch = getchar();
int num = ;
bool fl = ;
for (; !isdigit(ch); ch = getchar())
if (ch=='-') fl = ;
for (; isdigit(ch); ch = getchar())
num = (num<<)+(num<<)+ch-;
if (fl) num = -num;
return num;
}
bool check(int x)
{
int cnt = , sum = ;
for (int i=; i<=n; i++)
if (sum+a[i] > x) sum = a[i], cnt++;
else sum += a[i];
return cnt <= m;
}
int qmi(int a, int b)
{
int ret = ;
while (b)
{
if (b&) ret = 1ll*ret*a%MO;
a = 1ll*a*a%MO, b >>= ;
}
return ret;
}
int main()
{
// freopen("ex_t2.in","r",stdin);
n = read(), m = read(), C[][] = ;
for (int i=; i<=m; i++)
{
C[i][] = ;
for (int j=; j<=i; j++)
C[i][j] = (C[i-][j]+C[i-][j-])%MO;
}
for (int i=; i<=n; i++)
a[i] = read(), mxBound += a[i], mnBound = mnBound < a[i]?a[i]:mnBound, s[i] = s[i-]+a[i];
int l = mnBound, r = mxBound, head = , tot = ;
for (int mid=(l+r)>>; l<=r; mid=(l+r)>>)
if (check(mid)) ans = mid, r = mid-;
else l = mid+;
printf("%d\n",ans);
for (int i=; i<=n; i++)
{
tot += a[i];
while (tot > ans) tot -= a[head++];
t[i] = head;
}
g[][] = f[][] = , ans = ;
for (int i=; i<=n; i++)
{
g[i][] = ;
for (int j=; j<=m; j++)
{
int delta = g[i-][j-];
if (t[i]>) delta -= g[t[i]-][j-];
(f[i][j] += 1ll*delta+MO) %= MO;
(g[i][j] = 1ll*g[i-][j]+1ll*f[i][j]) %= MO;
}
}
for (int i=; i<=m; i++)
ans = (ans+1ll*C[m][i]*f[n][i])%MO;
printf("%d\n",ans);
return ;
}
但是强化版空间512M,这题却只有162M。
首当其冲想到滚动数组优化,不过由于在这题里滚动数组需要把$j$放在枚举的外层,因此在一些奇怪的原因影响下效率会变得非常低。
注意到本题的模数非常小(不知道是不是出题人的善意),于是可以把$f[i][j]$开成short int省去一半空间。
再注意到其实同时开$f[i][j]$与$g[i][j]$是没有必要的,我们完全可以把$f[i][j]$表示成$g[i][j]$的前缀和形式,从而又省去一半空间。
#include<bits/stdc++.h>
const int maxn = ;
const int maxm = ;
const int MO = ; int n,m,mxBound,mnBound,ans;
int a[maxn],s[maxn],t[maxn];
short int g[maxn][maxm]; int read()
{
char ch = getchar();
int num = ;
bool fl = ;
for (; !isdigit(ch); ch = getchar())
if (ch=='-') fl = ;
for (; isdigit(ch); ch = getchar())
num = (num<<)+(num<<)+ch-;
if (fl) num = -num;
return num;
}
bool check(int x)
{
int cnt = , sum = ;
for (int i=; i<=n; i++)
if (sum+a[i] > x) sum = a[i], cnt++;
else sum += a[i];
return cnt <= m;
}
int main()
{
// freopen("1044.in","r",stdin);
// freopen("1044.out","w",stdout);
n = read(), m = read()+;
for (int i=; i<=n; i++)
a[i] = read(), mxBound += a[i], mnBound = mnBound < a[i]?a[i]:mnBound, s[i] = s[i-]+a[i];
int l = mnBound, r = mxBound, head = , tot = ;
for (int mid=(l+r)>>; l<=r; mid=(l+r)>>)
if (check(mid)) ans = mid, r = mid-;
else l = mid+;
printf("%d ",ans);
for (int i=; i<=n; i++)
{
tot += a[i];
while (tot > ans) tot -= a[head++];
t[i] = head;
}
register int i,j;
ans = ;
for (i=; i<=n; i++) g[i][] = ;
for (int i=; i<=n; i++)
{
for (int j=; j<=m; j++)
{
int delta = g[i-][j-];
if (t[i]>) delta -= g[t[i]-][j-];
(g[i][j] = g[i-][j]+delta) %= MO;
}
}
for (i=; i<=m; i++)
ans = (ans+g[n][i]-g[n-][i]+MO)%MO;
printf("%d\n",ans);
return ;
}
END
【动态规划】bzoj1044: [HAOI2008]木棍分割的更多相关文章
- BZOJ1044: [HAOI2008]木棍分割
1044: [HAOI2008]木棍分割 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1580 Solved: 567[Submit][Statu ...
- bzoj1044[HAOI2008]木棍分割 单调队列优化dp
1044: [HAOI2008]木棍分割 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 4314 Solved: 1664[Submit][Stat ...
- BZOJ1044 [HAOI2008]木棍分割 【二分+Dp】
1044: [HAOI2008]木棍分割 Time Limit: 10 Sec Memory Limit: 162 MB Submit: 4281 Solved: 1644 [Submit][St ...
- [BZOJ1044][HAOI2008]木棍分割 二分+贪心+dp+前缀和优化
1044: [HAOI2008]木棍分割 Time Limit: 10 Sec Memory Limit: 162 MB Submit: 4112 Solved: 1577 [Submit][St ...
- 【czy系列赛】czy的后宫6 && bzoj1044 [HAOI2008]木棍分割
题目描述 众所周知的是丧尸czy有很多妹子(虽然很多但是质量不容乐观QAQ),今天czy把n个妹子排成一行来检阅.但是czy的妹子的质量实在--所以czy看不下去了.检阅了第i个妹子会增加czy a[ ...
- [bzoj1044][HAOI2008][木棍分割] (二分+贪心+dp+队列优化)
Description 有n根木棍, 第i根木棍的长度为Li,n根木棍依次连结了一起, 总共有n-1个连接处. 现在允许你最多砍断m个连接处, 砍完后n根木棍被分成了很多段,要求满足总长度最大的一段长 ...
- [BZOJ1044][HAOI2008]木棍分割 二分 + 单调队列优化dp + 滚动数组优化dp
Description 有n根木棍, 第i根木棍的长度为Li,n根木棍依次连结了一起, 总共有n-1个连接处. 现在允许你最多砍断m个连接处, 砍完后n根木棍被分成了很多段,要求满足总长度最大的一段长 ...
- bzoj1044: [HAOI2008]木棍分割 二分+dp
有n根木棍, 第i根木棍的长度为Li,n根木棍依次连结了一起, 总共有n-1个连接处. 现在允许你最多砍断m个连接处, 砍完后n根木棍被分成了很多段,要求满足总长度最大的一段长度最小, 并且输出有多少 ...
- BZOJ1044: [HAOI2008]木棍分割(dp 单调队列)
题意 题目链接 Sol 比较套路的一个题. 第一问二分答案check一下 第二问设\(f[i][j]\)表示前\(i\)个数,切了\(j\)段的方案数,单调队列优化一下. 转移的时候只需要保证当前段的 ...
随机推荐
- Mysql相关函数使用和总结(cast、convert)
一.类型转换 1.获取一个类型的值,并产生另一个类型的值,CAST()和CONVERT()函数. 用法: CAST(value as type); CONVERT(value, type); 解释:C ...
- 升级ruby的版本
升级ruby版本,有时候安装ruby的版本过低,需要进行升级,例如安装在centos6.7安装fpm需要ruby版本在1.9以上. 1.主机环境如下: [root@test ~]# cat /etc/ ...
- C 语言实例 - 字符串翻转
C 语言实例 - 字符串翻转 C 语言实例 C 语言实例 使用递归来翻转字符串. 实例 - 字符串翻转 #include <stdio.h> void reverseSentence(); ...
- Spring security + oauth2.0 + redis + mybatis plus 搭建微服务
上个星期一个朋友请求帮忙,让我搭建一个分布式授权中心的微服务,之前我也没搭建过,在网上撸了几天前辈们写的技术博客,搞出个模型,分享给大家: 前辈们博客地址: OAuth2.0 原理:https://b ...
- JSP && Servlet | AXIS 0配置 入门
Axis2是目前比较流行的WebService引擎.WebService被应用在很多不同的场景.例如,可以使用WebService来发布服务端 Java类的方法,以便使用不同的客户端进行调用.这样可以 ...
- 配置Gradle构建
构建基础配置 Android Studio包含一个顶级的构建文件和每个模块的构建文件.构建文件被称为 build.gradle,它是一个纯文本文件,它使用Groovy语法来配置由Android Gra ...
- hdu4553约会安排(线段树区间合并)
链接 poj3667的加强版 当时的题解 这里只不过对于女神需要另开算,DS的占用的时间不加在女神身上,女神的时间都要加,清空的时候也都要算. #include <iostream> #i ...
- [转]nopcommerce之权限模块
本文转自:http://www.nopchina.net/category/%E6%9E%B6%E6%9E%84.html 这篇文章简单介绍一下nopcommerce的权限模块,nopcommerce ...
- 前端之CSS常见兼容性问题
1.双倍浮动BUG: 描述:块状元素设置了float属性后,又设置了横向的margin值,在IE6下显示的margin值要比设置的值大: 解决方案:给float的元素添加 display:inline ...
- mongodb Gridfs操作
GridFS 介绍 GridFS是MongoDB规范用于存储和检索大文件,如图片,音频文件,视频文件等.这是一种文件系统用来存储文件,但数据存储于MongoDB集合中.GridFS存储文件比其文档大小 ...