【题解】Typesetting [Hdu6107]
【题解】Typesetting [Hdu6107]
传送门:\(\text{Typesetting}\) \(\text{[Hdu6107]}\)
【题目描述】
有一篇行数无限宽度 \(MaxW\) 已知的文章,中间有张图片,图片的高度 \(h\) 和放置的位置 \(x\) 可以任意,有若干个长度 \(a[i]\) 已知的词,要保持整个词的完整性,词和词不能重叠,词和图不能重叠,词必须从第一行开始放,词的顺序不能改变,词和词连续放在一行时中间要空一行,问放完所有的词和图片所需的最少行数。
【输入】
首先读入一个整数 \(T\) 表示一共有 \(T\) 组数据,对于每组数据,将会包括以下内容:
第一行四个整数 \(n,MaxW,PW,LW\) 分别表示单词个数,文章宽度,图片宽度和图片左间距(表示图片与左边界相隔的宽度),图片右间距自行计算。
第二行 \(n\) 个整数 \(a_i\) 表示每个单词的长度。
第三行一个整数 \(Q\),表示接下来有 \(Q\) 个询问,接下来 \(Q\) 行,每行两个整数 \(x,h\),表示图片将从第 \(x\) 行开始一共占 \(h\) 行的位置。
【输出】
对于每个询问,输出此时放完所有单词需要的最少行数。
【样例】
样例输入:
2
2 7 4 3
1 3
3
1 2
2 2
5 2
3 8 2 3
1 1 3
1
1 1
样例输出:
2
3
3
1
【数据范围】
\(100 \%:\)
\(T \leqslant 10\)
\(1 \leqslant n,Q,MaxW \leqslant 10^5\)
\(1 \leqslant a_i,PW \leqslant MaxW\)
\(0 \leqslant LW \leqslant MaxW-Pw\)
【分析】
图片的存在很讨厌,可以把有图片和没有图片的分开处理,步骤如下:
\((1).\) 先算 \(x-1\) 行(对应着 \([1,x-1]\))能放多少单词。
\((2).\) 如果放不完的话,再算在只用一部分合法空间的情况下 \(h\) 行(对应着 \([x,x+h-1]\))能放多少单词。
\((3).\) 如果还是没放完,再算剩下的单词需要多少行。
上述过程均可用倍增实现,用 \(dp_{1}[i][j]\) 表示在没有图片的情况下,从第 \(i\) 个单词开始放,使用 \(2^j\) 行可以放的单词个数,\(dp_{2}[i][j]\) 表示在有图片的情况下,只使用左右间距构成的空间,\(2^j\) 行可以放的单词个数,照着上面的思路模拟一下就可以了。
【Code】
#include<algorithm>
#include<cstring>
#include<cstdio>
#include<queue>
#define Re register int
using namespace std;
const int N=1e5+3,logN=17;
int n,x,y,T,Q,PW,LW,RW,maxW,a[N],dp1[N][20],dp2[N][20];
inline void in(Re &x){
int f=0;x=0;char c=getchar();
while(c<'0'||c>'9')f|=c=='-',c=getchar();
while(c>='0'&&c<='9')x=(x<<1)+(x<<3)+(c^48),c=getchar();
x=f?-x:x;
}
inline void sakura(){//预处理
for(Re i=1,j,tmp,flag;i<=n;++i){
//dp1[i][j]: 从第i个单词开始(包括i),使用2^j行可以放的单词个数
j=i-1,tmp=0,flag=0;//flag表示是否要添空格
while(j<n&&tmp+flag+a[j+1]<=maxW)++j,tmp+=a[j]+flag,flag=1;//暴力枚举初始化
dp1[i][0]=j-i+1;
//dp2[i][j]: 从第i个单词开始(包括i),使用被图片覆盖的2^j行可以放的单词个数
j=i-1,tmp=0,flag=0;
while(j<n&&tmp+flag+a[j+1]<=LW)++j,tmp+=a[j]+flag,flag=1;//左间距
tmp=0,flag=0;
while(j<n&&tmp+flag+a[j+1]<=RW)++j,tmp+=a[j]+flag,flag=1;//右间距
dp2[i][0]=j-i+1;
}
for(Re j=1;j<=logN;++j)
for(Re i=1;i<=n;++i){
if(dp1[i][j-1])dp1[i][j]=dp1[i][j-1]+dp1[i+dp1[i][j-1]][j-1];//要注意判断,如果前半部分放不了的话
if(dp2[i][j-1])dp2[i][j]=dp2[i][j-1]+dp2[i+dp2[i][j-1]][j-1];//不管后半部分怎么折腾都没用
}
}
inline int find1(Re i,Re limit,Re dp[][20]){//从第i个单词开始(现在需要放的位置),使用limit行,条件为dp1/dp2,求无法放到第?个单词
Re tmp=0;//tmp: 已经使用的行数
for(Re j=logN;j>=0;--j)
if(tmp+(1<<j)<=limit)
i+=dp[i][j],tmp+=(1<<j);
return i;
}
inline int find2(Re i){//在无图片的情况下,放第i~n个单词个单词需要多少行
Re ans=0;
for(Re j=logN;j>=0;--j)//不管j有多大,i+dp[i][j]-1永远都<=n
if(i+dp1[i][j]-1<n)//为了不造成浪费,要找到不能一次全部放完的最大的j,即满足i+dp1[i][j]-1<n
ans+=(1<<j),i+=dp1[i][j];
if(i<=n)ans+=(1<<0),i+=dp1[i][0];//如果i!=n+1,使用一下dp1[i][0]把剩下的全部放完
return ans;
}
int main(){
// freopen("123.txt","r",stdin);
in(T);
while(T--){
in(n),in(maxW),in(PW),in(LW);
RW=maxW-PW-LW;//maxW:最大宽度 //PW:图片宽度 //LW:左间距 //RW:右间距
for(Re i=1;i<=n;++i)in(a[i]);
memset(dp1,0,sizeof(dp1));
memset(dp2,0,sizeof(dp2));
sakura();
in(Q);
while(Q--){
in(x),in(y);
Re need=find2(1);
if(need<x){printf("%d\n",need+y);continue;}//不需要使用覆盖图片的部分就可以全部放完
Re now=find1(1,x-1,dp1);//先跑前面无限制的x-1行
now=find1(now,y,dp2);//再跑覆盖图片的y行
printf("%d\n",x-1+y+find2(now));//最后看还需要再跑多少无图片的行
}
}
}
【题解】Typesetting [Hdu6107]的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
随机推荐
- 【linux】linux 查看物理CPU个数、核数、逻辑CPU个数
①物理cpu数:主板上实际插入的cpu数量,可以数不重复的 physical id 有几个(physical id) cat /proc/cpuinfo| grep "physical id ...
- 123: The filename, directory name, or volume label syntax is incorrect今天玩nginx的时候报错
今天在win下玩nginx的时候 提示500错误 看了下nginx的logs 提示 123: The filename, directory name, or volume label syntax ...
- c# winform 窗体失去焦点关闭(钩子实现)
先来一个辅助类 using System; using System.Collections.Generic; using System.Linq; using System.Runtime.Inte ...
- SQLserver创建用户和给用户权限(学)
数据库基础知识:http://blog.csdn.net/u014600432/article/details/39645701 在SQL Server中创建用户角色及授权(使用SQL语句):http ...
- elasticsearch 心得
1.es 一台机器一般为一个节点.一台机器不设置的情况下是无法创建副本集的,副本集和主本必须不在一个节点下,方便故障转移等 2.es7.x后一个索引后只能创建一个类型,可以通过修改更改 出现这个的原因 ...
- git 配置远程仓库(同一个邮箱注册多个gitlab仓库)
之前配置的全局用户和邮箱,如果是多个注册账户就不能设置为全局账户 git config --global user.name "username" git config --glo ...
- SAP MM MB5L事务代码'仅总计'选项初探
SAP MM MB5L事务代码'仅总计'选项初探 MB5L,如下查询条件, 报表结果里显示有差异, 而如下查询条件, 原因在于当勾选了'仅总计'选项以后,系统不考虑MM以外的影响库存金额的事务,而只是 ...
- 安卓开发笔记(三十四):Material Design框架实现优美的左侧侧滑栏
首先我们先上图: 下面是主页面的代码,activity_main.xml: <?xml version="1.0" encoding="utf-8"?& ...
- Android 布局渲染流程与卡顿优化
文章内容概要 一.手机界面UI渲染显示流程 二.16ms原则 三.造成卡顿的原因 四.过度绘制介绍.检测工具.如何避免造成过度绘制造成的卡顿 一.手机界面UI渲染显示流程 大家都知道CPU(中央处理器 ...
- 关于 Android 状态栏的适配总结
1.要求状态栏透明,我们的内容布局延伸到系统状态栏,就是人们口中说的沉浸式状态栏: Android 5.0 及其以后版本:设置属性 View.SYSTEM_UI_FLAG_LAYOUT_FULLSCR ...