来源:《算法竞赛进阶指南》

Describe:

有M块饼干要分给N个孩子。当有k个孩子分到的饼干数比第i个孩子分到的多时,会产生g[i]*k的贡献。求最小的贡献及任意一种方案。

Solution:

根据不同的饼干数分给不同的孩子会产生不同的贡献值,而题目让我们求最小的贡献值,很显然我们就能想到可以用动态规划来解决。

根据贪心的思想:
『当g[i] <g[j]时,对于剩下所有的孩子来说,使a[i]>=a[j],产生的贡献值一定是最小的。』

我们先将序列g进行从大到小排序。接下来进行状态转移,

当枚举到第i个孩子,已经分了j块饼干时:

1.当第i个孩子分到的饼干数>=i时,我们可以将ta分到的饼干数里拿出来i-1块,来分给前i-1个人,这样对前面所有孩子产生的贡献不会发生改变;(时间复杂度O(1))

2.若第i个孩子分到的饼干数为1时,我们可以假设前面也有孩子分到了1块饼干,该状态的贡献可以由第k个孩子已经分了j-i+k块饼干进行转移;(时间复杂度O(N))

3.1<若第i个孩子分到的饼干数<i时,不进行转移。(时间复杂度O(0))

所以我们可以得到:(f[0][0]=0)

f[i][j] = min{f[i][j-i],min(0<=k<i){f[k][j-i+k]+k*cigma(p[k+1]...p[i])}}.

但是题目还让我们求方案数,则我们可以开一个数组来记录当前装填是由哪一个状态转移而来的。进行dfs时,当i=0且j=0时,递归结束。

对于一个二元组x,y,不妨设它是由x0,y0转移而来,则

先递归到最底层:

1.当x=x0时,f[i][j]由f[i][j-i]转移而来,所以我们将1到x-1的数都加1;

2.当x>x0时,f[i][j]由f[k][j-i+k]转移而来,则第x0+1到第i的数都是1.

Code:

#include<bits/stdc++.h>
using namespace std; const int N = 33, M = 5050; int n, m;
struct children{
int num, p;
}a[N]; bool cmp(children x, children y) {
return x.p > y.p;
} inline void init() {
scanf("%d%d", &n, &m);
for (int i = 1; i <= n; ++ i) scanf("%d", &a[i].p), a[i].num = i;
sort(a + 1, a + n + 1, cmp);
for (int i = 1; i <= n; ++ i) a[i].p += a[i - 1].p;
} inline int cigma(int l, int r) {
return a[r].p - a[l - 1].p;
} int f[N][M], pre1[N][M], pre2[N][M];
//the least value of before i-th child, has already given out j pieces of cookies void solve1() {
memset(f, 0x3f, sizeof(f));
f[0][0] = 0;
for (int i = 1; i <= n; ++ i) {
for (int j = i; j <= m; ++ j) {
f[i][j] = f[i][j - i];
pre1[i][j] = i;
pre2[i][j] = j - i;
for (int k = 0; k < i; ++ k) {
int sum = k * cigma(k + 1, i);
if (f[k][j - i + k] + sum < f[i][j]) {
f[i][j] = f[k][j - i + k] + sum;
pre1[i][j] = k;
pre2[i][j] = j - i + k;
}
}
}
}
} int ans[N]; void dfs(int x, int y) {
if (x == 0 && y == 0) return ;
int x0 = pre1[x][y], y0 = pre2[x][y];
dfs(x0, y0);
if (x == x0) {
for (int i = 1; i <= x; ++ i)
++ ans[a[i].num];
}
else {
for (int i = x0 + 1; i <= x; ++ i)
++ ans[a[i].num];
}
} void solve2() {
dfs(n, m);
} inline void work() {
solve1();
solve2();
} inline void outo() {
printf("%d\n", f[n][m]);
for (int i = 1; i <= n; ++ i) printf("%d ", ans[i]);
puts("");
} int main() {
init();
work();
outo();
return 0;
}

