题面

SDOI2013 淘金

有一个 \(X\)、\(Y\) 轴坐标范围为 \(1\sim n\) 的范围的方阵,每个点上有块黄金。一阵风来 \((x,y)\) 上的黄金到了 \((f(x),f(y))\),\(f(x)\) 为 \(x\) 各位上数字的乘积,如果黄金飘出方阵就没了。求在 \(k\) 个格子上采集黄金最多可以采集的黄金数。

数据范围:\(1\le n\le 10^{12}\),\(k\le \min(n^2,10^5)\)。


蒟蒻语

蒟蒻跟着 \(\it srf\) 大师的日报来做这题,然后发现自己的裸代码跑得比题解都快,方法也比较神奇,于是来跟巨佬们讲讲。


蒟蒻解

首先众所周知,对于 \(1\le i\le 10^{12}\),\(f(i)\) 只有 \(8282\) 种,所以可以先找出这 \(8282\) 种 \(f(i)\),蒟蒻有三种方法:set、枚举质因数和数位 dp。

为了优化可以用数位 dp(这是第一次数位 dp),正好求出 \(1\sim n\) 的所有 \(f(i)\not=0\):

int dn,d[13],cnt=0;
ll t[N]; unordered_map<ll,int> nt;
bool vis[13][N];
void init(int w,ll now,bool ava){ //w:位 now:乘积 ava=true:可以自由选择数字
if(!~w){if(!nt.count(now)) nt[t[cnt]=now]=cnt,cnt++;return;}
if(vis[w][nt[now]]) return;
vis[w][nt[now]]=true;
int up=ava?9:d[w];
for(int i=1;i<=up;i++) init(w-1,now*i,true||i<up);
}
void INIT(){
while(n) d[dn++]=n%10,n/=10;
for(int i=0;i<=dn;i++) init(i-1,1,i<dn);
}

然后是最重要的部分:求每种 \(f(i)\) 有多少个 \(i\)。

蒟蒻原来的做法是对每种 \(f(i)\) 来一次数位 dp,瞬间被 TLE 打脸,于是蒟蒻想出了一个不同于别的巨佬的做法:

记 \(f_{w,now}\) 表示到第 \(w\) 位,剩下 \(w\) 位乘积为 \(now\) 的方案数。

用记忆化搜索转移,dp 中会用到除法。

ll f[13][N];
ll dp(int w,int now,bool ava){ //w:位 now:剩下期望乘积 ava=true:可以自由选择数字
if(!~w) return t[now]==1;
if(ava&&~f[w][now]) return f[w][now];
int up=ava?9:d[w]; ll res=0;
for(int i=1;i<=up;i++)if(t[now]%i==0)
res+=dp(w-1,nt[t[now]/i],ava||i<up);
if(ava) f[w][now]=res;
return res;
}
void DP(){
memset(f,-1,sizeof f);
for(int i=0;i<cnt;i++)
for(int j=1;j<=dn;j++) a[i]+=dp(j-1,i,j<dn); // ai 是每种 fi 出现次数
}

然后给 \(a_i\) 排个序,用堆维护找最大 \(k\) 乘积即可,考虑到相乘可能会爆 long long,蒟蒻用了除法。

时间复杂度 \(\Theta(8282*12*10)\)。


代码

很明显蒟蒻的题解都是废话,只好放代码了

#include <bits/stdc++.h>
using namespace std; //Start
typedef long long ll;
typedef double db;
#define mp(a,b) make_pair(a,b)
#define x first
#define y second
#define be(a) a.begin()
#define en(a) a.end()
#define sz(a) int((a).size())
#define pb(a) push_back(a)
const int inf=0x3f3f3f3f;
const ll INF=0x3f3f3f3f3f3f3f3f; //Data
const int N=8282;
const int mod=1e9+7;
ll n,a[N];
int k,nex[N],ans;
struct cmp{
bool operator()(pair<int,int> p,pair<int,int> q){
return 1.*a[p.x]/a[q.y]<1.*a[q.x]/a[p.y];
}
};
priority_queue<pair<int,int>,vector<pair<int,int>>,cmp> q; //Digitdp
int dn,d[13],cnt=0;
ll t[N]; unordered_map<ll,int> nt;
bool vis[13][N];
void init(int w,ll now,bool ava){
if(!~w){if(!nt.count(now)) nt[t[cnt]=now]=cnt,cnt++;return;}
if(vis[w][nt[now]]) return;
vis[w][nt[now]]=true;
int up=ava?9:d[w];
for(int i=1;i<=up;i++) init(w-1,now*i,true||i<up);
}
void INIT(){
while(n) d[dn++]=n%10,n/=10;
for(int i=0;i<=dn;i++) init(i-1,1,i<dn);
}
ll f[13][N];
ll dp(int w,int now,bool ava){
if(!~w) return t[now]==1;
if(ava&&~f[w][now]) return f[w][now];
int up=ava?9:d[w]; ll res=0;
for(int i=1;i<=up;i++)if(t[now]%i==0)
res+=dp(w-1,nt[t[now]/i],ava||i<up);
if(ava) f[w][now]=res;
return res;
}
void DP(){
memset(f,-1,sizeof f);
for(int i=0;i<cnt;i++)
for(int j=1;j<=dn;j++) a[i]+=dp(j-1,i,j<dn);
} //Main
int main(){
ios::sync_with_stdio(0);
cin.tie(0),cout.tie(0);
cin>>n>>k,INIT(),DP();
sort(a+0,a+cnt,[&](ll p,ll q){return p>q;});
for(int i=0;i<cnt;i++) nex[i]=0,q.push(mp(i,nex[i]++));
while(k--&&sz(q)){
pair<int,int> u=q.top(); q.pop();
(ans+=(a[u.x]%mod)*(a[u.y]%mod)%mod)%=mod;
if(nex[u.x]<cnt) u.y=nex[u.x]++,q.push(u);
}
cout<<ans<<'\n';
return 0;
}

