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:

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

  1. LC 670. Maximum Swap

    Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...

  2. [LeetCode] Maximum Swap 最大置换

    Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...

  3. [Swift]LeetCode670. 最大交换 | Maximum Swap

    Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...

  4. 670. Maximum Swap

    Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...

  5. 670. Maximum Swap 允许交换一个数 求最大值

    [抄题]: Given a non-negative integer, you could swap two digits at most once to get the maximum valued ...

  6. LeetCode Maximum Swap

    原题链接在这里:https://leetcode.com/problems/maximum-swap/description/ 题目: Given a non-negative integer, yo ...

  7. [LeetCode] 670. Maximum Swap 最大置换

    Given a non-negative integer, you could swap two digits at most once to get the maximum valued numbe ...

  8. 1095. Maximum Swap —— Weekly Challenge

    题目限定输入是[0, 10^8],因而不用考虑负数或者越界情况,算是减小了难度. public class Solution { /** * @param num: a non-negative in ...

  9. 最大交换 Maximum Swap

    2018-07-28 16:52:20 问题描述: 问题求解: 使用bucket数组来记录每个数最后出现的位置,然后从左向右遍历一遍即可. public int maximumSwap(int num ...

随机推荐

  1. Java多态的向上转型和向下转型

    Java多态的向上转型和向下转型 向上转型:(子类转为父类,自动转型) 格式 :父类类型 变量名 = new 子类类型(); 注:子类赋值给父类,那这时这个父类就能调用子类的方法 向下转型:(父类转为 ...

  2. PLM修改数据库密码

    在WINDCHILL SHELL中执行以下命令: xconfmanager -s wt.pom.dbPassword=数据库密码 -t db/db.properties -p 重启服务

  3. http4e eclipse plugin 插件介绍

    感谢作者的分享: http://blog.csdn.net/wiker_yong/article/details/10066905 以及作者的破解jar.目前看网站留言说已经git了. 官网链接地址: ...

  4. break

    public class b {   public static void main(String[] args) {  int i=0;  for(;i<=10;i++){   if (i&g ...

  5. JMeter学习(九)FTP测试计划(转载)

    转载自 http://www.cnblogs.com/yangxia-test FTP服务主要提供上传和下载功能.有时间需要我们测试服务器上传和下载的性能.在这里我通过JMeter做一个FTP测试计划 ...

  6. Android开发之getX,getRawX,getWidth,getTranslationX等的区别

    转载请注明出处:http://blog.csdn.net/dmk877/article/details/51550031      好久没写博客了,最近工作确实挺忙的,刚刚结束了一个TV项目的开发,对 ...

  7. UA判断跳转

    <script type="text/javascript"> UA = navigator.userAgent.toLowerCase(); url = window ...

  8. DES算法实现(C++版)

    #include "memory.h" #include "stdio.h" enum {encrypt,decrypt};//ENCRYPT:加密,DECRY ...

  9. python 学习笔记---Locust 测试服务端性能

    由于人工智能的热度, python目前已经成为最受欢迎的编程语言,一度已经超越Java . 本文将介绍开源的python 测试工具: locust 使用步骤: 1. 安装python 3.0以上版本 ...

  10. FortiGate软件版本升级

    1.Web界面升级 1)注意:升级前,务必做好配置备份 2)要点 1.FortiGate防火墙的每款型号都有单独的版本文件,升级前务必确认下当前的设备型号: 2.升级包的后缀名必须为.out,前缀任意 ...