LeetCode第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 = "NEW"
输出:3
示例 4:

输入:word = "YEAR"
输出:7
 

提示:

2 <= word.length <= 300
每个 word[i] 都是一个大写英文字母。

思路:定义dp[i][j]:表示最后一次移动后的手指停留在位置i并且另一只手指在位置j的最小移动次数。

状态转移:

dp[i][i-1]=min(dp[i-1][i-1], dp[i][i-1]);

dp[i][i]=min(dp[i][i], dp[i-1][i-1]+dis(word.charAt(i-1),word.charAt(i)));

dp[i][j]=Math.min(dp[i][j],dp[i-1][j]+dis(word.charAt(i),word.charAt(i-1)));

dp[i][i-1]=Math.min(dp[i][i-1], dp[i-1][j]+dis(word.charAt(i),word.charAt(j)));

class Solution {
    public int minimumDistance(String word) {
    	int n=word.length();
    	int[][] dp=new int[n][n];
    	
    	for(int i=0;i<n;i++)
    		for(int j=0;j<n;j++)
    			dp[i][j]=1000;
    	dp[0][0]=0;
	for(int i=1;i<n;i++) {
		dp[i][i-1]=Math.min(dp[i-1][i-1], dp[i][i-1]);
		dp[i][i]=Math.min(dp[i][i], dp[i-1][i-1]+dis(word.charAt(i-1),word.charAt(i)));
		for(int j=0;j<i;j++) {
			dp[i][j]=Math.min(dp[i][j],dp[i-1][j]+dis(word.charAt(i),word.charAt(i-1)));
			dp[i][i-1]=Math.min(dp[i][i-1], dp[i-1][j]+dis(word.charAt(i),word.charAt(j)));
		}
	}
	int ans=Integer.MAX_VALUE;
	for(int i=0;i<n;i++) ans=Math.min(ans, dp[n-1][i]);
	return ans;
    }
    
    private int dis(char a,char b) {
    	int x1=(a-'A')/6;
    	int y1=(a-'A')%6;
    	int x2=(b-'A')/6;
    	int y2=(b-'A')%6;
    	return Math.abs(x1-x2)+Math.abs(y1-y2);
    }
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值