题目链接:

题目

Copying Books

Time limit: 3.000 seconds

问题描述

Before the invention of book-printing, it was very hard to make a copy of a book. All the contents had

to be re-written by hand by so called scribers. The scriber had been given a book and after several

months he finished its copy. One of the most famous scribers lived in the 15th century and his name

was Xaverius Endricus Remius Ontius Xendrianus (Xerox). Anyway, the work was very annoying and

boring. And the only way to speed it up was to hire more scribers.

Once upon a time, there was a theater ensemble that wanted to play famous Antique Tragedies. The

scripts of these plays were divided into many books and actors needed more copies of them, of course.

So they hired many scribers to make copies of these books. Imagine you have m books (numbered

1, 2, . . . , m) that may have different number of pages (p1, p2, . . . , pm) and you want to make one copy of

each of them. Your task is to divide these books among k scribes, k ≤ m. Each book can be assigned

to a single scriber only, and every scriber must get a continuous sequence of books. That means, there

exists an increasing succession of numbers 0 = b0 < b1 < b2, . . . < bk−1 ≤ bk = m such that i-th scriber

gets a sequence of books with numbers between bi−1 + 1 and bi

. The time needed to make a copy of

all the books is determined by the scriber who was assigned the most work. Therefore, our goal is to

minimize the maximum number of pages assigned to a single scriber. Your task is to find the optimal

assignment.

输入

The input consists of N cases. The first line of the input contains only positive integer N. Then follow

the cases. Each case consists of exactly two lines. At the first line, there are two integers m and k,

1 ≤ k ≤ m ≤ 500. At the second line, there are integers p1, p2, . . . , pm separated by spaces. All these

values are positive and less than 10000000.

输出

For each case, print exactly one line. The line must contain the input succession p1, p2, . . . pm divided

into exactly k parts such that the maximum sum of a single part should be as small as possible. Use

the slash character (‘/’) to separate the parts. There must be exactly one space character between any

two successive numbers and between the number and the slash.

If there is more than one solution, print the one that minimizes the work assigned to the first scriber,

then to the second scriber etc. But each scriber must be assigned at least one book.

样例

input

2

9 3

100 200 300 400 500 600 700 800 900

5 4

100 100 100 100 100

output

100 200 300 400 500 / 600 700 / 800 900

100 / 100 / 100 / 100 100

题意

给你n本书,现在让你把它们分为m堆,每一堆的书编号要连续,并且使得页数最多的那一堆的页数值最小。

题解

最大值最小,考虑二分。

二分最大堆的书页数小于等于x,然后只要贪心线性扫一遍,如果能分出少于m堆,则继续扩大答案,否则缩小。

由于多解的时候要让越前面的堆的页数越小。所以考虑从后面贪心往前面扫,因为后面的堆越大前面的就自然越小了,如果扫一遍发现分出的堆数不够,那就从前往后把没有分割的给分割开,直到分出m堆。

代码

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define lson (o<<1)
#define rson ((o<<1)|1)
#define M L+(R-L)/2
using namespace std; typedef long long LL;
const int maxn=555;
const LL INF=(LL)5e9; int m,k;
int arr[maxn],tag[maxn];
int Ma; bool ok(LL ma){
if(ma<Ma) return false;
LL sum=0;
int cnt=0;
for(int i=0;i<m;i++){
sum+=arr[i];
if(sum>ma){
sum=arr[i];
cnt++;
}
}
cnt++;
return cnt<=k;
} int main(){
int tc;
scanf("%d",&tc);
while(tc--){
Ma=-1;
memset(tag,0,sizeof(tag));
scanf("%d%d",&m,&k);
for(int i=0;i<m;i++){
scanf("%d",&arr[i]);
Ma=max(Ma,arr[i]);
}
LL l=Ma-1,r=INF;
while(l+1<r){
LL mid=l+(r-l)/2;
if(ok(mid)) r=mid;
else l=mid;
}
LL sum=0; int cnt=1;
for(int i=m-1;i>=0;i--){
sum+=arr[i];
if(sum>r){
sum=arr[i];
cnt++;
tag[i]=1;
}
}
for(int i=0;i<m-1&&cnt<k;i++){
if(!tag[i]){
tag[i]=1; cnt++;
}
}
for(int i=0;i<m-1;i++){
printf("%d ",arr[i]);
if(tag[i]) printf("/ ");
}
printf("%d\n",arr[m-1]);
}
return 0;
}

