数位dp从会打模板到不会打模板

打了几个数位$dp$,发现自己除了会打模板之外没有任何长进,遇到非模板题依然什么都不会

那么接下来这篇文章将介绍如何打模板(滑稽)

假设我们要处理$l----r$

采用记忆化搜索的方式,枚举$<=r$每一种情况,枚举每一位,然后再枚举$<=l-1$每一种情况,然后两个值相减即可

我们可以比较轻松打出一个模板

ll dfs(ll x,ll pre,ll lead,ll limit){if(x>pos) return 1;if(!limit&&f[x][pre]) return f[x][pre];ll ans=0;ll mx=limit?maxn[pos-x+1]:9;for(ll i=0;i<=mx;i++){if(????????) continue;if(lead&&i==0) ans+=dfs(x+1,-2,1,limit&&(i==mx));else ans+=dfs(x+1,i,0,limit&&(i==mx));}if(!limit&&!lead) f[x][pre]=ans;return ans;
}

解释一下$limit$是什么

假设有一个数

$1\   2\  3\  4\  5\  6$

$1\   2\  3\  ?\  ?\  ?$

我们枚举到第四位时最多枚举到$4$,

$1\   2\  3\  4\  5\  6$

$1\   2\  2\  ?\  ?\  ?$

这时我们枚举到第四位最多枚举到$9$

$limit$就是判断这个的

那么为什么要在$!limit$下才记忆化呢?

如果在所有情况下我们都记录f,那么假如之前枚举到$9$时你记录了一个答案,然后当前位有$limit$限制根本枚举不到$9$,你仍然用了这个f会出现错误

当然你这样记忆化也可以这样避免冲突

ll dfs(ll x,ll limit,ll tmp,ll d){if(f[x][limit][tmp][d]) return f[x][limit][tmp][d];………………    f[x][limit][tmp][d]=ans;return ans;
}

解释一下$lead$

处理前导0用的,

有的时候

$0\   0\  0\  4\  5\  6$

也被认为是合法的,这时处理可能会出现问题,$lead$特判特殊处理

那么我们看几道模板题

例题

windy数

Windy 定义了一种 Windy 数:不含前导零且相邻两个数字之差至少为2的正整数被称为 Windy 数。

Windy 想知道,在l和r 之间,包括l 和 r ,总共有多少个 Windy 数?

非常简单对不对

套模板

代码:

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define A 50
ll f[A][A],maxn[A];
ll pos=0,a,b;
ll dfs(ll x,ll pre,ll lead,ll limit){if(x>pos) return 1;if(!limit&&f[x][pre]) return f[x][pre];ll ans=0;ll mx=limit?maxn[pos-x+1]:9;
//    printf("mx=%lld\n",mx);for(ll i=0;i<=mx;i++){if(abs(pre-i)<2) continue;if(lead&&i==0) ans+=dfs(x+1,-2,1,limit&&(i==mx));else ans+=dfs(x+1,i,0,limit&&(i==mx));
//        printf("ans=%lld\n",ans);
    }if(!limit&&!lead) f[x][pre]=ans;return ans;
}
ll solve(ll x){pos=0;memset(f,0,sizeof(f));while(x){maxn[++pos]=x%10;x/=10;}ll ans=dfs(1,-2,1,1);return ans;
}
int main(){scanf("%lld%lld",&a,&b);swap(a,b);printf("%lld\n",solve(a)-solve(b-1));
}

不要62

$l---r$间数位上不含$4$且没有$62$相连的数个数

套模板

