数位dp有着很明显的特点,一般来说是给定区间[l,r]求满足某种条件区间中的数有多少个

朴素解法一般是O(n)的而n往往很大(10^8起步)

这时候我们就要想办法优化,于是就有了数位dp

数位有两个基本的原则

  1. 对于区间数的个数,我们转化为前缀和做(即ans=sum(r)-sum(l-1))

  2. 逐位确定

我认为第二条很关键,可以说是数位dp的精髓

一般来说数位dp分两步

  1. 打表 形如f[i,j]到有i位且最高位为j的满足条件的个数

  2. 统计前缀和

统计前缀和我们需要用到一个非常重要的结论

对于任意一个小于n的数,从高位到低位,必然出现了某一位小于n的那一位

这样我们就可以不受n的限制,用数位开始处理

以bzoj1026这道经典的题目为例

 var f:array[..,..] of longint;
    d:array[..] of longint;
    l,r,t,i,j,k:longint; function get(n:longint):longint;
  var i,s,x,j:longint;
  begin
    if (n>=) and (n<=) then exit(n);
    fillchar(d,sizeof(d),);
    s:=;
    while n<> do
    begin
      inc(s);
      d[s]:=n mod ;
      n:=n div ;
    end;
    get:=;
    for i:= to s- do   //统计位数小于n的位数的满足条件的数目
      for j:= to do
        get:=get+f[i,j];
//注意这里是不能直接加前导为0的数组,因为我们在计算前导为0的时候,实际上舍掉了后一位为0~的情况
    for i:=s downto do  //从高到低逐位统计位数为n的位数且小于n的满足条件的个数
    begin
      if i<> then x:=d[i]- else x:=d[i]; //小细节,注意n本身也可能是
      for j:= to x do   //当前位小于n的这位的满足条件的数
      begin
        if (i=s) and (j=) then continue;
        if (s=i) or (abs(j-d[i+])>=) then
          get:=get+f[i,j];
      end;
      if (s<>i) and (abs(d[i+]-d[i])<) then break;  //如果逐位统计n本身出现了不满足的情况,那显然要直接退出
    end;
  end; begin
  readln(l,r);
  t:=trunc(ln(r)/ln())+;
  for i:= to do   //题目的特殊要求
    f[,i]:=;
  for i:= to t do  //计算f[i,j]
  begin
    for j:= to do  //注意要包含前导为0的状况 
      for k:= to do
        if abs(j-k)>= then   
          f[i,j]:=f[i,j]+f[i-,k];
  end;
  writeln(get(r)-get(l-));
end.

bzoj1026

(话说我第一次拍这道题花了3h,实在太渣……)

当然我们也不能拘泥于这种套路,我觉得

当表打了没什么用的时候我们可以直接逐位计算,如poj3286(统计0的个数)

这里我们讨论,每一位对0的贡献度

 var d:array[..] of int64;
    i:longint;
    l,r:int64; function count(n:int64):int64;
  var i,t:longint;
      x,y:int64;
  begin
    for i:= to do
      if (n<d[i]) then break;
    t:=i-;
    count:=;
    for i:= to t do
    begin
      x:=n div d[i];    //当前位前面所组成的数
      y:=n mod d[i-];  //当前位后面所组成的数
      if (n div d[i-] mod )= then  //当前位
        count:=count+d[i-]*(x-)+y+   
//如果是x0y的情况,是第k位时,我们分2种情况讨论
  当小于n的数是p0q的,p∈[,x-],那么这个位上的0可以贡献(x-)*^(k-)
  当这个小于等于n的数是x0q, q∈[,y]那么这个位上的0可以贡献y+
      else count:=count+d[i-]*x;
//如果当前位不是0,那显然比n小的数中肯定存在当前位为0的;
设数为p0q, p∈[,x], q∈[,^(k-)-] 因为当前位已经小于则显然可以贡献x*^(k-)
    end;
  end; begin
  d[]:=;
  for i:= to do
    d[i]:=d[i-]*;
  readln(l,r);
  while (l<>-) do
  begin
    if l= then writeln(count(r))
    else writeln(count(r)-count(l-));
    readln(l,r);
  end;
end.

poj3286

数位dp需要大量的思考,有时候看起来对的实际上是错的

但由于这种题目暴力,数据都非常好弄

所以一定要孜孜不倦的对拍,恩恩

