一道十分经典的数位DP的题目。

dp[i][j]表示最高位是数字i,连同最高位在内共有j位。注意边界的初始化。

接下来就是区间划分,特殊情况处理.....对了,如果不知道自己的方法是否正确,可以写一个测试函数:

bool check(int n){  //判断n是否符合条件
	int a[20];
	int c=0;
	while(n>0){
		a[c++]=n%10;
		n/=10;
	}
	for(int i=0;i<c;++i) if(a[i]==4) return false;
	for(int i=0;i<c-1;++i){
		if(a[i]*10+a[i+1]==26) return false;
	}
	return true;
}
inline int solve(int n){  //可以求得0~n之间有多少符合条件的数
	int ans=0;
	for(int i=0;i<=n;++i)
	 	if(check(i)) ++ans;
	return ans;
}

AC代码:

#include<cstdio>
#include<cstring>
#include<cmath>
const int maxn=20;
int dp[10][maxn];
void deal(){
	memset(dp,0,sizeof(dp));
	//初始化边界
	for(int i=0;i<10;++i){
		if(i==4) continue;
		else dp[i][1]=1;
	}
	for(int i=2;i<10;++i){ //length
		for(int j=0;j<10;++j){
			if(j==4) continue;
			for(int k=0;k<10;++k){
				dp[j][i]+=dp[k][i-1];
			}
		}
		dp[6][i]-=dp[2][i-1];
	}
}
inline int solve(int pre,int n,int len){
	if(pre==62||pre%10==4||pre/10==4) return 0;
	if(len==1){
		int ans=0;
		for(int i=0;i<=n;++i)
			if(i==4) continue;
			else ++ans;
		if(pre%10==6&&n>=2) --ans;
		return ans;
	}
	int ans=0;
	int m=(int)pow(10,len-1);
	int up=n/m;
	for(int i=0;i<up;++i){
		ans+=dp[i][len];
	}
	if(pre%10==6&&up>2) ans-=dp[2][len];
	return ans+solve(pre%10*10+up,n%m,len-1);
}
inline int getlen(int n){
	if(n==0) return 1;
	int c=0;
	while(n>0){
		n/=10;
		++c;
	}
	return c;
}
int main(){
	deal();
	int l,r;
	while(scanf("%d%d",&l,&r)!=EOF&&l&&r){
		int a=solve(0,r,getlen(r));
		int b=solve(0,l-1,getlen(l-1));
		//printf("%d-%d=%d\n",a,b,a-b);
		printf("%d\n",a-b);
	}
	return 0;
}

如有不当之处欢迎指出!

hdu2089 不要62--经典数位DP的更多相关文章

  1. ☆ [HDU2089] 不要62「数位DP」

    类型:数位DP 传送门:>Here< 题意:问区间$[n,m]$的数字中,不含4以及62的数字总数 解题思路 数位DP入门题 先考虑一般的暴力做法,整个区间扫一遍,判断每个数是否合法并累计 ...

  2. HDU2089 ------不要62(数位dp)

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

  3. HDU2089 不要62 题解 数位DP

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2089 题目大意:求区间 \([l,r]\) 范围内不包含数字"4"且不包含连续的数 ...

  4. hdu2089不要62(数位dp)

    #include <stdio.h> #include <string.h> ][]; ]; /* dp[i][0] 不含62,4 dp[i][1] 2开头 dp[i][2] ...

  5. HDU 2089 不要62(数位dp入门)

    题意:统计区间 [a,b] 中不含 4 和 62 的数字有多少个. 题解:这是数位DP的入门题了,首先要理解数DP的原理,DP[i][j]:代表第i位的第j值,举个栗子:如4715   数位数是从右向 ...

  6. HDU 2089 不要62 【数位dp】

    <题目链接> 不要62 Problem Description 杭州人称那些傻乎乎粘嗒嗒的人为62(音:laoer).杭州交通管理局经常会扩充一些的士车牌照,新近出来一个好消息,以后上牌照 ...

  7. HDU 2089 不要62(数位dp模板题)

    http://acm.hdu.edu.cn/showproblem.php?pid=2089 题意:求区间内不包含4和连续62的数的个数. 思路: 简单的数位dp模板题.给大家推荐一个好的讲解博客.h ...

  8. HDU 2089 不要62【数位DP入门题】

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

  9. 【ACM】不要62 (数位DP)

    题目:http://acm.acmcoder.com/showproblem.php?pid=2089 杭州人称那些傻乎乎粘嗒嗒的人为62(音:laoer).杭州交通管理局经常会扩充一些的士车牌照,新 ...

随机推荐

  1. python_协程方式操作数据库

    # !/usr/bin/python3 # -*- coding: utf-8 -*- import requests import gevent import pymysql from gevent ...

  2. linux根据端口号查询来源程序

    1.根据端口号查询进程  netstat -tunlp|grep port 2.根据进程查询来源程序 ps aux | grep pid 上图看出所属进程为2281 上图看出占用8083端口的程序为n ...

  3. Git知识总览(四) git分支管理之rebase 以及 cherry-pick相关操作

    上篇博客聊了<Git知识总览(三) 分支的创建.删除.切换.合并以及冲突解决>,本篇博客我们主要来看一下 rebase 变基相关的操作.rebase 操作和 merge 操作最终都可以达到 ...

  4. sp_getAppLock使用

    sp_getAppLock  获取程序资源锁,简单的说就是调用此函数可以达到我们程序中.NET的lock锁的作用. 作用域是当前数据库下 四个参数: @resource(必填):资源名称,类型nvar ...

  5. MYSQL 5.7 修改密码、登录问题

    mysql5.7 关于密码问题 报错: ERROR 1862 (HY000): Your password has expired. To log in you must change it usin ...

  6. Unity DoTween 动画使用案例

    这边我就直接放一个标准的Dotween动画的使用demo吧. 这个案例满足应该可以完成你所想实现的几乎所有复杂动画. void PlayTween() { //set tween data float ...

  7. 【转】 C++易混知识点2. 函数指针和指针函数的区别

    我们时常在C++开发中用到指针,指针的好处是开销很小,可以很方便的用来实现想要的功能,当然,这里也要涉及到指针的一些基本概念.指针不是基本数据类型,我们可以理解他为一种特殊类型的对象,他占据一定空间, ...

  8. cobbler自动化安装系统

    笔者Q:972581034 交流群:605799367.有任何疑问可与笔者或加群交流 在很久很久以前,使用kickstart实现自动化安装的时候,我一直认为装系统是多么高大上的活,直到cobbler的 ...

  9. 用户 'IIS APPPOOL\.NET v4.5 Classic' 登录失败。

    我在win8.1系统下用vs2013+SqlServer08编写完项目后,挪到另一台win8.1系统(安装了Vs2010+SqlServer08)中,把网址挂到IIs中时,出现如下错误 : 解决方案: ...

  10. mysql修改root用户密码

    自我总结,欢迎拍砖! 目的:若root用户密码忘记,则需要重新设置root用户的密码. 步骤: 1.找到mysql安装目录下的 my.ini 文件,找到[mysqlId]一行,在下方添加语句:skip ...