祝大家学习愉快!

题解-SDOI2013 淘金的更多相关文章

  1. Bzoj 3131 [Sdoi2013]淘金 题解

    3131: [Sdoi2013]淘金 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 733  Solved: 363[Submit][Status][ ...

  2. [Bzoj3131][Sdoi2013]淘金(数位dp)(优先队列)

    3131: [Sdoi2013]淘金 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 847  Solved: 423[Submit][Status][ ...

  3. [BZOJ3131] [Sdoi2013]淘金

    [BZOJ3131] [Sdoi2013]淘金 Description 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐 ...

  4. 【bzoj 3131】[Sdoi2013]淘金

    Description 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐标点上均有一块金子,共N*N块.    一阵风吹 ...

  5. SDOI2013 淘金

    题目链接:戳我 昨天做的题了,今天补一发题解. 是一个比较奇怪的数位DP.详细的我还是写代码注释里好了,感觉直接说不好描述. 代码如下: #include<iostream> #inclu ...

  6. bzoj 3131 [Sdoi2013]淘金(数位dp)

    题目描述 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐标点上均有一块金子,共N*N块. 一阵风吹过,金子的位置发生了 ...

  7. P3303 [SDOI2013]淘金

    题目描述 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐标点上均有一块金子,共N*N块. 一阵风吹过,金子的位置发生了 ...

  8. bzoj 3131: [Sdoi2013]淘金

    #include<cstdio> #include<iostream> #include<queue> #include<algorithm> #def ...

  9. bzoj 3131 [Sdoi2013]淘金(数位DP+优先队列)

    Description 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐标点上均有一块金子,共N*N块.    一阵风吹 ...

随机推荐

  1. Cephfs 操作输出到日志查询系统

    前言 文件系统当中如果某些文件不见了,有什么办法判断是删除了还是自己不见了,这个就需要去日志里面定位了,通常情况下是去翻日志,而日志是会进行压缩的,并且查找起来非常的不方便,还有可能并没有开启 这个时 ...

  2. python any,call,init,下划线知识汇总

    python补充 any() [来自菜鸟教程] any() 函数用于判断给定的可迭代参数 iterable 是否全部为 False,则返回 False,如果有一个为 True,则返回 True. 元素 ...

  3. [web安全原理分析]-文件上传漏洞基础

    简介 前端JS过滤绕过 待更新... 文件名过滤绕过 待更新 Content-type过滤绕过 Content-Type用于定义网络文件的类型和网页编码,用来告诉文件接收方以什么形式.什么编码读取这个 ...

  4. Hbase启动报java异常

    在conf文件夹下的hbase-env.sh文件中的j添加ava_home的环境变量, ******************************************************** ...

  5. 这行代码告诉你!为什么你地下城与勇士(DNF)的装备强化老是失败?

    模拟地下城与勇士(DNF)的装备强化 tip1: DNF装备强化在+1-+3 不会失败: +4-+7,失败后物品原有强化等级降低1级: +8-+10,失败后掉3级: 10上11或以上就爆了. tip2 ...

  6. node.js报错:Cannot find module 'xxx'的解决办法

    从别处拷贝一份node.js项目,如图 控制台启动 nodemon index.js 后报错:Cannot find module xxx.删除node_modules,在启动时仍提示Cannot f ...

  7. ABBYY FineReader 14新建任务窗口给我们哪些帮助?

    当您启动ABBYY FineReader时, 新任务 将打开一个窗口,在其中您可以轻松打开.扫描.创建或对比文档. 如果您没有看到此 内置任务 窗口(比如,如果您关闭了该窗口,或者您通过在 Windo ...

  8. MathType如何对齐公式

    作为强大的公式编辑器,MathType为我们的学习.工作带来了极大的便利.比如在写论文时,有了它,就可以轻松就把论文里的公式码完:老师在编写试卷时,利用它,可以快速编写出一份试卷.那么在编写公式时,也 ...

  9. 如何查看CDR文件是出自哪个版本?

    如何才能知道某个cdr文件用哪个版本的CorelDRAW软件打开?网上CorelDRAW软件有很多版本,都不知该下哪个了?这是我听到大家问道最多的问题,这是因为CDR低版本软件打不开高版本文件. 方法 ...

  10. 小程序ui自动化(一),用uiAutormatorViewer定位元素失败,如何解决

    1.定位元素 用android ADT自带工具:uiAutormatorViewer,会报如下错误 可能是环境与手机不兼容 可以用以下方法解决:(参考:https://blog.csdn.net/qq ...