动态规划-Race Car
2018-10-26 21:06:54
问题描述:


问题求解:
方法一、BFS
首先将使用BFS进行解空间的遍历,也就是将本问题转化成了搜索问题,但是有两个地方需要注意:
1、状态保存的问题,每个位置的状态由其位置信息和速度信息构成,但是如果将所有的位置出现过的速度进行保存会MLE,这里进行了一步简化,只保存当前位置速度绝对值为1的状态,定义这些状态不再后续的求解中被重复扩展;
2、Pruning,必须剪枝,如果不剪枝,则会TLE,这里采取的剪枝策略是将所有扩展到的距离target长度大于target的pos放弃。
3、状态保存采用了字符串拼接,这里可以进行进一步的优化。
public int racecar(int target) {
int step = 0;
Queue<int[]> q = new LinkedList<>();
q.add(new int[]{0, 1});
Set<String> set = new HashSet();
set.add("0_1");
set.add("0_-1");
while (!q.isEmpty()) {
step++;
int size = q.size();
for (int i = 0; i < size; i++) {
int[] cur = q.poll();
int pos = cur[0] + cur[1];
int speed = cur[1] * 2;
if (pos == target) return step;
if (Math.abs(pos - target) < target)
q.add(new int[]{pos, speed});
pos = cur[0];
speed = cur[1] > 0 ? -1 : 1;
String state = String.valueOf(pos) + "_" + String.valueOf(speed);
if (set.contains(state)) continue;
set.add(state);
q.add(new int[]{pos, speed});
}
}
return -1;
}
方法二、BFS + Integer状态
使用String来保存状态,在每次用Hash查询的时候开销非常大,这里可以使用Integer来进行优化。时间是原来是1/4,应该来说速度上提升还是很多的。
public int racecar(int target) {
int step = 0;
Queue<int[]> q = new LinkedList<>();
q.add(new int[]{0, 1});
Set<Integer> set = new HashSet();
set.add(0 << 2 | 1);
set.add(0 << 2 | 2);
while (!q.isEmpty()) {
step++;
int size = q.size();
for (int i = 0; i < size; i++) {
int[] cur = q.poll();
int pos = cur[0] + cur[1];
int speed = cur[1] * 2;
if (pos == target) return step;
if (Math.abs(pos - target) < target)
q.add(new int[]{pos, speed});
pos = cur[0];
speed = cur[1] > 0 ? -1 : 1;
Integer state = pos << 2 | ((speed == 1) ? 1 : 2);
if (set.contains(state)) continue;
set.add(state);
q.add(new int[]{pos, speed});
}
}
return -1;
}
方法三、DP
本题使用DP是最优解。
dp[i] : 在位置i的最短步数
有一个特殊情况,就是当前的pos正好等于2 ^ n - 1,此时所需步数可以直接运算出来,即d[i] = n if i == 2 ^ n - 1
如果当前的pos不是最佳情况,那么就有两种策略,一是先经过pos,在往回倒,二是在到达之前进行倒车再前进。
算法的时间复杂度分析:总的状态数为t,每个状态求解的均摊时间在logt,所以时间复杂度为O(nlogn)。使用Java运行时间为3ms,beat 100%。
public int racecar(int target) {
int[] dp = new int[target + 1];
return helper(target, dp);
}
private int helper(int target, int[] dp) {
if (dp[target] != 0) return dp[target];
int n = (int) Math.ceil(Math.log(target + 1) / Math.log(2));
if (1 << n == target + 1) return dp[target] = n;
dp[target] = n + 1 + helper((1 << n) - 1 - target, dp);
for (int m = 0; m < n - 1; m++) {
int pos = (1 << (n - 1)) - (1 << m);
dp[target] = Math.min(dp[target], n + m + 1 + helper(target - pos, dp));
}
return dp[target];
}
动态规划-Race Car的更多相关文章
- 【DP专辑】ACM动态规划总结
转载请注明出处,谢谢. http://blog.csdn.net/cc_again?viewmode=list ---------- Accagain 2014年5月15日 ...
- 【DP专辑】ACM动态规划总结(转)
http://blog.csdn.net/cc_again/article/details/25866971 动态规划一直是ACM竞赛中的重点,同时又是难点,因为该算法时间效率高,代码量少,多元性强, ...
- 集训第五周动态规划 H题 回文串统计
Hrdv is interested in a string,especially the palindrome string.So he wants some palindrome string.A ...
- (转)dp动态规划分类详解
dp动态规划分类详解 转自:http://blog.csdn.NET/cc_again/article/details/25866971 动态规划一直是ACM竞赛中的重点,同时又是难点,因为该算法时间 ...
- HDU 1052 Tian Ji -- The Horse Racing【贪心在动态规划中的运用】
算法分析: 这个问题很显然可以转化成一个二分图最佳匹配的问题.把田忌的马放左边,把齐王的马放右边.田忌的马A和齐王的B之间,如果田忌的马胜,则连一条权为200的边:如果平局,则连一条权为0的边:如果输 ...
- 增强学习(三)----- MDP的动态规划解法
上一篇我们已经说到了,增强学习的目的就是求解马尔可夫决策过程(MDP)的最优策略,使其在任意初始状态下,都能获得最大的Vπ值.(本文不考虑非马尔可夫环境和不完全可观测马尔可夫决策过程(POMDP)中的 ...
- 简单动态规划-LeetCode198
题目:House Robber You are a professional robber planning to rob houses along a street. Each house has ...
- 动态规划 Dynamic Programming
March 26, 2013 作者:Hawstein 出处:http://hawstein.com/posts/dp-novice-to-advanced.html 声明:本文采用以下协议进行授权: ...
- 动态规划之最长公共子序列(LCS)
转自:http://segmentfault.com/blog/exploring/ LCS 问题描述 定义: 一个数列 S,如果分别是两个或多个已知数列的子序列,且是所有符合此条件序列中最长的,则 ...
随机推荐
- Android - Resource 之 Drawable小结
本篇直接选择性地翻译官方开发指南 ============================= Drawable有十种类型,如下 (1) - Bitmap file:这个简单,也可以用xml来更详细的定 ...
- 20145208 蔡野 《网络对抗》Exp5 MSF基础应用
20145208 蔡野 <网络对抗>Exp5 MSF基础应用 链接地址 主动攻击:利用ms08_067_netapi进行攻击 对浏览器攻击:MS10-002 对客户端攻击:adobe_to ...
- I2C总线的仲裁机制
在多主的通信系统中.总线上有多个节点,它们都有自己的寻址地址,可以作为从节点被别的节点访问,同时它们都可以作为主节点向其他的节点发送控制字节和传 送数据.但是如果有两个或两个以上的节点都向总线上发送启 ...
- Base64编码为什么会使数据量变大?
当把byte[]通过Convert.ToBase64String转换成Base64编码字符串时数据量明显变大,为何呢?这里就得先探究一下什么是Base64编码. Base64编码的思想是是采用64个基 ...
- JQuery---高级类选择器
1.ContentFilters 1.1 语法:$('div:contains(edu)').css('backgroundColor','yellow'); 只看div 本身是否包含内容 1.2 语 ...
- printf和std::cout ...endl
printf效率要比std::cout...endl高些,可以减少打印所花时间
- 4、keepalived高可用nginx负载均衡
keepalived: HTTP_GET //使用keepalived获取后端real server健康状态检测 SSL_GET(https) //这里以为这后端使用的是http协议 ...
- Model中时间格式化
MVC 中 @Html中的时间格式化 @Html.TextBoxFor(model => model.StartTime, "{0:yyyy-MM-dd HH:mm:ss}" ...
- _event_phase
EventId 事件ID Phase 阶段ID,从1开始 StopGUID 击杀生物或摧毁物体当前阶段结束,,正数为生物,负数为物体
- 两个DIV并排显示
今天做的一个项目,需要做3个div,一个是总框(Div1),另外两个是子框,按比例填满div1,我设置好两个div的width和height,发现效果是两个子div上下显示,如图所示: 要想将两个DI ...