P2657 [SCOI2009]windy数 题解

博客园同步

原题链接

简要题意:

一个 相邻两个数字差的绝对值都 ≥ 2 \geq 2 2 且不含前导零 的数 被称为 “windy数”。问从 a a a b b b 的 “windy数”的个数。

首先,我们考虑 1 1 1 ~ n n n 的 “windy数” 的个数怎么求。

f i , j f_{i,j} fi,j 表示有 i i i 位,最高位为 j j j 的方案数。

那么,从 i − 1 → i i-1 \rightarrow i i1i 只需要在 最高位前面拼上合法的一位

即:

f i , j = ∑ k = 0 9 f i − 1 , k [ abs ⁡ j , k ≥ 2 ] f_{i,j} = \sum_{k=0}^9 f_{i-1,k} [\operatorname{abs}{j,k} \geq 2] fi,j=k=09fi1,k[absj,k2]

k k k 即枚举合法的数字。

对于 i = 1 i=1 i=1 的情况, f i , j = 1 f_{i,j} = 1 fi,j=1. 则:

f i , j = { ∑ k = 0 9 f i − 1 , k [ abs ⁡ j , k ≥ 2 ] , i ≠ 1 1 , i = 1 f_{i,j} = \begin{cases} \sum_{k=0}^9 f_{i-1,k} [\operatorname{abs}{j,k} \geq 2] , i \not = 1 \\ 1 , i = 1 \\ \end{cases} fi,j={k=09fi1,k[absj,k2],i=11,i=1

如何求答案呢?

我们分为三类:

  1. 位数比 n n n 小的数。

  2. 位数和 n n n 一样,但最高位比 n n n 小的数。

  3. 位数和最高位都和 n n n 一样,但比 n n n 小的数。

第一部分,答案为 ( l e n len len x x x 的位数)

∑ i = 1 l e n − 1 ∑ j = 1 9 f i , j \sum_{i=1}^{len-1} \sum_{j=1}^9 f_{i,j} i=1len1j=19fi,j

第二部分则枚举最高位即可。

第三部分有点复杂,只需要枚举位数和次高位即可。

时间复杂度: O ( siz ⁡ 2 n ) O(\operatorname{siz}^2{n}) O(siz2n).

实际得分: 100 p t s 100pts 100pts.

#pragma GCC optimize(2)
#include<bits/stdc++.h>
using namespace std;

inline int read(){char ch=getchar(); int f=1; while(ch<'0' || ch>'9') {if(ch=='-') f=-f; ch=getchar();}
	int x=0; while(ch>='0' && ch<='9') x=(x<<3)+(x<<1)+ch-'0',ch=getchar(); return x*f;}

int n,m,a[15];
int f[15][15];
	
inline int calc(int x) { //计算 1 ~ x 的答案
	memset(a,0,sizeof(a));
	int len=0; while(x) a[++len]=x%10,x/=10;
	int ans=0;
	for(int i=1;i<len;i++) 
	for(int j=1;j<=9;j++) ans+=f[i][j]; //第一部分
	for(int i=1;i<a[len];i++) ans+=f[len][i]; //第二部分
	for(int i=len-1;i>=1;i--) {
		for(int j=0;j<a[i];j++)
			if(abs(j-a[i+1])>=2) ans+=f[i][j];
		if(abs(a[i+1]-a[i])<2) break;	//第三部分
	} return ans;
}

int main() {
	n=read(),m=read();
	for(int i=0;i<=9;i++) f[1][i]=1;
	for(int i=2;i<=10;i++)
	for(int j=0;j<=9;j++)
	for(int k=0;k<=9;k++)
		if(abs(j-k)>=2) f[i][j]+=f[i-1][k]; //求出 f
	printf("%d\n",calc(m+1)-calc(n));	
	return 0;
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值