有趣的数
 
问题描述
  我们把一个数称为有趣的,当且仅当:   
1. 它的数字只包含0, 1, 2, 3,且这四个数字都出现过至少一次。   
2. 所有的0都出现在所有的1之前,而所有的2都出现在所有的3之前。   
3. 最高位数字不为0。   
因此,符合我们定义的最小的有趣的数是2013。除此以外,4位的有趣的数还有两个:2031和2301。   
请计算恰好有n位的有趣的数的个数。由于答案可能非常大,只需要输出答案除以1000000007的余数。
输入格式
  输入只有一行,包括恰好一个正整数n (4 ≤ n ≤ 1000)。
输出格式
  输出只有一行,包括恰好n 位的整数中有趣的数的个数除以1000000007的余数。
样例输入
4
样例输出
3
 
原理:动态规划
假设我们从高位一直到低位对每个位进行赋值,那么我们会发现根据赋值情况,我们可以根据已经使用的数字(0,1,2,3)将数据分为6种状态:
1、首先我们从最高位开始,由于0,2分别需在1,3前面而0又不能放置在最高位,最高位只能是2,因此我们得到第一种状态,即前n-1位都是为2的情况。
2、现在我们再放置一个数,目前有0,1,3没用,由于0必须在1前面,所以我们在放置1是必须先放置0。此时我们可以放置0或者3,基于此,我们得到两种状态,一种是前n-1为只有2、0;另一种前n-1位只有2、3。
3、现在我们再放置一个数,对于已经放置2和0的状态,我们可以放置1或3,则又得到两种状态,分别是2、0、1和2、0、3;对于已经放置2和3的状态,我们只能放置0(0必须优先放于1前面)。得到状态2、3、0(此状态和2、0、3相同)
4、最后一种所有0,1,2,3都被使用
 
6重状态如下:
   0--用了2,剩0,1,3
  1--用了0,2,剩1,3
  2--用了2,3,剩0,1
  3--用了0,1,2,剩3
  4--用了0,2,3,剩1
  5--0,1,2,3
后面的状态由前面状态转化而来。如:
第5种状态全部使用可以由第5中状态自身维持或第3种状态或第4种状态转化而来
转化如下:假设从n-1到n位
第5中状态自身维持:可以在n位放置1或3(维持自身状态不变只能放置1或3,因为前面已经有1,3所以再放0,2就会违反规则)
第3种状态:可以在n位放置3
第4种状态:可以在n位放置1
得到如下公式:
states[i][5] = (states[j][3](达到状态五仅有在1后加3) + states[j][4] ](达到状态五仅有2301)+ states[j][5] * 2(达到状态五仅有加1/3) % mod;
其它同理。
 
代码如下:
 1 #include <iostream>
2
3 using namespace std;
4
5 int main(){
6 long long mod = 1000000007;
7 long long n;
8 cin>>n;
9 long long **states = new long long*[n+1];
10 for(long long i =0;i<n+1;i++)
11 states[i]=new long long[6];
12 for(long long i =0;i<6;i++)
13 states[0][i]=0;
14 /*6种状态
15 * 0--剩013
16 * 1--剩13
17 * 12-剩01
18 * 3--剩3
19 * 4--剩1
20 * 5--无
21 */
22 for(long long i=1;i<=n;i++)
23 {
24 long long j = i-1;
25 states[i][0] = 1;
26 states[i][1] = (states[j][0] + states[j][1] * 2) % mod;
27 states[i][2] = (states[j][0] + states[j][2]) % mod;
28 states[i][3] = (states[j][1] + states[j][3] * 2) % mod;
29 states[i][4] = (states[j][1] + states[j][2] + states[j][4] * 2) % mod;
30 states[i][5] = (states[j][3] + states[j][4] + states[j][5] * 2) % mod;
31 }
32 cout<<states[n][5]<<endl;
33 return 0;
34 }