代码

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define A 50
ll f[A][A],maxn[A];
ll pos=0,a,b;
ll dfs(ll x,ll pre,ll limit){if(!x) return 1;if(!limit&&f[x][pre]) return f[x][pre];
//    printf("x=%lld pre=%lld  limit=%lld\n",x,pre,limit);ll ans=0;ll mx=limit?maxn[x]:9;for(ll i=0;i<=mx;i++){if(pre==6&&i==2) continue;if(i==4) continue;ans+=dfs(x-1,i,limit&&(i==mx));}
//    printf("ans=%lld\n",ans);if(!limit) f[x][pre]=ans;return ans;
}
ll solve(ll x){pos=0;memset(f,0,sizeof(f));while(x){maxn[++pos]=x%10;x/=10;}ll ans=dfs(pos,0,1);return ans;
}
int main(){a=233,b=233;while(a!=0&&b!=0){scanf("%lld%lld",&a,&b);if(a<b)swap(a,b);if(a==0||b==0){return 0;}printf("%lld\n",solve(a)-solve(b-1));}
}

手机号码

至少$3$个相邻的相同的数,$8$ $4$不能同时出现

套模板,注意下特判,不然会70到自闭

代码

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define A 30
ll f[A][A][A][2][2][2],pos[A];
ll tot=0,a,b;
ll dfs(ll x,ll prer,ll pre,ll limit,ll _4,ll _8,ll ok){if(_4&&_8) return 0;if(x>tot&&!ok) return 0;if(x>tot&&ok) return 1;if(!limit&&f[x][prer][pre][_4][_8][ok]) return f[x][prer][pre][_4][_8][ok];ll maxn=limit?pos[tot-x+1]:9;ll ans=0;for(ll i=0;i<=maxn;i++){if(x==1&&i==0) continue;if(_4&&i==8) continue;if(_8&&i==4) continue;if(pre==prer&&i==pre&&i==prer)ans+=dfs(x+1,pre,i,(limit&&i==maxn),(_4||i==4),(_8||i==8),1);else ans+=dfs(x+1,pre,i,limit&&(i==maxn),(_4||i==4),(_8||i==8),ok);
//        printf("ans=%lld i=%lld maxn=%lld\n",ans,i,maxn);
    }if(!limit)f[x][prer][pre][_4][_8][ok]=ans;return ans;
}
ll solve(ll x){tot=0;if(x<1e10) return 0;memset(f,0,sizeof(f));while(x){pos[++tot]=x%10;x/=10;}ll ans=dfs(1,0,0,1,0,0,0);return ans;
}
int main(){scanf("%lld%lld",&a,&b);if(a<b) swap(a,b);printf("%lld\n",solve(a)-solve(b-1));
}

花神的数论题

设 sum(i) 表示 i 的二进制表示中 1 的个数。给出一个正整数 N ,花神要问你派(Sum(i)),也就是 sum(1)—sum(N) 的乘积。

套模板,等等,我要套什么,

这个题不算很板,值得思考思考。

数位dp计算出所有二进制下sum个数,然后快速幂处理一下,

代码

#include<bits/stdc++.h>
using namespace std;
#define A 53
#define mod 10000007
#define ll long long
ll n,tot=0,sum=0;
ll f[A][2][A][A],ans[A],pos[A];
ll dfs(ll cur,ll up,ll tmp,ll d){if(!cur)return tmp==d;if(~f[cur][up][tmp][d])return f[cur][up][tmp][d];ll lim=up?pos[cur]:1;ll ret=0;for(ll i=0;i<=lim;i++)ret+=dfs(cur-1,up&&i==lim,tmp+(i==1),d);return f[cur][up][tmp][d]=ret;
}
ll meng(ll a,ll b){ll ret=1;while(b)ret=ret*(b&1?a:1)%mod,a=a*a%mod,b>>=1;return ret;
}
ll solve(ll x){sum=1;while(x){pos[++tot]=x&1;x>>=1;}for(ll i=1;i<=tot;i++){memset(f,-1,sizeof(f));ans[i]=dfs(tot,1,0,i);}for(ll i=1;i<=tot;i++){sum=(sum*max(meng(i,ans[i]),1ll))%mod;}return sum;
}
int main(){scanf("%lld",&n);printf("%lld\n",solve(n));
}

剩下一些题都不算很板

数数

淘金

方伯伯商场之旅

 

题解慢慢补充八

 

那么你现在会模板了吗?

转载于:https://www.cnblogs.com/znsbc-13/p/11328399.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/349748.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

