• 数组问题之《下一个排列》、《旋转图像》以及二分查找之《搜索二维矩阵》


    一、下一个排列

    1、题目说明

            实现获取下一个排列的函数,算法需要将给定数字序列重新排列成字典序中下一个更大的排列。

            如果不存在下一个更大的排列,则将数字重新排列成最小的排列(即升序排列)。

            必须原地修改,只允许使用额外常数空间。

    以下是一些例子,输入位于左侧列,其相应输出位于右侧列。

    1. 1,2,3 → 1,3,2
    2. 3,2,1 → 1,2,3
    3. 1,1,5 → 1,5,1

    2、方法一:暴力法

            最简单的想法就是暴力枚举,我们找出由给定数组的元素形成的列表的每个可能的排列,并找出比给定的排列更大的排列。

            但是这个方法要求我们找出所有可能的排列,这需要很长时间,实施起来也很复杂。因此,这种算法不能满足要求。 我们跳过它的实现,直接采用正确的方法

            复杂度分析:

                    时间复杂度:O(n!),可能的排列总计有 n! 个。

                    空间复杂度:O(n),因为数组将用于存储排列。


    3、方法二:一遍扫描

    首先,我们观察到对于任何给定序列的降序排列,就不会有下一个更大的排列。

    例如,以下数组不可能有下一个排列:

    [9, 5, 4, 3, 1]

    这时应该直接返回升序排列。

            所以对于一般的情况,如果有一个“升序子序列”,那么就一定可以找到它的下一个排列。具体来说,需要从右边找到第一对两个连续的数字 a[i] 和 a[i-1],它们满足 a[i]>a[i-1]。

            所以一个思路是,找到最后一个的“正序”排列的子序列,把它改成下一个排列就行了。

     不过具体操作会发现,如果正序子序列后没数了,那么子序列的“下一个”一定就是整个序列的“下一个”,这样做没问题;但如果后面还有逆序排列的数,这样就不对了。比如:

    [1,3,8,7,6,2]

            最后的正序子序列是[1,3,8],但显然不能直接换成[1,8,3]就完事了;而是应该考虑把3换成后面比3大、但比8小的数,而且要选最小的那个(6)。接下来,还要让6之后的所有数,做一个升序排列,得到结果:

    [1,6,2,3,7,8]

    代码实现:

    1. public void nextPermutation(int[] nums) {
    2. int k = nums.length - 2;
    3. while( k >= 0 && nums[k] >= nums[k+1] )
    4. k--;
    5. // 如果全部降序,以最小序列输出
    6. if( k < 0 ){
    7. Arrays.sort(nums);
    8. return;
    9. }
    10. int i = k + 2;
    11. while( i < nums.length && nums[i] > nums[k] )
    12. i++;
    13. // 交换nums[k]和找到的nums[i-1]
    14. int temp = nums[k];
    15. nums[k] = nums[i-1];
    16. nums[i-1] = temp;
    17. // k以后剩余的部分反转成升序
    18. int start = k + 1, end = nums.length - 1;
    19. while( start < end ){
    20. int reTemp = nums[start];
    21. nums[start] = nums[end];
    22. nums[end] = reTemp;
    23. start++;
    24. end--;
    25. }
    26. }

    复杂度分析:

            时间复杂度:

                    O(N),其中 NN 为给定序列的长度。我们至多只需要扫描两次序列,以及进行一次反转操作。

            空间复杂度:O(1),只需要常数的空间存放若干变量。


    二、旋转图像

    1、题目说明

            给定一个 n × n 的二维矩阵表示一个图像。

            将图像顺时针旋转 90 度。

     说明

            你必须在原地旋转图像,这意味着你需要直接修改输入的二维矩阵。请不要使用另一个矩阵来旋转图像。

    示例:

            

    1. 给定 matrix =
    2. [
    3. [1,2,3],
    4. [4,5,6],
    5. [7,8,9]
    6. ],
    7. 原地旋转输入矩阵,使其变为:
    8. [
    9. [7,4,1],
    10. [8,5,2],
    11. [9,6,3]
    12. ]
    1. 给定 matrix =
    2. [
    3. [ 5, 1, 9,11],
    4. [ 2, 4, 8,10],
    5. [13, 3, 6, 7],
    6. [15,14,12,16]
    7. ],
    8. 原地旋转输入矩阵,使其变为:
    9. [
    10. [15,13, 2, 5],
    11. [14, 3, 4, 1],
    12. [12, 6, 8, 9],
    13. [16, 7,10,11]
    14. ]

    2、分析

            旋转图像,这个应用在图片处理的过程中,非常常见。我们知道对于计算机而言,图像,其实就是一组像素点的集合(所谓点阵),所以图像旋转的问题,本质上就是一个二维数组的旋转问题。


    3、方法一:数学方法(转置再翻转)

    我们可以利用矩阵的特性。所谓顺时针旋转,其实就是先转置矩阵,然后翻转每一行。

    代码如下:

    1. public void rotate(int[][] matrix) {
    2. int n = matrix.length;//获取二维数组的长度
    3. // 转置矩阵
    4. /***
    5. * 例如:二维数组是
    6. * [
    7. * [1,2,3],
    8. * [4,5,6],
    9. * [7,8,9]
    10. * ],
    11. * 实现如下效果:
    12. *[
    13. * [1,4,7],
    14. * [2,5,8],
    15. * [3,6,9]
    16. * ],
    17. */
    18. for (int i = 0; i < n; i++)
    19. for (int j = i; j < n; j++) {
    20. int tmp = matrix[i][j];
    21. matrix[i][j] = matrix[j][i];
    22. matrix[j][i] = tmp;
    23. }
    24. // 翻转行
    25. /***
    26. * 将上面得到的
    27. * [
    28. * [1,4,7],
    29. * [2,5,8],
    30. * [3,6,9]
    31. * ],
    32. * 转换为:
    33. * [
    34. * [7,4,1],
    35. * [8,5,2],
    36. * [9,6,3]
    37. * ]
    38. */
    39. for( int i = 0; i < n; i++ ){
    40. for( int j = 0; j < n/2; j++ ){
    41. int tmp = matrix[i][j];
    42. matrix[i][j] = matrix[i][n-j-1];
    43. matrix[i][n-j-1] = tmp;
    44. }
    45. }
    46. }

    复杂度分析:

    • 时间复杂度:O(N^2)

    这个简单的方法已经能达到最优的时间复杂度O(N^2) ,因为既然是旋转,那么每个点都应该遍历到,N^2的复杂度不可避免。

    • 空间复杂度:O(1)。旋转操作是原地完成的,只耗费常数空间。

    4、方法二:分治(分为四部分旋转)

            方法 1 使用了两次矩阵操作,能不能只使用一次操作的方法完成旋转呢?

            为了实现这一点,我们来研究每个元素在旋转的过程中如何移动。

            第一次旋转:1,3,9,7

            第二次旋转:2,6,8,4

    这提供给我们了一个思路,可以将给定的矩阵分成四个矩形并且将原问题划归为旋转这些矩形的问题。这其实就是分治的思想。

    具体解法也很直接,可以在每一个矩形中遍历元素,并且在长度为 4 的临时列表中移动它们。

     代码如下:

    1. public void rotate(int[][] matrix) {
    2. int n = matrix.length;//获取二维数组长度
    3. for (int i = 0; i < n / 2 + n % 2; i++) {
    4. for (int j = 0; j < n / 2; j++) {
    5. int[] tmp = new int[4];
    6. int row = i;
    7. int col = j;
    8. for (int k = 0; k < 4; k++) {
    9. tmp[k] = matrix[row][col];
    10. // 定位下一个数
    11. int x = row;
    12. row = col;
    13. col = n - 1 - x;
    14. }
    15. for (int k = 0; k < 4; k++) {
    16. matrix[row][col] = tmp[(k + 3) % 4];
    17. int x = row;
    18. row = col;
    19. col = n - 1 - x;
    20. }
    21. }
    22. }
    23. }

    复杂度分析:

    • 时间复杂度:O(N^2) 是两重循环的复杂度。
    • 空间复杂度:O(1) 由于我们在一次循环中的操作是“就地”完成的,并且我们只用了长度为 4 的临时列表做辅助。

    5、方法三:分治法改进(单次循环内完成旋转)

            大家可能也发现了,我们其实没有必要分成4个矩阵来旋转。这四个矩阵的对应关系,其实是一目了然的,我们完全可以在一次循环内,把所有元素都旋转到位。

            因为旋转的时候,是上下、左右分别对称的,所以我们遍历元素的时候,只要遍历一半行、一半列就可以了(1/4元素)。

    代码如下:

    1. public void rotate(int[][] matrix) {
    2. int n = matrix.length;
    3. // 不区分子矩阵,直接遍历每一个元素
    4. for( int i = 0; i < (n + 1)/2; i++ ){
    5. for( int j = 0; j < n/2; j++ ){
    6. int temp = matrix[i][j];
    7. matrix[i][j] = matrix[n-j-1][i];
    8. matrix[n-j-1][i] = matrix[n-i-1][n-j-1]
    9. matrix[n-i-1][n-j-1] = matrix[j][n-i-1];;
    10. matrix[j][n-i-1] = temp;
    11. }
    12. }
    13. }

    复杂度分析:

    • 时间复杂度:O(N^2),是两重循环的复杂度。
    • 空间复杂度:O(1)。我们在一次循环中的操作是“就地”完成的。

    三、二分查找

     1、简介

            二分查找也称折半查找(Binary Search),它是一种效率较高的查找方法,前提是数据结构必须先排好序,可以在对数时间复杂度内完成查找。

            二分查找事实上采用的就是一种分治策略,它充分利用了元素间的次序关系,可在最坏的情况下用O(log n)完成搜索任务。

            它的基本思想是:假设数组元素呈升序排列,将n个元素分成个数大致相同的两半,取a[n/2]与欲查找的x作比较,如果x=a[n/2]则找到x,算法终止;如 果x数组a的左半部继续搜索x;如果x>a[n/2],则我们只要在数组a的右 半部继续搜索x。

    二分查找问题也是面试中经常考到的问题,虽然它的思想很简单,但写好二分查找算法并不是一件容易的事情。


    2、实现一个对int数组的二分查找(注意:传递进来的数组必须有序)。

    1. public static int binarySearch(int[] a, int key){
    2. int low = 0;
    3. int high = a.length - 1;
    4. if ( key < a[low] || key > a[high] )
    5. return -1;
    6. while ( low <= high){
    7. int mid = ( low + high ) / 2;
    8. if( a[mid] < key)
    9. low = mid + 1;
    10. else if( a[mid] > key )
    11. high = mid - 1;
    12. else
    13. return mid; // 查找成功
    14. }
    15. // 未能找到
    16. return -1;
    17. }

    3、用递归的方式实现

    1. public static int binarySearch(int[] a, int key, int fromIndex, int toIndex){
    2. if ( key < a[fromIndex] || key > a[toIndex] || fromIndex > toIndex)
    3. return -1;
    4. int mid = ( fromIndex + toIndex ) / 2;
    5. if ( a[mid] < key )
    6. return binarySearch(a, key, mid + 1, toIndex);
    7. else if ( a[mid] > key )
    8. return binarySearch(a, key, fromIndex, mid - 1);
    9. else
    10. return mid;
    11. }

    我们总结一下二分查找:

    • 优点是比较次数少,查找速度快,平均性能好;
    • 缺点是要求待查表为有序表,且插入删除困难。

    因此,二分查找方法适用于不经常变动而查找频繁的有序列表。使用条件:查找序列是顺序结构,有序。


    四、二分查找案例之搜索二维矩阵

    1、编写一个高效的算法来判断 m x n 矩阵中,是否存在一个目标值。该矩阵具有如下特性:

    • 每行中的整数从左到右按升序排列
    • 每行的第一个整数大于前一行的最后一个整数

    示例 1

     

    1. 输入:matrix = [[1,3,5,7],[10,11,16,20],[23,30,34,50]], target = 3
    2. 输出:true
    3. 示例 2:
    4. 输入:matrix = [[1,3,5,7],[10,11,16,20],[23,30,34,50]], target = 13
    5. 输出:false
    6. 示例 3:
    7. 输入:matrix = [], target = 0
    8. 输出:false

    提示:

    • m == matrix.length
    • n == matrix[i].length
    • 0 <= m, n <= 100
    • -104 <= matrix[i][j], target <= 104

    2、需求分析

            既然这是一个查找元素的问题,并且数组已经排好序,我们自然可以想到用二分查找是一个高效的查找方式。

    输入的 m x n 矩阵可以视为长度为 m x n的有序数组:

    行列坐标为(row, col)的元素,展开之后索引下标为idx = row * n + col反过来,对于一维下标为idx的元素,对应二维数组中的坐标就应该是:

    row = idx / n;  col = idx % n;


    3、代码实现

    1. public boolean searchMatrix(int[][] matrix, int target){
    2. int m = matrix.length;
    3. if (m == 0) return false;
    4. int n = matrix[0].length;
    5. int left = 0;
    6. int right = m * n - 1;
    7. // 二分查找,定义左右指针
    8. while ( left <= right ){
    9. int midIdx = (left + right) / 2;
    10. int midElement = matrix[midIdx/n][midIdx%n];
    11. if ( midElement < target )
    12. left = midIdx + 1;
    13. else if ( midElement > target )
    14. right = midIdx - 1;
    15. else
    16. return true; // 找到target
    17. }
    18. return false;
    19. }

    复杂度分析:​​​​​​​

    • 时间复杂度 : 由于是标准的二分查找,时间复杂度为O(log(m n))
    • 空间复杂度没有用到额外的空间,复杂度为O(1)
  • 相关阅读:
    grpc、https、oauth2等认证专栏实战11:授权码模式介绍
    Kubernetes(k8s)的流量负载组件Ingress安装与使用
    硬件笔记:组装“固态 U 盘”的八年,从 100 块到 1000 块
    【C++】运算符重载 ② ( 类内部定义云算符重载 - 成员函数 | 类外部定义运算符重载 - 全局函数 | 可重载的运算符 )
    Vue 组件之间的通信,动态组件和插槽
    linux 安装配置github
    【Python】计算机视觉 手掌图片穴位识别
    【python】python内置函数——zip()打包可迭代对象为一个个元组,并返回这些元组组成的列表
    我的备考分享:良好的企业治理需要CGEIT
    JavaScript-方法的定义和调用、apply、内部对象
  • 原文地址:https://blog.csdn.net/K_520_W/article/details/126085574