Cookies题解的更多相关文章

  1. 计蒜客 UCF 2015

    #A.Find the twins # 题意 找出每个序列是否有特定的值 # 题解 坑,原始序列输出的时候每一行最后一个不能有空格 #include<bits/stdc++.h> #def ...

  2. LeetCode题解之 Assign Cookies

    1.题目描述 2.问题分析 使用贪心算法. 3 代码 class Solution { public: int findContentChildren(vector<int>& g ...

  3. AGC030 简要题解

    A - Poisonous Cookies 题意 有\(A\)个能解毒的普通饼干,\(B\)个能解毒的美味饼干,\(C\)个有毒的美味饼干,求最多能吃多少个美味饼干,每次吃完有毒的饼干后要解毒后才能继 ...

  4. CodeForces 1099F - Cookies - [DFS+博弈+线段树]

    题目链接:https://codeforces.com/problemset/problem/1099/F Mitya and Vasya are playing an interesting gam ...

  5. LeetCode All in One题解汇总(持续更新中...)

    突然很想刷刷题,LeetCode是一个不错的选择,忽略了输入输出,更好的突出了算法,省去了不少时间. dalao们发现了任何错误,或是代码无法通过,或是有更好的解法,或是有任何疑问和建议的话,可以在对 ...

  6. scrapy cookies:将cookies保存到文件以及从文件加载cookies

    我在使用scrapy模拟登录新浪微博时,想将登录成功后的cookies保存到本地,下次加载它实现直接登录,省去中间一系列的请求和POST等.关于如何从本次请求中获取并在下次请求中附带上cookies的 ...

  7. ASP.Net MVC Session和Cookies的简单使用

    目标:用Session和Cookies实现登陆信息保存和展现 Cookies实现: Controller: //把登陆用户名存到cookies中 HttpCookie cook = new HttpC ...

  8. Webform(六)——登录状态保持(Cookies内置对象)

    用户用浏览器访问一个网站,由于采用的http的特性,Web服务器并不能知道是哪一个用户正在访问,但一些网站,希望能够知道访问者的一些信息,例如是不是第一次访问,访问者上次访问时是否有未做完的工作,这次 ...

  9. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

随机推荐

  1. 第3篇 Scrum 冲刺博客(专✌️团队)

    目录 一.站立式会议 1.1 会议照片 1.2 成员完成情况 二.项目燃尽图 三.成员代码/文档签入记录 3.1 代码签入 3.2 Issue链接 3.3 CodeReview代码规范文档 四.最新项 ...

  2. vmd与ovito的对比

    1.minimize后,lammps生成的data文件 2.pdb:

  3. C++ 7种排序方法代码合集

    class Solution { public: /******************************************************************** 直接插入排 ...

  4. v-html渲染富文本图片宽高问题

    v-html渲染富文本v-html是用来渲染html的节点及字符串的,但是渲染后富文本里的图片宽高会溢出所在div的区域但是使用css直接给img是没有办法设置img的宽高的,需要使用深层级来给img ...

  5. day03 每日一行

    day03 每日一行 问题描述 用列表解释式 .生成器表达式实现 字典列表为: [{'first': 'john', 'last': 'smith', 'email': 'jsmith@exsampl ...

  6. 原生 JavaScript30 练习 Day 1 (原生JS控制键盘模拟击鼓)

     代码如下   <!DOCTYPE html> <html lang="en"> <head>     <meta charset=&qu ...

  7. UVA 11292-Dragon of Loowater (贪心)

    Once upon a time, in the Kingdom of Loowater, a minor nuisance turned into a major problem. The shor ...

  8. 2019HNCPC C Distinct Substrings 后缀自动机

    题意 给定一个长度为n字符串,字符集大小为m(1<=n,m<=1e6),求\(\bigoplus_{c = 1}^{m}\left(h(c) \cdot 3^c \bmod (10^9+7 ...

  9. Layui + tp3.2 配合表格搜索

    html 部分 <!--搜索--><fieldset class="layui-elem-field layui-field-title" style=" ...

  10. C:获取屏幕输入

    代码: #include "stdafx.h" #include "stdio.h" int _tmain(int argc, _TCHAR* argv[]) ...