javafx简单吗_JavaFX即将推出您附近的Android或iOS设备吗?

javafx简单吗已经有大新闻最近在世界上的JavaFX的关于JavaFX的是许多更多的组件开源&#xff0c;开源的广告在2012 JavaOne大会 。 在2月的开放源代码更新中 &#xff0c; Richard Bair编写了一份JavaFX项目表&#xff0c;该表在撰写本文时&#xff08;2013年2月11日&#xff0…

【webrtc】webrtc的rtp重传代码分析

pgm不太能用&#xff0c;没有想象中的可靠&#xff0c;重传机制貌似仍然使用组播重传&#xff0c;丢包率80%的网络感觉没啥改进&#xff0c;如果有所好转延迟估计也是个不小的问题。 后听说rtp也有nack机制&#xff0c;webrtc基于rtp实现了重传在一定程度上保证可靠性。 在各路…

了解java.nio.file.Path – 1

介绍 Java的最后几个发行版本&#xff0c;即Java 7&#xff0c;Java 8和即将到来的Java 9&#xff0c;具有许多功能&#xff0c;这些功能使Java开发人员的生活更加轻松。 &#xff08;我知道Java 9会使它变得更困难&#xff0c;但是只有在您采用新的范例时才可以。之后&#xf…

socket-01

对于所有的Web应用&#xff0c;本质上其实就是一个socket服务端&#xff0c;用户的浏览器其实就是一个socket客户端 转载于:https://www.cnblogs.com/yanhuaqiang/p/11329925.html

ActionScript3学习笔记2-包

在 ActionScript 3.0 中&#xff0c;包是用命名空间实现的&#xff0c;但包和命名空间并不同义。在声明包时&#xff0c; 可以隐式创建一个特殊类型的命名空间并保证它在编译时是已知的。显式创建的命名空间在 编译时不必是已知的。 下面的示例使用 package 指令来创建一个包含…

grep v grep_使用grep4j轻松测试分布式组件上的SLA

grep v grep因此&#xff0c;您的分布式体系结构如下图所示&#xff0c;您刚刚从企业那里收到了一项要求&#xff0c;以确保生产者发送并随后传输到下游系统&#xff08;消费者&#xff09;的消息的SLA必须快且永远不会慢于此。 400毫秒。 要求说&#xff1a; 从生产者发送到…

得到指定进程所有窗口。显示 影藏 置顶。

这里使用一个外挂程序测试&#xff0c;因为外挂程序没有做功能限制的处理 只是做了 窗口影藏。 全局变量 HWND hwnd[100]{0};int number0;DWORD Tpid0; 局部变量 char username[1028]; 先找到进程ID 1 HWND SelectPor() 2 { 3 bool isYesfalse; 4 string porcessName&q…

jquery部分方法

offset([coordinates]) 概述&#xff1a;获取匹配元素在当前视口的相对偏移。返回的对象包含两个整型属性&#xff1a;top 和 left。此方法只对可见元素有效。 比如&#xff0c;获取第二段的偏移&#xff1a; HTML 代码: <p>Hello</p><p>2nd Paragraph</p…

JUnit 5扩展模型的生命周期

JUnit5最终版本即将来临 &#xff08;当前是M4&#xff09;&#xff0c;我已经开始研究如何编写扩展。 在JUnit5中 &#xff0c;您没有使用Runners &#xff0c; Rules &#xff0c; ClassRules等&#xff0c;而是只有一个Extension API来实现自己的扩展。 JUnit5提供了多个接…

让IE6、IE7、IE8支持CSS3

我们都知道IE6&#xff0c;7并不支持CSS3的属性&#xff0c;IE8也不能很好的支持CSS3。但是有一个小脚本能够做到&#xff0c;它可以让IE支持 CSS3&#xff0c;包括&#xff1a;border-radius (rounded), box-shadow ( shadow), text-shadow等…… 如果你需要一个支持CSS3 的bo…

NOIP模拟测试16「Drink·blue·weed」