CCF-有趣的数(数位DP)的更多相关文章

  1. CCF 201312-4 有趣的数 (数位DP, 状压DP, 组合数学+暴力枚举, 推公式, 矩阵快速幂)

    问题描述 我们把一个数称为有趣的,当且仅当: 1. 它的数字只包含0, 1, 2, 3,且这四个数字都出现过至少一次. 2. 所有的0都出现在所有的1之前,而所有的2都出现在所有的3之前. 3. 最高 ...

  2. 【CCF】有趣的数 数位dp

    [思路] dp[i][j]表示前i个数为第j种状态,考虑6种状态 0: 出现且仅出现 2 1: 出现且仅出现 2 0 2: 出现且仅出现 2 3 3: 出现且仅出现 2 0 1 4: 出现且仅出现 2 ...

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

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

  4. 【BZOJ-1026】windy数 数位DP

    1026: [SCOI2009]windy数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5230  Solved: 2353[Submit][Sta ...

  5. bzoj 1026 [SCOI2009]windy数 数位dp

    1026: [SCOI2009]windy数 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline ...

  6. CSP201312-4 有趣的数【dp】

    问题描述 试题编号: 201312-4 试题名称: 有趣的数 时间限制: 1.0s 内存限制: 256.0MB 问题描述: 问题描述 我们把一个数称为有趣的,当且仅当: 1. 它的数字只包含0, 1, ...

  7. luogu P2657 [SCOI2009]windy数 数位dp 记忆化搜索

    题目链接 luogu P2657 [SCOI2009]windy数 题解 我有了一种所有数位dp都能用记忆话搜索水的错觉 代码 #include<cstdio> #include<a ...

  8. 【BZOJ 3326】[Scoi2013]数数 数位dp+矩阵乘法优化

    挺好的数位dp……先说一下我个人的做法:经过观察,发现这题按照以往的思路从后往前递增,不怎么好推,然后我就大胆猜想,从前往后推,发现很好推啊,维护四个变量,从开始位置到现在有了i个数 f[i]:所有数 ...

  9. 洛谷P2657 [SCOI2009]windy数 [数位DP,记忆化搜索]

    题目传送门 windy数 题目描述 windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道, 在A和B之间,包括A和B,总共有多少个win ...

  10. 【bzoj1026】[SCOI2009]windy数 数位dp

    题目描述 windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? 输入 包含两个整数 ...

随机推荐

  1. 风炫安全web安全学习第二十九节课 CSRF防御措施

    风炫安全web安全学习第二十九节课 CSRF防御措施 CSRF防御措施 增加token验证 对关键操作增加token验证,token值必须随机,每次都不一样 关于安全的会话管理(SESSION) 不要 ...

  2. .netcore利用perf分析高cpu使用率

    目录 一 在宿主机运行perf 二 容器内安装perf 1,重新构建镜像 2,下载火焰图生成脚本 3,安装linux-perf 三 CPU占用分析 1,perf record捕获进程 2,生成火焰图 ...

  3. Go语言从入门到放弃(结构体常见的tag)

    什么是tag Tag是结构体中某个字段别名, 可以定义多个, 空格分隔 type Student struct {     Name string `ak:"av" bk:&quo ...

  4. leetcode-222完全二叉树的节点个数

    题目 给出一个完全二叉树,求出该树的节点个数. 说明: 完全二叉树的定义如下:在完全二叉树中,除了最底层节点可能没填满外,其余每层节点数都达到最大值,并且最下面一层的节点都集中在该层最左边的若干位置. ...

  5. 【十天自制软渲染器】DAY 02:画一条直线(DDA 算法 & Bresenham’s 算法)

    推荐关注公众号「卤蛋实验室」或访问博客原文,更新更及时,阅读体验更佳 第一天我们搭建了 C++ 的运行环境并画了一个点,根据 点 → 线 → 面 的顺序,今天我们讲讲如何画一条直线. 本文主要讲解直线 ...

  6. kubernets之Ingress资源

    一  Ingress集中式的kubernets服务转发控制器 1.1  认识Ingress的工作原理 注意:图片来源于kubernets in action一书,如若觉得侵权,请第一时间联系博主进行删 ...

  7. leetcode 940. 不同的子序列 II (动态规划 ,字符串, hash,好题)

    题目链接 https://leetcode-cn.com/problems/distinct-subsequences-ii/ 题意: 给定一个字符串,判断里面不相同的子串的总个数 思路: 非常巧妙的 ...

  8. LeetCode454. 四数相加 II

    题目 给定四个包含整数的数组列表 A , B , C , D ,计算有多少个元组 (i, j, k, l) ,使得 A[i] + B[j] + C[k] + D[l] = 0. 分析 关键是如何想到用 ...

  9. 实操|如何将 Containerd 用作 Kubernetes runtime

    日前专为开发者提供技术分享的又拍云 OpenTalk 公开课邀请了网易有道资深运维开发工程师张晋涛,直播分享<Containerd 上手实践 >,详细介绍 Containerd 的发展历程 ...

  10. python中IF语句容易犯的错误CASE

    python中没有switch   case类似的语句,但是下面的IF语句却与之类似,却又不同: A = B = C = D = E = 1 if A == 1: B=2 elif B ==2: C= ...