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的更多相关文章

  1. 【DP专辑】ACM动态规划总结

    转载请注明出处,谢谢.   http://blog.csdn.net/cc_again?viewmode=list          ----------  Accagain  2014年5月15日 ...

  2. 【DP专辑】ACM动态规划总结(转)

    http://blog.csdn.net/cc_again/article/details/25866971 动态规划一直是ACM竞赛中的重点,同时又是难点,因为该算法时间效率高,代码量少,多元性强, ...

  3. 集训第五周动态规划 H题 回文串统计

    Hrdv is interested in a string,especially the palindrome string.So he wants some palindrome string.A ...

  4. (转)dp动态规划分类详解

    dp动态规划分类详解 转自:http://blog.csdn.NET/cc_again/article/details/25866971 动态规划一直是ACM竞赛中的重点,同时又是难点,因为该算法时间 ...

  5. HDU 1052 Tian Ji -- The Horse Racing【贪心在动态规划中的运用】

    算法分析: 这个问题很显然可以转化成一个二分图最佳匹配的问题.把田忌的马放左边,把齐王的马放右边.田忌的马A和齐王的B之间,如果田忌的马胜,则连一条权为200的边:如果平局,则连一条权为0的边:如果输 ...

  6. 增强学习(三)----- MDP的动态规划解法

    上一篇我们已经说到了,增强学习的目的就是求解马尔可夫决策过程(MDP)的最优策略,使其在任意初始状态下,都能获得最大的Vπ值.(本文不考虑非马尔可夫环境和不完全可观测马尔可夫决策过程(POMDP)中的 ...

  7. 简单动态规划-LeetCode198

    题目:House Robber You are a professional robber planning to rob houses along a street. Each house has ...

  8. 动态规划 Dynamic Programming

    March 26, 2013 作者:Hawstein 出处:http://hawstein.com/posts/dp-novice-to-advanced.html 声明:本文采用以下协议进行授权: ...

  9. 动态规划之最长公共子序列(LCS)

    转自:http://segmentfault.com/blog/exploring/ LCS 问题描述 定义: 一个数列 S,如果分别是两个或多个已知数列的子序列,且是所有符合此条件序列中最长的,则 ...

随机推荐

  1. 谈论linux同组多用户操作问题

    同组多用户,最先起到什么作用这个我也是不明白的, 然后乱搞了一堆, 下面我做个笔记帮组日后分析. 一个用户组承载多个用户, 像这样 这是我原先的思路.然后就是chenglee用户和chenglee12 ...

  2. 20165310 《Java程序设计》课程总结

    20165310 <Java程序设计>课程总结 一.每周作业及实验报告博客链接汇总 预备作业一:我期望的师生关系 20165310 我期望的师生关系 预备作业二:学习基础和C语言基础调查 ...

  3. 题说proxy

    昨天在和群友讨论时遇到一题是这样的. 题目描述 //Tomy非常敏感,不喜欢别人碰他的东西.一旦有人碰他就会大喊Don't Touch Me. //完成tomy这个对象,禁止对tomy的内容进行修改( ...

  4. 无旋转Treap简介

    无旋转Treap是一个神奇的数据结构,能够支持插入,删除,查询k大,查询某个数的排名,查询前驱后继,支持各种区间操作和持久化.基于旋转的Treap无法实现区间反转等操作,但是无旋Treap可以轻易地支 ...

  5. Python3 tkinter基础 Listbox Scrollbar 创建垂直滚动条

             Python : 3.7.0          OS : Ubuntu 18.04.1 LTS         IDE : PyCharm 2018.2.4       Conda ...

  6. (一)flutter第一天

    import 'package:flutter/material.dart'; void main() => runApp(new MyApp()); class MyApp extends S ...

  7. linux内核中的IPIs是什么?

    答: 处理器间中断(Interprocessor Interrupts)

  8. 使用kubeadm 安装 kubernetes 1.12.0

    目录 简介: 架构说明: 系统配置: 1.1 关闭防火墙 1.2 禁用SELinux 1.3 关闭系统Swap 1.4 安装docker 使用kubeadm部署Kubernetes: 2.1 安装ku ...

  9. SpringBoot 使用jwt进行身份验证

    这里只供参考,比较使用jwt方式进行身份验证感觉不好,最不行的就是不能退出 登陆时设定多长过期时间,只能等这个时间过了以后才算退出,服务端只能验证请求过来的token是否通过验证 Code: /** ...

  10. 102. Binary Tree Level Order Traversal 广度优先遍历

    Given a binary tree, return the level order traversal of its nodes' values. (ie, from left to right, ...