题目链接:http://acm.hdu.edu.cn/showproblem.php?

pid=5355

题面:

Cake

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)

Total Submission(s): 1632    Accepted Submission(s): 273

Special Judge

Problem Description
There are m
soda and today is their birthday. The 1-st
soda has prepared n
cakes with size 1,2,…,n.
Now 1-st
soda wants to divide the cakes into m
parts so that the total size of each part is equal.



Note that you cannot divide a whole cake into small pieces that is each cake must be complete in them
parts. Each cake must belong to exact one of m
parts.
 
Input
There are multiple test cases. The first line of input contains an integerT,
indicating the number of test cases. For each test case:



The first contains two integers n
and m(1≤n≤105,2≤m≤10),
the number of cakes and the number of soda.

It is guaranteed that the total number of soda in the input doesn’t exceed 1000000. The number of test cases in the input doesn’t exceed 1000.
 
Output
For each test case, output "YES" (without the quotes) if it is possible, otherwise output "NO" in the first line.



If it is possible, then output m
lines denoting the m
parts. The first number si
of i-th
line is the number of cakes in i-th
part. Then si
numbers follow denoting the size of cakes in i-th
part. If there are multiple solutions, print any of them.
 
Sample Input
4
1 2
5 3
5 2
9 3
 
Sample Output
NO
YES
1 5
2 1 4
2 2 3
NO
YES
3 1 5 9
3 2 6 7
3 3 4 8
 
Source
2015 Multi-University Training Contest 6





解题:

    比赛的时候以为是贪心,小数据试了几组发现,仅仅要当前最大的数没取,而且小于我如今剩余的值。那么就取并标记,实际上是错的。比方取了13,纵然还能取12,却不一定要取12,由于取了12后,会造成无法凑成那个平均数,因此如此贪心是不正确的。看赛后情况,预计绝大多数队伍都是水过的(貌似说当时Special Judge坏了),看是否正确跑下23 6这组数据大概就能知分晓了。

    正确解法(题解方法):先预处理出40之内全部能划分的情况。然后兴许的仅仅须要往前面处理好的情况上凑就好了。

比方先处理好了6 3的情况,计算54 3时:



     2 6 1

     2 5 2

     2 4 3



    18 6 1 7 12 13 18 19 24 25 30 31 36 37 42 43 48 49 54 

    18 5 2 8 11 14 17 20 23 26 29 32 35 38 41 44 47 50 53

    18 4 3 9 10 15 16 21 22 27 28 33 34 39 40 45 46 51 52



    红色部分是取的6 3处理好的结果,而后面蓝色部分则是两端对称构造的结果。

    感觉题解讲的并非非常清晰。可能会有人误觉得是一直用2m去减n,直至n小于40。那么便能够取前面已经预处理好的结果了,实际上还要满足(n-2m*x)要能被划分为y块。因此并非从后往前减,而是从前往后搜寻第一个合法的状态。至于题解中40是怎样产生的。渣渣实在是证明不了。但用了20去构造,发现是会不够用的,(測试中越界了)。

    构造过程事实上挺好理解的。就是高斯第1项和尾项的和等于第2项和倒数第2项的和,不断内移即可了。

总结:

    有时候不要太过想当然,上次bc的三角形数也是。贪心不要乱用,得有一定根据。

多校很喜欢考察构造题,上次一张图,再上次24算,发现某些状态数比較大的时候,就应该联想到构造。









疑点:      

    题解中的40,不知道是怎么蹦出来的,求大神证明!







代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#define LL long long
#define mod 1000000007
using namespace std;
//訪问和成功标记
bool status[45],sign;
//记录搜索路径
int path[45];
//平均值。cnt为当前第几组解
int ave,m,n,cnt;
//flag[x][y]表示x块蛋糕能不能划分为y部分
bool flag[45][12];
struct division
{
int a[45],sz;
}store[45][12][12];
//三维分别表示共多少,划分为几块。它的第几块是,以及内部存储各自是哪几块
//预处理。搜索
//pos为当前位置,left为剩余容量。p为路径中第几个数字
void dfs(int pos,int left,int p)
{
//当前状态下已经找到一种方案
if(sign)return;
//成功找到解
if(left==0)
{
//把路径中的值存入到数组中
for(int i=0;i<p;i++)
store[n][m][cnt].a[i]=path[i];
store[n][m][cnt].sz=p;
//成功标记
sign=true;
return;
}
for(int i=pos;i>=1;i--)
{
if(!status[i]&&left>=i)
{
//该点已訪问
status[i]=1;
path[p]=i;
dfs(i,left-i,p+1);
//避免把已经成功的点又一次置为0
if(sign)return;
status[i]=0;
}
}
}
//预处理
void prep()
{
memset(flag,0,sizeof(flag));
int tmp,total;
for(int i=1;i<=20;i++)
{
//tmp是依据n要小于等于平均值。计算出来的上界
tmp=(i+1)/2;
total=i*(i+1)/2;
for(int j=1;j<=tmp;j++)
{
//假设能够划分的话
if(total%j==0)
{
flag[i][j]=1;
memset(status,0,sizeof(status));
ave=total/j;
m=j;
n=i;
//循环找多组解
for(int k=0;k<j;k++)
{
cnt=k;
sign=false;
dfs(n,ave,0);
}
}
}
}
}
int main()
{
//预处理
prep();
int t,x,y,a,b,sz;
//读入
scanf("%d",&t);
LL total;
while(t--)
{
scanf("%d%d",&x,&y);
total=1LL*(1+x)*x/2;
ave=total/y;
//排除不能划分的情况
if((total%y)||(ave<x))
{
printf("NO\n");
continue;
}
//其余的都是可构造的
printf("YES\n");
m=y;
//注意m已经乘以2
m<<=1;
int g;
//找到第一个合法的状态
for(g=1;;g++)
{
//假设能够划分。并且后面多出的数刚好能够构造
if(flag[g][y]&&((x-g)%m)==0)
break;
}
//b为构造次数
b=(x-g)/m;
for(int i=0;i<y;i++)
{
sz=store[g][y][i].sz;
//总数量
printf("%d",sz+(b<<1));
//原有解
for(int k=0;k<sz;k++)
printf(" %d",store[g][y][i].a[k]);
//构造解
for(int k=0;k<b;k++)
printf(" %d %d",k*m+g+i+1,(k+1)*m+g-i);
printf("\n");
}
}
return 0;
}

