bzoj3131
这是一道很好也很烦的综合题……
首先我们肯定要先把f(i)处理出来这是毫无疑问的
我们要求出数位乘积为now的个数,首先是空间上的问题
直接肯定会爆空间,不难发现
乘积的质因数只有2,3,5,7,并且指数也不是特别大
暴力可得到不同的乘积最多只有15000不到
然后我们就可以对其离散化然后数位dp
dp完之后,对于点(p,q),这上面的金子个数是sum(p)*sum(q)个 (sum表示数位乘积为p的数的个数)
然后我们要求金子前k多的点,当状态太多无法算出所有状态求最优值时,我们常常用堆来维护
首先我们对sum排序(假定降序),对于每个乘积x,设乘积x离散化后对应的编号为w(x)
在x行上金子最多的点一定是sum[w(x)]*sum[1],次大的点一定是sum[w(x)]*sum[2]……
然后我们对当前这n行上最多的点维护一个大根堆
当我们每次将堆顶的点取出时,堆顶所在行的下一大的点可能是之后选取结果产生影响,
因此我们要将这行下一大的点加入堆,一共只要弹k次,所以复杂度为O(klogt)
const maxn=;
mo=; type arr=array[..maxn] of int64; var a,sum,h,loc,num:arr;
b:array[..] of int64;
f:array[..,..maxn] of int64;
n,ans,x:int64;
c,m,t,i:longint; procedure swap(var a,b:int64);
var c:int64;
begin
c:=a;
a:=b;
b:=c;
end; procedure sort(l,r:longint;var a:arr);
var i,j:longint;x,y:int64;
begin
i:=l;
j:=r;
x:=a[(l+r) shr ];
repeat
while a[i]<x do inc(i);
while x<a[j] do dec(j);
if i<=j then
begin
swap(a[i],a[j]);
inc(i);
dec(j);
end;
until i>j;
if i<r then sort(i,r,a);
if j>l then sort(l,j,a);
end; procedure prepare;
var i,j,k,l:int64;
begin
i:=;
while i<=n do
begin
j:=i;
while j<=n do
begin
k:=j;
while k<=n do
begin
l:=k;
while l<=n do
begin
inc(m);
a[m]:=l;
l:=l*;
end;
k:=k*;
end;
j:=j*;
end;
i:=i*;
end;
sort(,m,a);
end; procedure work;
begin
t:=;
x:=n;
while x<> do
begin
inc(t);
b[t]:=x mod ;
x:=x div ;
end;
end; function find(l,r:longint;x:int64):longint;
var mid:longint;
begin
while l<r do
begin
mid:=(l+r)shr ;
if a[mid]=x then exit(mid);
if a[mid]<x then l:=mid+ else r:=mid-;
end;
if a[l]=x then exit(l);
exit();
end; procedure count;
var i,j,k:longint;
now,y:int64;
begin
work;
f[][]:=; //f[i,j]表示到第i位,乘积为j(j是离散化后的排名)的方案数
for i:= to t- do
for j:= to m do
for k:= to do
if (a[j] mod k=) then
f[i,j]:=f[i,j]+f[i-,find(,j,a[j] div k)]; now:=;
for i:=t downto do
begin
for j:= to b[i]- do //肯定不可能是0
for k:= to m do
if (a[k]>=now*j) and ((a[k] div now) mod j=) and (a[k] mod now=) then
sum[k]:=sum[k]+f[i-,find(,k,a[k] div (now*j))];
now:=now*b[i];
if now= then break;
end;
if now<> then inc(sum[find(,m,now)]);
for i:= to m do
for j:= to t- do
sum[i]:=sum[i]+f[j,i];
end; procedure sift(i,n:longint);
var j:longint;
x:int64;
begin
x:=h[i];
j:=i shl ;
while j<=n do
begin
if (j<n) and (h[j]<h[j+]) then inc(j);
if x>=h[j] then exit
else begin
swap(h[i],h[j]);
swap(loc[i],loc[j]);
swap(num[i],num[j]);
i:=j;
j:=j shl ;
end;
end;
end; begin
readln(n,c);
prepare;
work;
count;
sort(,m,sum); //为了统一形式,这里是升序
for i:= to m do
begin
h[i]:=sum[i]*sum[m]; //当前第i行上最大的点
loc[i]:=m;
num[i]:=i; //所代表的行
end;
for i:=m downto do
sift(i,m);
ans:=;
for i:= to c do
begin
ans:=(ans+(h[] mod mo)) mod mo;
dec(loc[]); //加入这行下一大的点
h[]:=sum[num[]]*sum[loc[]];
sift(,m);
end;
writeln(ans);
end.
bzoj3131的更多相关文章
- [BZOJ3131] [Sdoi2013]淘金
[BZOJ3131] [Sdoi2013]淘金 Description 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1..N.初始的时候,所有的整数坐 ...
- bzoj千题计划268:bzoj3131: [Sdoi2013]淘金
http://www.lydsy.com/JudgeOnline/problem.php?id=3131 如果已知 s[i]=j 表示有j个<=n数的数码乘积=i 那么就会有 s[a1]*s[a ...
- [您有新的未分配科技点]数位DP:从板子到基础(例题 bzoj1026 windy数 bzoj3131 淘金)
只会统计数位个数或者某种”符合简单规律”的数并不够……我们需要更多的套路和应用 数位dp中常用的思想是“分类讨论”思想.下面我们就看一道典型的分类讨论例题 1026: [SCOI2009]windy数 ...
- 【数位dp】bzoj3131: [Sdoi2013]淘金
思路比较自然,但我要是考场上写估计会写挂:好像被什么不得了的细节苟住了?…… Description 小Z在玩一个叫做<淘金者>的游戏.游戏的世界是一个二维坐标.X轴.Y轴坐标范围均为1. ...
- [Bzoj3131][Sdoi2013]淘金(数位dp)(优先队列)
3131: [Sdoi2013]淘金 Time Limit: 30 Sec Memory Limit: 256 MBSubmit: 847 Solved: 423[Submit][Status][ ...
- [bzoj3131]淘金[sdoi2013][数位DP]
求出每个数i可以被转移到的数目$f[i]$,则点$(i,j)$中的金子数目为$f[i]*f[j]$,我们就可以用优先队列求解前$k$大. 首先所有的积数目在$10^4$左右,可以先Dfs搜索出所有的数 ...
- bzoj2757
非常神的数位dp,我调了几乎一天首先和bzoj3131类似,乘积是可以预处理出来的,注意这里乘积有一个表示的技巧因为这里质因数只有2,3,5,7,所以我们可以表示成2^a*3^b*5^c*7^d,也就 ...
- 【Richard 的刷(水)题记录】
大概想了想,还是有个记录比较好. 9/24 网络流一日游: 最大流:bzoj1711[Usaco2007 Open]Dining 拆点 BZOJ 3993 Sdoi2015 星际战争 二分 P.S.这 ...
- OI动态规划&&优化 简单学习笔记
持续更新!! DP的难点主要分为两类,一类以状态设计为难点,一类以转移的优化为难点. DP的类型 序列DP [例题]BZOJ2298 problem a 数位DP 常用来统计或者查找一个区间满足条件的 ...
随机推荐
- ios paper for facebook 使用第三方库
facebook paper使用的第三方库 Facebook Paper使用的第三方库 第三方库名 简介 链接 ACE code editor https://github.com/ajaxorg/a ...
- NewtonSoft.json 序列化和反序列化实例
在百度 API Store 找个旅游的 API 来当成本次 Demo 的例子 接口地址:http://apis.baidu.com/apistore/attractions/spot AIPKEY: ...
- C++数组(指针)作为函数参数
本文的学习内容参考:http://blog.csdn.net/wwdlk/article/details/6322843 1.当用数组名作为函数参数时,函数的实参和形参都应为数组名(或者指针): Ex ...
- 浅析JAVA设计模式(三)
4.接口隔离原则: ISP(Interface Segregation Principle) 客户端不应该依赖它不需要的接口,或者说类的依赖的关系应该建立在最小的接口上.举个例子,直接上代码: 1 ...
- IOS分类(Category)
分类(Category):拓展原有类的方法,而不影响原有类的子类,分类中不能创建成员变量. 分类的使用: 1.创建分类: 如图点击(File)选择(New)->(File).. 或者使用快捷键c ...
- IE8+等兼容、360调用webkit内核小记
首先是处理IE8.9等的兼容问题,注意以下几点: 1,尽可能严格要求自己使用w3c推荐的方式编写html/css 2,在html页面顶部添加<!DOCHTML html>,不清楚请查看参考 ...
- H5小内容(三)
Canvas(画布) 基本内容 简单来说,HTML5提供的新元素<canvas> Canvas在HTML页面提供画布的功能 在画布中绘制各种图形 C ...
- 网站加载有商务通、商桥,定义js函数触发快商通代码
有的网站已经加载了商务通.商桥的,前期定义了js函数 触发商务通.商桥代码的,可以重新定义新的函数对之前的函数进行覆盖,其 js代码为: var domain = document.domain; / ...
- MYSQL数据库备份与恢复
mysqldump -h主机名 -P端口 -u用户名 -p密码 (–database) 数据库名 > 文件名.sql 备份MySQL数据库的命令 mysqldump -hhostname -u ...
- C# dataGridView不显示默认行的解决办法
当页面只有一个dataGirdView时,调用From的Activated函数,在Activated函数里调用以下两个函数,可清除默认选择行 private void From_Activated(o ...