题目链接

先考虑,对于确定的一个数,怎样移动代价最少(或者移到哪个位置最优)?

假设我们都移到下标\(1\)位置(设集合点为\(1\)),那么移动到下标\(2\)与\(1\)相比代价差为:\(下标<1的石子数和-下标>1的石子数和\)。

如果它为负,那么把移到\(1\)的代价加上它,令集合点变为\(2\)...

这样一直改变集合点,直到 \(下标<p的石子数和 \geq 下标>p的石子数和\)。那么移到\(p\)就是最优的。

这样感觉很对。怎么证明?

我们发现式子左边其实就是前缀和,右边是后缀和。因为石子数非负,所以随着\(p\)移动,前缀和是递增的,后缀和递减。

即如果出现 \(前缀和 \geq 后缀和\) 的情况,前缀和就永远大于等于后缀和了。

那么我们对\([l,r]\)的所有数都进行这个贪心。

首先我们要算出所有数集合到1的代价和。这个可以用数位DP算出(递推数的个数,用个数求和)。

然后枚举\(p=2\sim n\)位,我们可以求 以\(p\)为分界,前缀数位和 小于 后缀数位和 且 在\([0,r]\) 的数的个数。其中每个数会减少的代价就是\(前缀和-后缀和\)。

因为数位和最多差不多是230,可以直接枚举这两个状态。同样数位DP。

\(f[i][j][k][0/1]\)表示当前到第\(i\)位,总数位和为\(j\),\(p\)位之前的数位和为\(k\),是否到上界,的数的个数。

另外还可以直接减掉\(k\)那一维。。

记忆化就好写的多了(还快)。

//49592kb	404ms
#include <cstdio>
#include <cstring>
#include <algorithm>
typedef long long LL;
const int N=52,M=245; int A[N];
LL g[N][2],sum[N][2],f[N][M][M][2]; LL Calc(LL x,int base)
{
int n=0;
for(; x; x/=base) A[++n]=x%base;
std::reverse(A+1,A+1+n);// memset(g,0,sizeof g), memset(sum,0,sizeof sum);
g[0][1]=1;
for(int i=0; i<n; ++i)//好像还是从0方便。。
{
LL v=g[i][1]; int ai=A[i+1];
g[i+1][1]+=v, sum[i+1][1]=sum[i][1]+v*i*ai;
for(int j=0; j<ai; ++j) g[i+1][0]+=v, sum[i+1][0]+=sum[i][1]+v*i*j;
v=g[i][0];
for(int j=0; j<base; ++j) g[i+1][0]+=v, sum[i+1][0]+=sum[i][0]+v*i*j;
} LL ans=sum[n][0]+sum[n][1];
for(int p=1; p<n; ++p)
{
f[0][0][0][1]=1;
for(int i=0; i<n; ++i)
{
int ai=A[i+1];
if(i+1<=p)
{
LL v;
for(int j=0,lim=i*(base-1); j<=lim; ++j)
{
if(v=f[i][j][j][1])//好不直观。。
{
f[i+1][j+ai][j+ai][1]+=v;//+=
for(int k=0; k<ai; ++k) f[i+1][j+k][j+k][0]+=v;
}
if(v=f[i][j][j][0])
for(int k=0; k<base; ++k) f[i+1][j+k][j+k][0]+=v;
}
}
else
{
LL v;
for(int j=0,lim=i*(base-1); j<=lim; ++j)
for(int k=0,lim2=p*(base-1); k<=lim2; ++k)
{
if(v=f[i][j][k][1])
{
f[i+1][j+ai][k][1]+=v;
for(int l=0; l<ai; ++l) f[i+1][j+l][k][0]+=v;
}
if(v=f[i][j][k][0])
for(int l=0; l<base; ++l) f[i+1][j+l][k][0]+=v;
}
}
}
for(int i=0,lim=p*(base-1); i<=lim; ++i)//pre
for(int j=i+1,lim2=n*(base-1); i+j<=lim2; ++j)//suf
ans+=(i-j)*(f[n][i+j][i][0]+f[n][i+j][i][1]);
for(int i=1; i<=n; ++i)
for(int j=0,lim=i*(base-1); j<=lim; ++j)
for(int k=0,lim2=p*(base-1); k<=lim2; ++k)
f[i][j][k][0]=0, f[i][j][k][1]=0;
}
return ans;
} int main()
{
LL L,R; int K; scanf("%lld%lld%d",&L,&R,&K);
printf("%lld\n",Calc(R,K)-Calc(L-1,K));
return 0;
}

