博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
HDU 4433 locker(12年天津,DP)
阅读量:6418 次
发布时间:2019-06-23

本文共 3087 字,大约阅读时间需要 10 分钟。

4576 Accepted 860 KB 140 ms 2063 B 2014-10-16 09:51:19

哎,为啥1000*100*100的复杂度的dp就不敢敲了呢,,,真是2

涉及到可能有后效性的时候,一维就不行了,要扩维。本题,一个状态的变化会影响后两个,所以要用三维。

 

lockerTime Limit: 6000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 1418    Accepted Submission(s): 620

Problem Description
A password locker with N digits, each digit can be rotated to 0-9 circularly. You can rotate 1-3 consecutive digits up or down in one step. For examples: 567890 -> 567901 (by rotating the last 3 digits up) 000000 -> 000900 (by rotating the 4th digit down) Given the current state and the secret password, what is the minimum amount of steps you have to rotate the locker in order to get from current state to the secret password?
 
Input
Multiple (less than 50) cases, process to EOF. For each case, two strings with equal length (≤ 1000) consists of only digits are given, representing the current state and the secret password, respectively.
 
Output
For each case, output one integer, the minimum amount of steps from the current state to the secret password.
 
Sample Input
111111 222222 896521 183995
 
Sample Output
2 12
 
Source
 
Recommend
zhoujiaqi2010   |   We have carefully selected several similar problems for you:            
 
 
题意及题解转自:

题目:给出两个串,每次可以选择连续的1-3个数字,进行同时加1或者同时减1,问最少经过多少次操作,将一个串转变为另外一个串

 

以前有类似的题目,BFS就可以了

不过这题的数据量太大,听说也有不少是搜索过的

我写了一个矬B的搜索,反正是挂了,没加什么优化

训练的时候,yobobobo的DP解法比较接近,可是最终貌似卡在后效性上?

dp[i][j][k]表示 前i个已经完全匹配,而这时候,第i+1个已经加了j位,第i+2位已经加了k

转移分为两步,枚举加,枚举减

注意如果第i位加了a,第i+1位加了b,第i+2位加了c,那么a>=b>=c这个关系不能错

 

以下题解转自:

这题的意思就相当于是一个数字密码锁。

 

每次可以正向或者反向旋转连续的1-3个数字。求从现在状态转到目标状态需要的最少步数。

 

题目给了两个长度一样的由0-9组成的字符串。就相当于每次操作可以选择连续的1-3个数字加1或者减1.这不过这个加和减是循环的。0减1到9,9加1到0.

 

 

 

一看就是DP。

 

这不过DP方程不好想,也不好表示状态。

 

 

 

dp[i][x][y]表示处理到第i个,后面两个数字是x,y,把前i位转正确需要的最少步数。

 

计算dp[i][x][y]时,前i-2位是题目给的现在状态的值,第i-1位是x,第i位是y,就是把前i位转正确。

 

 

 

要把dp[i]的状态转移到dp[i-1]去。

 

把第i位从x转到目标态b[i]去,就可以把状态转移了。

 

 

 

和第i位相关的转动有三种:一是单转第i位,二是转第i位和第i-1位,三是转第i位、第i-1位和第i-2位。

 

根据三种可以确定 dp[i-1][xx][yy]中的xx,yy;

 

转动分为正转和反转。

 

如果第i位是正转,转正确需要d1步。

 

那么第i-1和第i-2位正转的不是是小于等于d1的。而且i-2的步数小于等于i-1。

 

如果第i位是正转,转正确需要d2步。
那么第i-1和第i-2位正转的不是是小于等于d2的。而且i-2的步数小于等于i-1。
 
这样DP的时候i从1到n转移过去。
处理dp[i]的时候,dp[1~(n-1)][0~9][0~9]都是已知的。就很容易确定dp[i][0~9][0~9]的最小值了。
 
注意处理的是初始化过程,全部初始化为INF,dp[0][0][0]=0;
还有转移的时候,i==1和i==2单独处理下。。
转移最多是1000*100*100。
是可以接受的。
 
我一开始做的时候增加了以为dp[i][x][y][z]表示了后面三位,好理解,但是TLE了。
减少一维表示就AC了。
 
 
具体看代码吧!
注释的很清楚了。

 

 

1 #include
2 #include
3 #include
4 #include
5 #include
6 #include
7 #include
8 #include
9 #include
10 #include
11 //#include
12 13 #define N 1005 14 #define M 1000005 15 #define mod 1000000007 16 #define inf 0x3f3f3f3f 17 //#define p 10000007 18 #define mod2 100000000 19 #define ll long long 20 #define LL long long 21 #define maxi(a,b) (a)>(b)? (a) : (b) 22 #define mini(a,b) (a)<(b)? (a) : (b) 23 24 using namespace std; 25 26 char s1[N]; 27 char s2[N]; 28 int a[N]; 29 int dp[N][12][12]; 30 int n; 31 32 void ini() 33 { 34 n=strlen(s1); 35 memset(dp,inf,sizeof(dp)); 36 int i; 37 a[0]=0; 38 for(i=0;i

 

转载于:https://www.cnblogs.com/njczy2010/p/4028114.html

你可能感兴趣的文章
request:通过表单手机客户机数据
查看>>
response发送状态码
查看>>
python-django(框架结构)
查看>>
常用dos命令
查看>>
跨线程调用DataGridView控件
查看>>
input框限制只能输入正整数,逻辑与和或运算
查看>>
【angularJS】定义模块angular.module
查看>>
Windows7+IIS7.5+PHP修改上传文件大小的解决方法
查看>>
java Bean的映射工具
查看>>
无法加载程序集,因为该程序集包含EdmSchemaAttribute,并按名称加载结束类型。不允许同时按名称和特性进行加载...
查看>>
缓存算法
查看>>
Windows 8 动手实验系列教程 实验5:进程生命周期管理
查看>>
Android开发计划
查看>>
application/x-www-form-urlencoded接口响应报文中文乱码
查看>>
SpringMVC 简介及入门案例
查看>>
物联网硬件安全分析基础-硬件分析初探
查看>>
4、javascript中各种提示框的使用
查看>>
POJ3525 Most Distant Point from the Sea
查看>>
Activity与Service通信(不同进程之间)
查看>>
《数据库系统概论》第九章笔记
查看>>