话说这次考试 Drink 非常棒的一道卡常练习题&#xff0c;适合练习卡常 真的很棒 前置卡常知识 1.char要比int快 char是最快的 输出putchar&#xff0c;输入getchar 在这个题快了7000豪 2.read 快读非常棒&#xff0c;让你变得更快&#xff0c;fread更棒&#xff0c;fread会爆炸…

Spring Boot Web Slice测试–示例

春天开机推出 测试切片而回&#xff0c;它已经采取了一些时间来解决它我的头&#xff0c;并探讨一些细微的差别。 背景 使用此功能的主要原因是减少样板。 考虑一个看起来像这样的控制器&#xff0c;仅适用于使用Kotlin编写的各种控制器。 RestController RequestMapping(&qu…

DECODE函数

DECODE函数相当于一条件语句(IF),它将输入数值与函数中的参数列表相比较&#xff0c;根据输入值返回一个对应值。函数的参数列表是由若干数值及其对应结果值组成的若干序偶形式。当然&#xff0c;如果未能与任何一个实参序偶匹配成功&#xff0c;则函数也有默认的返回值。区别于…

多线程练习

写两个线程&#xff0c;其中一个线程打印1-52&#xff0c;另一个线程打印A-Z&#xff0c;打印顺序应该是12A34B56C......5152Z。 该习题需要用到多线程通信的知识。 思路分析&#xff1a; 把打印数字的线程称为线程N&#xff0c;打印字母的线程称为线程L. 1.线程N完成打印后&am…

java jee curd_Java / JEE中的有效日志记录–映射的诊断上下文

java jee curd这一切始于当我和一位同事坐在一起解决一些应用程序问题时&#xff0c;当我注意到一些有趣的事情时。 他正在合并代码&#xff0c;我的眼睛吸引了此类“ org.apache.log4j.MDC”的注意。 这导致了以下发现&#xff1a; 什么是MDC&#xff1f; MDC代表“ 映射诊断…

Learning Cocos2d-x for WP8(7)——让Sprite动起来

C#(wp7)兄弟篇Learning Cocos2d-x for XNA&#xff08;7&#xff09;——让Sprite动起来 本讲将详细介绍Cocos2d-x游戏中动画Animate的创建方式&#xff0c;通过逐帧数组播放动画和创建动画集合播放动画&#xff0c;比较两者的异同&#xff0c;让Sprite动起来。 工程文件&#…

GWT的渐进式Web应用程序配方

渐进或不渐进… 如果您一段时间以来一直在设计或开发Web应用程序&#xff0c;那么您可能会遇到无数次“渐进式Web应用程序”一词&#xff0c;并且在未来几年内可能会这样做。 您可能想知道PWA的确切定义是什么&#xff0c;如何识别PWA&#xff0c;以及如何构建PWA。 根据字典&a…

问题 1047: [编程入门]报数问题

题目描述有n人围成一圈&#xff0c;顺序排号。从第1个人开始报数&#xff08;从1到3报数&#xff09;&#xff0c;凡报到3的人退出圈子&#xff0c;问最后留下的是原来的第几号的那位。输入初始人数n输出最后一人的初始编号样例输入3 样例输出2分析&#xff1a;因为每次报3都会…

NOIP模拟测试17「入阵曲·将军令·星空」

入阵曲 题解 应用了一种美妙移项思想&#xff0c; 我们先考虑在一维上的做法 维护前缀和$(sum[r]-sum[l-1])\%k0$可以转化为 $sum[r]\% ksum[l-1]\%k$开个桶维护一下即可 然后拓展到二维上 把两行之间所有行拍扁看作一维上的区间&#xff0c; 我们枚举两行和行之间所有列开个桶…

理解sizeof

1、sizeof返回的是字节个数&#xff0c;内存编址的最小单元是字节。因此&#xff0c;空对象&#xff0c;bool值占用的内存也是一个字节。 2、可以对哪些东西求sizeof ? a、对象和类型。如int a; sizeof(a)&#xff0c; sizeof(int)&#xff0c;二者是等价的。同一类型的对象&a…