BZOJ.3598.[SCOI2014]方伯伯的商场之旅(贪心 数位DP)的更多相关文章

  1. bzoj 3598: [Scoi2014]方伯伯的商场之旅【数位dp】

    参考了这个http://www.cnblogs.com/Artanis/p/3751644.html,好像比一般方法好写 大概思想就是先计算出把所有石子都合并到1位置的代价,这样显然有一些是不优的,然 ...

  2. bzoj 3598 [Scoi2014]方伯伯的商场之旅——数位dp

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3598 TJ:https://www.cnblogs.com/Zinn/p/9351218.h ...

  3. BZOJ3598 SCOI2014方伯伯的商场之旅(数位dp)

    看到数据范围就可以猜到数位dp了.显然对于一个数最后移到的位置应该是其中位数.于是考虑枚举移到的位置,那么设其左边和为l,左右边和为r,该位置数为p,则需要满足l+p>=r且r+p>=l. ...

  4. 洛谷P3286 [SCOI2014]方伯伯的商场之旅

    题目:洛谷P3286 [SCOI2014]方伯伯的商场之旅 思路 数位DP dalao说这是数位dp水题,果然是我太菜了... 自己是不可能想出来的.这道题在讲课时作为例题,大概听懂了思路,简单复述一 ...

  5. [BZOJ3598][SCOI2014]方伯伯的商场之旅(数位DP,记忆化搜索)

    3598: [Scoi2014]方伯伯的商场之旅 Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 449  Solved: 254[Submit][Sta ...

  6. 【bzoj3598】: [Scoi2014]方伯伯的商场之旅

    Description 方伯伯有一天去参加一个商场举办的游戏.商场派了一些工作人员排成一行.每个人面前有几堆石子.说来也巧,位置在 i 的人面前的第 j 堆的石子的数量,刚好是 i 写成 K 进制后的 ...

  7. [SCOI2014]方伯伯的商场之旅

    Description 方伯伯有一天去参加一个商场举办的游戏.商场派了一些工作人员排成一行.每个人面前有几堆石子.说来也巧,位置在 i 的人面前的第 j 堆的石子的数量,刚好是 i 写成 K 进制后的 ...

  8. 【数位DP】SCOI2014 方伯伯的商场之旅

    题目内容 方伯伯有一天去参加一个商场举办的游戏.商场派了一些工作人员排成一行.每个人面前有几堆石子. 说来也巧,位置在 \(i\) 的人面前的第 \(j\) 堆的石子的数量,刚好是 \(i\) 写成 ...

  9. 【bzoj3598】 Scoi2014—方伯伯的商场之旅

    http://www.lydsy.com/JudgeOnline/problem.php?id=3598 (题目链接) 题意 Solution 原来这就是极水的数位dp,呵呵= =,感觉白学了.htt ...

随机推荐

  1. Python中__repr__和__str__区别

    Python中__repr__和__str__区别 看下面的例子就明白了 class Test(object): def __init__(self, value='hello, world!'): ...

  2. ARMV8 datasheet学习笔记4:AArch64系统级体系结构之编程模型(3)- 异常

    1.前言 本文介绍异常相关内容,包括异常类型,异常进入,异常返回,异常层次结构,异常的路由等 2.  RESET ARMV8体系结构支持两种类型的RESET Cold reset:Reset PE所有 ...

  3. python httplib和urllib的性能比较

    httplib代码: urlParseResult = urlparse(url) host = urlParseResult.hostname path = urlParseResult.path ...

  4. OCM_第六天课程:Section3 —》数据库可用性

    注:本文为原著(其内容来自 腾科教育培训课堂).阅读本文注意事项如下: 1:所有文章的转载请标注本文出处. 2:本文非本人不得用于商业用途.违者将承当相应法律责任. 3:该系列文章目录列表: 一:&l ...

  5. CentOS切换为iptables防火墙并进行相关配置

    CentOS切换为iptables防火墙 切换到iptables首先应该关掉默认的firewalld,然后安装iptables服务. 1.关闭firewall: service firewalld s ...

  6. 浏览器的userAgent归纳

    IE IE6 User-Agent:Mozilla/4.0 (Windows; MSIE 6.0; Windows NT 5.2) IE7 User-Agent:Mozilla/4.0 (compat ...

  7. django----Form扩展

    用第二种方式需要加上下面的这个: 三.判断用户民是不存在,存在就不添加了 from django.core.exceptions import ValidationError initial  修改时 ...

  8. poj2155二维树状数组区间更新

    垃圾poj又交不上题了,也不知道自己写的对不对 /* 给定一个矩阵,初始化为0:两种操作 第一种把一块子矩阵里的值翻转:0->1,1->0 第二种询问某个单元的值 直接累计单元格被覆盖的次 ...

  9. hdu5289 ST表+二分

    用裸的St表+暴力枚举查询时稳TLE的,可以枚举每个区间的起点+二分满足条件的区间右端,这样复杂度是O(nlogn) #include<iostream> #include<cstr ...

  10. AC自动机(trie图版)

    AC自动机是一个多模字符串匹配的自动机(网上说的),主要作用是在一个长串中同时进行多个字符串的匹配 基础芝士: trie树(字典树) 烤馍片kmp单模字符串匹配 如果不会的建议去网上学一下(本篇讲解略 ...