LeetCode-1320-二指输入的的最小距离
二指输入法定制键盘在 XY 平面上的布局如上图所示,其中每个大写英文字母都位于某个坐标处,例如字母 A 位于坐标 (0,0),字母 B 位于坐标 (0,1),字母 P 位于坐标 (2,3) 且字母 Z 位于坐标 (4,1)。
给你一个待输入字符串 word,请你计算并返回在仅使用两根手指的情况下,键入该字符串需要的最小移动总距离。坐标 (x1,y1) 和 (x2,y2) 之间的距离是 |x1 - x2| + |y1 - y2|。
注意,两根手指的起始位置是零代价的,不计入移动总距离。你的两根手指的起始位置也不必从首字母或者前两个字母开始。
示例 1:
1 | 输入:word = "CAKE" |
解释:
使用两根手指输入 “CAKE” 的最佳方案之一是:
手指 1 在字母 ‘C’ 上 -> 移动距离 = 0
手指 1 在字母 ‘A’ 上 -> 移动距离 = 从字母 ‘C’ 到字母 ‘A’ 的距离 = 2
手指 2 在字母 ‘K’ 上 -> 移动距离 = 0
手指 2 在字母 ‘E’ 上 -> 移动距离 = 从字母 ‘K’ 到字母 ‘E’ 的距离 = 1
总距离 = 3示例 2:
1 | 输入:word = "HAPPY" |
解释:
使用两根手指输入 “HAPPY” 的最佳方案之一是:
手指 1 在字母 ‘H’ 上 -> 移动距离 = 0
手指 1 在字母 ‘A’ 上 -> 移动距离 = 从字母 ‘H’ 到字母 ‘A’ 的距离 = 2
手指 2 在字母 ‘P’ 上 -> 移动距离 = 0
手指 2 在字母 ‘P’ 上 -> 移动距离 = 从字母 ‘P’ 到字母 ‘P’ 的距离 = 0
手指 1 在字母 ‘Y’ 上 -> 移动距离 = 从字母 ‘A’ 到字母 ‘Y’ 的距离 = 4
总距离 = 6示例 3:
1 | 输入:word = "NEW" |
示例 4:
1 | 输入:word = "YEAR" |
提示:
2 <= word.length <= 300
每个 word[i]。都是一个大写英文字母。
Solution1(记忆化):
我们首先要定义一个递归方程,思考后发现有两类主要元素:第一类是要记录到哪个字符,第二类是两个手指当前的位置。
然后发现“记录到哪个字符”这个状态不太好与两个手指进行关联(只能和一个手指进行绑定),所以我们转化一下思路,将记录到哪个字符转变为从第i个字符起,到字符串的末尾这整个区间,然后记录两个手指的状态。这样就得到了如下的递归函数参数:
1 | // i: [i:w.size()]区间内花费的最小距离, |
转移方程为:
1 | dfs(i,l,c) = min(dfs(i+1,l,c)+cost(r,c),dfs(i+1,c,r)+cost(l,c)); |
然后添加记忆化数组,整理就得到了下面的解决方案:
1 | class Solution { |
Solution2(DP):
我们可以试着将记忆化搜索转化为dp求解,很明显,状态表示和递归函数的状态是一致的,都是三维表示。
1 | int dp[i][j][k]; //第一个手指一动到i,第二个手指移动到j,已经移动了k个字符的最小代价。 |
这里与记忆化搜索定义的字符区间范围正好相反,递归中用的是从i到n的区间,而这里由于天然的记忆化过程,可以直接表示从0到k。
接着写状态转移方程:
1 | //从上一个字符移动到当前字符使用第一个手指,计算的最小价值。 |
最后我们查看手指停在任何字符上的代价取最小即为答案。
1 | class Solution { |