171场周赛

5310. 二指输入的的最小距离

二指输入法定制键盘在 XY 平面上的布局如上图所示,其中每个大写英文字母都位于某个坐标处,例如字母 A 位于坐标 (0,0),字母 B 位于坐标 (0,1),字母 P 位于坐标 (2,3) 且字母 Z 位于坐标 (4,1)。

给你一个待输入字符串 word,请你计算并返回在仅使用两根手指的情况下,键入该字符串需要的最小移动总距离。坐标 (x1,y1) 和 (x2,y2) 之间的距离是 |x1 - x2| + |y1 - y2|。

注意,两根手指的起始位置是零代价的,不计入移动总距离。你的两根手指的起始位置也不必从首字母或者前两个字母开始。

示例 1:

输入:word = “CAKE”
输出:3
解释
使用两根手指输入 “CAKE” 的最佳方案之一是:
手指 1 在字母 ‘C’ 上 -> 移动距离 = 0
手指 1 在字母 ‘A’ 上 -> 移动距离 = 从字母 ‘C’ 到字母 ‘A’ 的距离 = 2
手指 2 在字母 ‘K’ 上 -> 移动距离 = 0
手指 2 在字母 ‘E’ 上 -> 移动距离 = 从字母 ‘K’ 到字母 ‘E’ 的距离 = 1
总距离 = 3

示例 2:

输入:word = “HAPPY”
输出:6
解释
使用两根手指输入 “HAPPY” 的最佳方案之一是:
手指 1 在字母 ‘H’ 上 -> 移动距离 = 0
手指 1 在字母 ‘A’ 上 -> 移动距离 = 从字母 ‘H’ 到字母 ‘A’ 的距离 = 2
手指 2 在字母 ‘P’ 上 -> 移动距离 = 0
手指 2 在字母 ‘P’ 上 -> 移动距离 = 从字母 ‘P’ 到字母 ‘P’ 的距离 = 0
手指 1 在字母 ‘Y’ 上 -> 移动距离 = 从字母 ‘A’ 到字母 ‘Y’ 的距离 = 4
总距离 = 6

示例 3:

输入:word = “HAPPY”
输出:7

Solution

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
func minimumDistance(_ word: String) -> Int {
let chars = Array(word)
let table = Array("abcdefghijklmnopqrstuvwxyz".uppercased())
var charDic = [Character: Int]()
for i in table.indices {
charDic[table[i]] = i
}

var dp = [[[Int]]](repeating: [[Int]](repeating: [Int](repeating: 0, count: 301), count: 27), count: 27)
func cost(_ from: Int, _ to: Int) -> Int {
if from == 26 { return 0 }
return abs(from/6 - to/6) + abs(from%6 - to%6)
}

func distance(_ pos: Int, _ left: Int, _ right: Int) -> Int {
if pos >= word.count { return 0 }
if dp[left][right][pos] == 0 {
let to = charDic[chars[pos]]!
dp[left][right][pos] = min(cost(left, to) + distance(pos + 1, to, right),
cost(right, to) + distance(pos + 1, left, to))
}
return dp[left][right][pos]

}
return distance(0, 26, 26)
}