初探数位dp的更多相关文章

  1. 数位dp入门 hdu2089 不要62

    数位dp入门 hdu2089 不要62 题意: 给定一个区间[n,m] (0< n ≤ m<1000000),找出不含4和'62'的数的个数 (ps:开始以为直接暴力可以..貌似可以,但是 ...

  2. [bzoj1833][ZJOI2010]count 数字计数——数位dp

    题目: (传送门)[http://www.lydsy.com/JudgeOnline/problem.php?id=1833] 题解: 第一次接触数位dp,真的是恶心. 首先翻阅了很多很多一维dp,因 ...

  3. 数位dp初探

    我这种蒟蒻就一直不会写数位dp.. 于是开了个坑.. 1833: [ZJOI2010]count 数字计数 这道被KPM大爷说是入门题..嗯似乎找找规律然后减掉0的情况后乱搞就可以了..(但是还是写了 ...

  4. 数位类统计问题--数位DP

    有一类与数位有关的区间统计问题.这类问题往往具有比较浓厚的数学味道,无法暴力求解,需要在数位上进行递推等操作.这类问题往往需要一些预处理,这就用到了数位DP. 本文地址:http://www.cnbl ...

  5. 动态规划——数位dp

    通过先前在<动态规划——背包问题>中关于动态规划的初探,我们其实可以看到,动态规划其实不是像凸包.扩展欧几里得等是具体的算法,而是一种在解决问题中决策的思想.在不同的题目中,我们都需要根据 ...

  6. 【BZOJ1662】[Usaco2006 Nov]Round Numbers 圆环数 数位DP

    [BZOJ1662][Usaco2006 Nov]Round Numbers 圆环数 Description 正如你所知,奶牛们没有手指以至于不能玩"石头剪刀布"来任意地决定例如谁 ...

  7. bzoj1026数位dp

    基础的数位dp 但是ce了一发,(abs难道不是cmath里的吗?改成bits/stdc++.h就过了) #include <bits/stdc++.h> using namespace ...

  8. uva12063数位dp

    辣鸡军训毁我青春!!! 因为在军训,导致很长时间都只能看书yy题目,而不能溜到机房鏼题 于是在猫大的帮助下我发现这道习题是数位dp 然后想起之前讲dp的时候一直在补作业所以没怎么写,然后就试了试 果然 ...

  9. HDU2089 不要62[数位DP]

    不要62 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submis ...

随机推荐

  1. C盘清理大作战

    C盘会随着使用时间慢慢变满(即使你不在C盘装程序),下面就记录几个C盘清理的方法: 1.使用清理软件清理C盘(360卫视,腾讯管家) 2.转移虚拟内存:计算机右键属性——高级管理设置——高级——性能- ...

  2. asp.net mvc 伪静态路由配置

    asp.net mvc实现伪静态路由必须按如下方式设置好,才能访问 .htm 或者.html页面 C:\WINDOWS\Microsoft.NET\Framework\v4.0.30319\aspne ...

  3. having count(*) > 1

    select phone from aa group by phone having count(*) > 1 后面的having子句是什么意思啊? 以phone分组,分组后每组里面phone出 ...

  4. 【趟坑】公共引用的jar包 pom的配置方法

    http://www.cnblogs.com/viewcozy/p/4789877.html 接上一篇 ,内部moudule生成jar的方式 上一篇已经实现了,想把jar作为公共的部分让任何项目都可以 ...

  5. 传统ASP.NET开发和MVC的设计思想

    传统ASP.NET开发 第一步:客户端请求服务器: 第二步:服务器从数据库取得数据处理后响应给客户端页面. MVC的设计思想 第一步:客户端请求控制器(里面的一个方法): 第二步:控制器从数据库里取得 ...

  6. 实验五 Java网络编程及安全

    北京电子科技学院 实      验      报      告 课程:移动平台应用开发实践  班级:201592   姓名:曾俊宏  学号:20159210 成绩:___________  指导老师: ...

  7. 改变navigationbar 标题颜色

    navigationController.navigationBar.titleTextAttributes=@{NSForegroundColorAttributeName:[UIColor yel ...

  8. ExtJs 自定义Vtype验证

    最近公司开发项目在用ExtJs,碰到验证的就大概的总结了一些常用的验证.自定义的验证主要有两种方式:一种是单字段的自定义验证,另一种是多字段间的验证.对于单字段的验证主要通过regex配置项指定自定义 ...

  9. 通过分析WP的代码来学习PHP。1

    下载了WP的代码,并且应用到了网站上面,现在也在正常的运行中,地址是:www.freealgorithm.tk .具体的申请过程就不赘述了,学习WP的代码. 他的目录结构就不看了,可以下载同名文件我会 ...

  10. Ehcache 整合Spring 使用页面、对象缓存(转载)

    Ehcache在很多项目中都出现过,用法也比较简单.一般的加些配置就可以了,而且Ehcache可以对页面.对象.数据进行缓存,同时支持集群/分布式缓存.如果整合Spring.Hibernate也非常的 ...