题目链接

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

假设我们都移到下标\(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. GCC的符号可见性——解决多个库同名符号冲突问题

    引用自:https://github.com/wwbmmm/blog/wiki/gcc_visibility 问题 最近项目遇到一些问题,场景如下 主程序依赖了两个库libA的funcA函数和libB ...

  2. centos7使用haproxy1.7.5实现反向代理负载均衡实战

    使用haproxy实现反向代理负载均衡实战环境准备:两台虚拟机 # yum install -y gcc glibc gcc-c++ make screen tree lrzsz node1源码编译安 ...

  3. mysql忘记root密码的处理方式

    1.停用mysql服务 service mysqld stop 2.修改my.cnf    利用vim命令打开mysql配置文件my.cnf 添加skip-grant-tables,添加完成后,执行w ...

  4. Java Map 键值对排序 按key排序和按Value排序

    一.理论准备 Map是键值对的集合接口,它的实现类主要包括:HashMap,TreeMap,Hashtable以及LinkedHashMap等. TreeMap:基于红黑树(Red-Black tre ...

  5. matplotlib画堆叠条形图

    import matplotlib.pyplot as plt%matplotlib inlineplt.style.use('ggplot') plt.style.use("ggplot& ...

  6. JavaScript对象之深度克隆

    也不知道从什么时候开始,前端圈冒出了个新词:对象深度克隆.看起来好像很高大上的样子,实际上并不新鲜,在我们的实际项目开发中,你可能早已用到,只不过由于汉字的博大精深,有些原本很简单的事物被一些看似专业 ...

  7. 自己的vim配置

    nmap <F11> :source ~/.vimrc<CR> "n 普通模式 F11映射为 :source ~/.vimrc winpos 5 5 "wi ...

  8. kmp算法专题总结

    next数组的含义:next[i]表示以字符串s的第i个字符为结尾的后缀与s前缀匹配的长度 next数组也可以当做fail数组,即当模式串s[j]与串t[i]不匹配时,只要将j转换到next[j]继续 ...

  9. pytest八:skip 跳过用例

    这是一个快速指南,介绍如何在不同情况下跳过模块中的测试1.无条件地跳过模块中的所有测试:pytestmark = pytest.mark.skip("all tests still WIP& ...

  10. js获取当前有效样式

      js获取有效样式   节点.currentStyle["属性名"]        兼容ie方法(只有ie有效) getcomputedStyle(节点)["属性名&q ...