当前位置: 首页 > news >正文

【每日一题Day166】LC1053交换一次的先前排列 | 贪心

交换一次的先前排列【LC1053】

给你一个正整数数组 arr(可能存在重复的元素),请你返回可在 一次交换(交换两数字 arr[i]arr[j] 的位置)后得到的、按字典序排列小于 arr 的最大排列。

如果无法这么操作,就请返回原数组。

虽然写出来了,但是花了55分钟…
还是有思路但是思路不清晰,然后直接敲代码,改来改去,老毛病了
笔试绝对ggg
看了别人的贪心,感觉自己笨笨的

  • 思路:贪心

    假设交换的左边元素为arr[i]arr[i]arr[i],右边的元素为arr[j]arr[j]arr[j]

    • 怎样交换一次可以使字典序减小?

      交换元素arr[i]>arr[j]arr[i]>arr[j]arr[i]>arr[j]时,可以使字典序较小,所以数组必须是非递增的

    • 如何使字典序小于原数组的情况下最大?【贪心】

      保留前面高位部分的数组,尽可能交换低位部分的数组,即尽可能最小化jjj的同时,最大化iii

      枚举每个右端点,此时的右端点rrr必须小于等于nums[j]nums[j]nums[j],找到在[i,r−1][i,r-1][i,r1]范围内,从右往左第一个大于其的左端点进行交换

      • 如果arr[r]>arr[j]arr[r] > arr[j]arr[r]>arr[j], 那么从右往左第一个大于arr[r]arr[r]arr[r]的左端点一定在i的左边包括i,那么我们无法获得比目前的排列更大的排列
      • 如果arr[r]==arr[j]arr[r] == arr[j]arr[r]==arr[j], 那么从右往左第一个大于arr[r]arr[r]arr[r]的左端点还是为iii,只需要修改右端点
      • 如果arr[r]<arr[j]arr[r] < arr[j]arr[r]<arr[j], 那么lll需要在[i+1,r−1][i+1,r-1][i+1,r1]的范围内才可能是字典序增大
  • 实现

    class Solution {public int[] prevPermOpt1(int[] arr) {int n = arr.length;int l = 0;// 升序数组本身就是最小排列while (l < n - 1 && arr[l] <= arr[l + 1]){l++;}if (l == n - 1) return arr; // 升序数组// 非升序数组 枚举每个右端点 找到从右往左第一个大于其的左端点进行交换// 之后交换的右端点必须小于等于arr[j],并且左端点l必须大于i才能使交换结果变小// 如果arr[r] > arr[j], 那么从右往左第一个大于arr[r]的左端点一定在i的左边包括i,那么我们无法获得比目前的排列更大的排列// 如果arr[r] == arr[j], 那么从右往左第一个大于arr[r]的左端点还是为i,只需要修改右端点// 如果arr[r] < arr[j], 那么l需要在[i+1,r-1]的范围内才可能是字典序增大int i = -1, j = -1;// 记录最终的交换结果for (int r = n - 1; r > i; r--){if (j != -1 && arr[r] > arr[j]) continue;if (j != -1 && arr[r] == arr[j]) {j = r;continue;}for (l = r - 1; l >= (i != -1 ? i + 1 : 0); l--){if (arr[l] > arr[r]){i = l;j = r;break;}}}// 交换int tmp = arr[i];arr[i] = arr[j];arr[j] = tmp;return arr;}
    }
    
    • 复杂度
      • 时间复杂度:O(n2)O(n^2)O(n2)
      • 空间复杂度:O(1)O(1)O(1)
  • 别人的贪心
    我们先从右到左遍历数组,找到第一个满足 arr[i−1]>arr[i]arr[i−1]>arr[i]arr[i1]>arr[i]的下标 iii,此时 arr[i−1]arr[i−1]arr[i1]就是我们要交换的数字,我们再从右到左遍历数组,找到第一个满足 arr[j]<arr[i−1]arr[j]<arr[i−1]arr[j]<arr[i1]arr[j]≠arr[j−1]arr[j] \neq arr[j - 1]arr[j]=arr[j1] 的下标 j,此时我们交换 arr[i−1]arr[i−1]arr[i1]arr[j]arr[j]arr[j] 后返回即可。

class Solution {public int[] prevPermOpt1(int[] arr) {int n = arr.length;for (int i = n - 1; i > 0; --i) {if (arr[i - 1] > arr[i]) {for (int j = n - 1; j > i - 1; --j) {if (arr[j] < arr[i - 1] && arr[j] != arr[j - 1]) {int t = arr[i - 1];arr[i - 1] = arr[j];arr[j] = t;return arr;}}}}return arr;}
}作者:ylb
链接:https://leetcode.cn/problems/previous-permutation-with-one-swap/solutions/2205403/python3javacgotypescript-yi-ti-yi-jie-ta-pxxt/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
  • 复杂度
    • 时间复杂度:O(n)O(n)O(n)
    • 空间复杂度:O(1)O(1)O(1)
http://www.lryc.cn/news/56254.html

相关文章:

  • Canal增量数据订阅和消费——原理详解
  • 为什么要使用线程池
  • 在云服务部署前后端以及上传数据库
  • Onedrive for Business迁移方案 | 分享一
  • pt01数据类型、语句选择
  • ChatGPT 存在很大的隐私问题
  • 图的迭代深度优先遍历
  • 华为OD机试-开放日活动-2022Q4 A卷-Py/Java/JS
  • 两亲性聚合物:Lauric acid PEG Maleimide,Mal-PEG-Lauric acid,月桂酸PEG马来酰亚胺,试剂知识分享
  • FB使用入口点函数例子
  • 学习周报4/9
  • 49天精通Java,第14天,Java泛型方法的定义和使用
  • 20230402英语学习
  • Java知识复习(十七)SpringCloud
  • MySQL 数据库操作
  • Cesium更换地球背景
  • 测试人员的瓶颈期
  • HTML5 <form> 标签
  • 编译技术-词法理论
  • 【20】核心易中期刊推荐——计算机科学电子通信(EI索引)
  • Vue 3.0 风格指南 2
  • ChatGPT遭多国调查,OpenAI凌晨就安全问题发文,GPT-5要暂缓?
  • 网络安全书籍推荐
  • 【独家】华为OD机试 - 狼羊过河 or 羊、狼、农夫过河(C 语言解题)
  • 东八区的 springboot 如何配置序列化
  • 论文阅读_LLaMA
  • 腾讯空降测试工程师,绩效次次拿S,真是砂纸擦屁股,给我露了一手啊
  • 真题详解(计算机总线)-软件设计(四十五)
  • 剪格子
  • 【Nowcoder-BC146.添加逗号 -OR63.删除公共字符】