UVA 714 Copying Books 二分的更多相关文章

  1. UVa 714 Copying Books - 二分答案

    求使最大值最小,可以想到二分答案. 然后再根据题目意思乱搞一下,按要求输出斜杠(这道题觉得就这一个地方难). Code /** * UVa * Problem#12627 * Accepted * T ...

  2. uva 714 Copying Books(二分法求最大值最小化)

    题目连接:714 - Copying Books 题目大意:将一个个数为n的序列分割成m份,要求这m份中的每份中值(该份中的元素和)最大值最小, 输出切割方式,有多种情况输出使得越前面越小的情况. 解 ...

  3. UVa 714 Copying Books(二分)

    题目链接: 传送门 Copying Books Time Limit: 3000MS     Memory Limit: 32768 KB Description Before the inventi ...

  4. UVA 714 Copying Books 最大值最小化问题 (贪心 + 二分)

      Copying Books  Before the invention of book-printing, it was very hard to make a copy of a book. A ...

  5. 【NOIP提高组2015D2T1】uva 714 copying books【二分答案】——yhx

    Before the invention of book-printing, it was very hard to make a copy of a book. All the contents h ...

  6. uva 714 - Copying Books(贪心 最大值最小化 二分)

    题目描写叙述开头一大堆屁话,我还细致看了半天..事实上就最后2句管用.意思就是给出n本书然后要分成k份,每份总页数的最大值要最小.问你分配方案,假设最小值同样情况下有多种分配方案,输出前面份数小的,就 ...

  7. UVa 714 Copying books 贪心+二分 最大值最小化

    题目大意: 要抄N本书,编号为1,2,3...N, 每本书有1<=x<=10000000页, 把这些书分配给K个抄写员,要求分配给某个抄写员的那些书的编号必须是连续的.每个抄写员的速度是相 ...

  8. UVA 714 Copying Books 抄书 (二分)

    题意:把一个包含m个正整数的序列划分成k个非空的连续子序列.使得所有连续子序列的序列和Si的最大值尽量小. 二分,每次判断一下当前的值是否满足条件,然后修改区间.注意初始区间的范围,L应该为所有正整数 ...

  9. UVA - 714 Copying Books (抄书)(二分+贪心)

    题意:把一个包含m个正整数的序列划分成k个(1<=k<=m<=500)非空的连续子序列,使得每个正整数恰好属于一个序列(所有的序列不重叠,且每个正整数都要有所属序列).设第i个序列的 ...

随机推荐

  1. Part 17 Temporary tables in SQL Server

    Temporary tables in SQL Server

  2. VHDL MOD和REM(转)

    mod(取模)and rem(取余) VHDL has mod and rem. They return the same value if both arguments are positive. ...

  3. iOS 复杂tableView的 cell一般设计思路

  4. 问题:LVM lvextend增加空间后,df查看还是原来空间

    1.LVM的调整空间大小: #lvextend -L +1300M /dev/mapper/ycgsstore_sdb-wmy #lvdisplay wmy ycgsstore_sdb -wi-ao- ...

  5. CentOS学习笔记--基本命令--目录的相关操作

    Linux基本命令--目录的相关操作 常见的处理目录的命令吧: cd:变换目录 pwd:显示目前的目录 mkdir:创建一个新的目录 rmdir:删除一个空的目录 cd (变换目录) cd是Chang ...

  6. mongodb 入门笔记

    选择Mongo的关键是:这是一个 JSON 文档数据库. 1. Mongo 的术语 文档:一条完整的数据就是一个文档(对应于 MySQL 的一行). 集合:一组文档构成一个集合.类似 MySQL 中表 ...

  7. JS判断鼠标从什么方向进入一个容器

    偶然将想到的一个如何判断鼠标从哪个方向进入一个容器的问题.首先想到的是给容器的四个边添加几个块,然后看鼠标进入的时候哪个块先监听到鼠标事件.不过这样麻烦太多了.google了一下找到了一个不错的解决方 ...

  8. ipc telnet 攻击

    ping %1 -n 2net use \\%1sc \\%1 config tlntsvr start= autosc \\%1 start tlntsvrtelnet %1

  9. AndroidSDK无法下载API包的解决方法

    1:打开Android SDK Manager”,然后点击菜单栏中的“Tools”菜单选项,接下来只需选择“Options”选项即可打开设置窗体,在打开的选项中找到Others框,里面勾选第一个: F ...

  10. Linux之父Linus Torvalds:讨厌C++

    "Linux内核的创始人Linus Torvalds最近在一封邮件中说明了内核开发需要使用C语言而非C++的理由.在庞大的项目中,人们对不是自己开发的模块并不了解,能快速理解其他模块中函数的 ...