HDU 5355 Cake (WA后AC代码,具体解析,构造题)的更多相关文章

  1. HDU 5355 Cake

    HDU 5355 Cake 更新后的代码: 今天又一次做这道题的时候想了非常多种思路 最后最终想出了自觉得完美的思路,结果却超时 真的是感觉自己没救了 最后加了记忆化搜索,AC了 好了先说下思路吧.不 ...

  2. 多校第六场 1003 hdu 5355 Cake(贪心)

    题目链接:(数据加强后wa了) hdu 5355 题目大意: 给出一个蛋糕.切成1~n大小的n块.问是否能在不继续分割的情况下拼凑出m等份. 题目分析: 首先我们是可以知道每份蛋糕的尺寸的,利用n*( ...

  3. 2015多校第6场 HDU 5355 Cake 贪心,暴力DFS

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5355 题意:给你n个尺寸大小分别为1,2,3,…,n的蛋糕,要求你分成m份,要求每份中所有蛋糕的大小之 ...

  4. HDU 1005 Number Sequence(AC代码)

    #include <stdio.h> #include <string.h> int main() { int a,b,n; int i; ]={}; f[]=; f[]=; ...

  5. HDU 1003 Max Sum(AC代码)

    #include <stdio.h> int main(){ int i,t,j,n,x; int start,end,temp,max,sum; scanf("%d" ...

  6. HDU 5355 Cake (构造 + 暴力)

    题意:给定 n,m,让你把 1 ~ n 分成 m 部分,而且每部分和是一样大的. 析:首先先判断不能分成的,第一种是 sum (1 ~ n 的和)不能被 m 整除,或者 sum / m < n, ...

  7. HDU 5355 Cake(2015多校第六场,搜索 + 剪枝)

    Cake Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others) Total Sub ...

  8. HDU 1001 Sum Problem(AC代码)

    #include <stdio.h> int main(){ int k,sum; while(scanf("%d",&k)!=EOF){ ==){ sum=( ...

  9. HDU 2222 Keywords Search(AC自己主动机模板题)

    题意:给出一个字符串和若干个模板,求出在文本串中出现的模板个数. 思路:由于有可能有反复的模板,trie树权值记录每一个模板出现的次数就可以. #include<cstdio> #incl ...

随机推荐

  1. 按需要对Androguard进行定制增强

    按需对Androguard进行增强和定制修改 Androguard是一个对android应用程序进行分析的基于python的平台,功能强大.但是在使用的过程中,提供的功能不一定如我们所需,所以需要进行 ...

  2. ARM指令中的函数调用

    1. 重要寄存器 SP 栈指针,  每一种异常模式都有其自己独立的r13,它通常指向异常模式所专用的堆栈,也就是说五种异常模式.非异常模式(用户模式和系统模式),都有各自独立的堆栈,用不同的堆栈指针来 ...

  3. expect获取返回值

    对于获取多台server状态且不用交互须要用到expect,但有时候expect无法获取返回值.这里解释一下expect怎样获取返回值 expect -c "     spawn $1;   ...

  4. OpenLayers 3 之 加入地图鼠标右键菜单

    加入右键菜单,首先我们要监听鼠标右键点击的操作,我们知道鼠标右键事件名是 contextmenu.当鼠标在 html 元素之上,点击鼠标右键,便会触发 contextmenu 事件,在 context ...

  5. iOS项目工程添加.a文件遇到的Dsymutil Error

    将.a文件加入工程,很多教程讲的都是: 右键选择Add->Existing Files…,选择.a文件和相应的.h头文件.或者将这两个文件拖入XCode工程目录结构中,在弹出的界面中勾选Copy ...

  6. ios 缩放图片(平铺)

    //缩放图片(平铺) - (UIImage *)resizeImage:(NSString *)imgName { UIImage *bgImage =  [UIImage imageNamed:im ...

  7. Delphi 与 C/C++ 数据类型对照表(最新的tokyo)

    更新,下面这table为最新的tokyo基本数据类型与C++的对照关系: Delphi to C++ types mapping   Go Up to Support for Delphi Data ...

  8. 转 【MQTT】在Windows下搭建MQTT服务器

    MQTT简介 MQ 遥测传输 (MQTT) 是轻量级基于代理的发布/订阅的消息传输协议,设计思想是开放.简单.轻量.易于实现.这些特点使它适用于受限环境.该协议的特点有: 使用发布/订阅消息模式,提供 ...

  9. 【JQuery Easy UI】后台管理系统的简单布局分享

    重要说明:本博已迁移到 石佳劼的博客.有疑问请到 文章新地址 留言..! 近期做的一个简单的后台管理系统,当中用到了JQuery Easy UI框架,对于撸主这样的把控件能摆整齐就谢天谢地的码农来说, ...

  10. Android应用架构之MVP---&gt;天气实例

    我们知道.Android App 本质上抽象成两个层次:视图和数据.为了App在发展过程中高速的适应变化,方便维护和高速迭代,我们要将数据和视图解耦,而在解藕方面我们的前辈们在漫长的软件开发经验中为我 ...