Maximum Swap LT670
Given a non-negative integer, you could swap two digits at most once to get the maximum valued number. Return the maximum valued number you could get.
Example 1:
Input: 2736
Output: 7236
Explanation: Swap the number 2 and the number 7.
Example 2:
Input: 9973
Output: 9973
Explanation: No swap.
Note:
- The given number is in the range [0, 108]
Idea 1. To get a biggest digit for each position, the digit need to swap with the biggest in the number, since it aims to get the maximum valued number, the digit (digits[i]) need to swap with the biggest digit on the right(digits[j] > digits[i], j > i), the digit needs to be the leftmost one which has a bigger digit on the right side. Since needs index to swap, build the maxIndex array. 从右往左建一个maxIndexArray, 再从左到右遍历 找到第一个digits[i] < digits[maxIndex[i]], swap(i, maxIndex[i]).
Note: right most maxIndex, if there are duplicates like 27736, 77236 > 72736
Time complexity: O(n) 2 scan, if string construction is O(n)
Space complexity: O(n)
class Solution {
public int maximumSwap(int num) {
char[] digits = Integer.toString(num).toCharArray();
int[] maxIndex = new int[digits.length]; maxIndex[digits.length-1] = digits.length-1;
for(int i = digits.length-2; i >=0; --i) {
maxIndex[i] = i;
if(digits[maxIndex[i]] - '0' <= digits[maxIndex[i+1]] - '0') {
maxIndex[i] = maxIndex[i+1];
}
} for(int i = 0; i < digits.length; ++i) {
if(digits[i] - '0' < digits[maxIndex[i]] - '0') {
char c = digits[i];
digits[i] = digits[maxIndex[i]];
digits[maxIndex[i]] = c;
break;
}
} return Integer.parseInt(new String(digits));
}
}
Idea 1.a, to avoid maxIndex array, record the maxIndex on the way, 1 scan from right to left
class Solution {
private void swap(char[] digits, int leftIndex, int rightIndex) {
char c = digits[leftIndex];
digits[leftIndex] = digits[rightIndex];
digits[rightIndex] = c;
}
public int maximumSwap(int num) {
char[] digits = Integer.toString(num).toCharArray(); int leftIndex = -1, rightIndex = -1;
int maxIndex = digits.length-1;
for(int i = digits.length-1; i >=0; --i) {
if(digits[i] > digits[maxIndex]) {
maxIndex = i;
}
else if(digits[i] < digits[maxIndex]) {
leftIndex = i;
rightIndex = maxIndex;
}
} if(leftIndex == -1) {
return num;
} swap(digits, leftIndex, rightIndex); return Integer.parseInt(new String(digits));
}
}
Idea 1.c. Remove the use of toCharArray, convert to digit on the way from right to left.
Time complexity: O(n)
Space complexity: O(1)
class Solution {
public int maximumSwap(int num) {
int leftDigit = -1, rightDigit = -1;
int leftBase = 0, rightBase = 0;
int curr = num; int maxDigit = -1;
int maxBase = 0;
int base = 1;
while(curr != 0) {
int digit = curr % 10; if(digit > maxDigit) {
maxDigit = digit;
maxBase = base;
}
else if(digit < maxDigit) {
leftDigit = digit;
leftBase = base;
rightDigit = maxDigit;
rightBase = maxBase;
}
base = base * 10;
curr = curr/10;
} if(leftDigit == -1) {
return num;
} num = num - leftDigit*leftBase - rightDigit*rightBase
+ leftDigit* rightBase + rightDigit * leftBase; return num;
}
}
Idea 2. 官方的妙法,数字只有0-9,建立一个数组记录每个数字出现在最右边的index(从左到右扫), 再从左到右扫,寻找第一个digits[i] < last[d] (d > digits[i] and last[d] > i), swap(digits, i, last[d]).
class Solution {
private void swap(char[] digits, int i, int j) {
char c = digits[i];
digits[i] = digits[j];
digits[j] = c;
}
public int maximumSwap(int num) {
char[] digits = Integer.toString(num).toCharArray(); int[] last = new int[10];
for(int i = 0; i < digits.length; ++i) {
last[digits[i] - '0'] = i;
} for(int i = 0; i < digits.length; ++i) {
for(int d = 9; d > digits[i] - '0'; --d) {
if(last[d] > i) {
swap(digits, i, last[d]);
return Integer.valueOf(new String(digits));
}
}
} return num;
}
}
Idea 3. 虽然也感觉和LT31 Next Permutation有相似的,没有找出规律,网上看到的妙法,从左到右找到第一个valley, 继续valley后找到最大值作为要交换的rightIndex, 然后再从左到右找一个小于最大值的作为leftIndex, swap(digits, leftIndex, rightIndex); LT31是从右到左找第一个peak, peak的左边是rightIndex, 再从右到左找第一个比digits[rightIndex]小的作为leftIndex, 最后交换就是了.
Note. duplicates, rightMost index like 27736, 77236 > 72736, 又犯了错,下次记住这个test case啊
class Solution {
private void swap(char[] digits, int i, int j) {
char c = digits[i];
digits[i] = digits[j];
digits[j] = c;
}
public int maximumSwap(int num) {
char[] digits = Integer.toString(num).toCharArray(); int rightIndex = 1;
while(rightIndex < digits.length && digits[rightIndex-1] >= digits[rightIndex]) {
++rightIndex;
} if(rightIndex == digits.length) {
return num;
} for(int i = rightIndex+1; i < digits.length; ++i) {
if(digits[i] >= digits[rightIndex]) {
rightIndex = i;
}
} for(int i = 0; i < digits.length; ++i) {
if(digits[i] < digits[rightIndex]) {
swap(digits, i, rightIndex);
break;
}
} return Integer.parseInt(new String(digits));
}
}
Maximum Swap LT670的更多相关文章
- LC 670. Maximum Swap
Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...
- [LeetCode] Maximum Swap 最大置换
Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...
- [Swift]LeetCode670. 最大交换 | Maximum Swap
Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...
- 670. Maximum Swap
Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...
- 670. Maximum Swap 允许交换一个数 求最大值
[抄题]: Given a non-negative integer, you could swap two digits at most once to get the maximum valued ...
- LeetCode Maximum Swap
原题链接在这里:https://leetcode.com/problems/maximum-swap/description/ 题目: Given a non-negative integer, yo ...
- [LeetCode] 670. Maximum Swap 最大置换
Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...
- 1095. Maximum Swap —— Weekly Challenge
题目限定输入是[0, 10^8],因而不用考虑负数或者越界情况,算是减小了难度. public class Solution { /** * @param num: a non-negative in ...
- 最大交换 Maximum Swap
2018-07-28 16:52:20 问题描述: 问题求解: 使用bucket数组来记录每个数最后出现的位置,然后从左向右遍历一遍即可. public int maximumSwap(int num ...
随机推荐
- Java计算计算活了多少天
Java计算计算活了多少天 思路: 1.输入你的出现日期: 2.利用日期转换,将字符串转换成date类型 3.然后将date时间换成毫秒时间 4.然后获取当前毫秒时间: 5.最后计算出来到这个时间多少 ...
- mybatis 插入返回自增后的id
//serviceImpl int customerId = customerDao.insertDynamic(customer1); System.out.println("id==== ...
- Shader基础(固定管线着色器)
在Shader的编码中,要养成不加空格的习惯,否则会有时候出现一些错误 固定管线着色器: 优点:实现简单 缺点:处理的效果比较差 //设置Shader的路径 Shader "MyFixedS ...
- metasploitable使用
DVWA默认的用户有5个,用户名密码如下(一个足以): admin/password gordonb/abc123 1337/charley pablo/letmein smithy/password
- linux按键映射
.Xmodmap keycode = Mode_switch keysym k = k K Left keysym j = j J Down keysym l = l L Right keysym h ...
- python中matplotlib 的简单使用
1.简单折线图的画图,轴标签.图的颜色,风格,等等参数,本文只介绍最常用的几个参数: import matplotlib.pyplot as plt import numpy as np x = np ...
- DHCP server 冒充及DOS攻击处理方案
一.DHCP服务器在运维上存在的常见问题: 1. DHCP服务器冒充 在DHCP服务器和客户端之间没有认证机制,如果在DHCP server覆盖的网络上随意接入一个DHCP server,就有可能造成 ...
- swift - 快速代码块 - 创建 tableview等一些控件 基本属性
1.创建tableview private lazy var cellId = "cellId" fileprivate lazy var tv : UITableView = { ...
- python颜色及背景
linux终端中的颜色是用转义序列控制的,转义序列是以ESC开头,可以用\033完成相同的工作(ESC的ASCII码用十进制表示就是27,等于用八进制表示的33). 书写格式,和相关说明如下: 1 2 ...
- Java06-java基础语法(五)数组
Java06-java基础语法(五)数组 一.循环的嵌套 在一个循环体内部再含有一个或多个循环 强调:内循环全部做完以后再去执行下一次的外循环 int k = 0; for(int i = 0; i& ...