数据结构--旋转数组

数据结构–旋转数组

有一个二维数组,请在不使用额外变量的情况下,完成对其选择90度
在这里插入图片描述

分析(旋转遍历法)

找到坐标变化规律

设这个矩阵长宽为N
我们可以知道第一行,将会变为倒数第一列
那么对于第I行,就会变为倒数第I列
对于第J列,就会变成第几行
所以对应关系为[I,J] -> [J,N-1-I] (这里为N-1是因为坐标从0开始所以最大的坐标为N-1)
旋转时每个坐标都有与之对应的3个坐标点,总共4个坐标点相互互换位置
[I,J] -> [J,N-1-I] -> [N-1-I,N-1-J] -> [N-1-J,I] -> [I,J]
所以我们只需要遍历 1/4 的坐标,然后变动对应的 4 组坐标 即可实现全部坐标的替换

两种选择方式

在这里插入图片描述
正方形旋转(图1):
对于N为偶数是便于旋转的,但是如果N为奇数,则不好使用统一的代码来管理(只能旋转图1中的那六个点),对于重复的部分,不能旋转,因为如图所示有一个小正方形,你不能用自己的元素旋转到自己那里
等腰三角形旋转(图2):
图二的遍历范围为 I(0->N),J(I->N-1-I) 总面积为 N * N/2 * 1/2=N方/4(三角形面积)
相当于把正方形分为 4 个等腰直角三角形
这种分割方式即使 N 为奇数也可以正常等分,图例中可以在每个三角形中取到独占的 6 个点
需要注意的是(0,4),(1,3) 这两个点不能包含起来 所以遍历的时候判断条件不能等于

所以使用图 2 的遍历方式 把遍历到的每个点和与之对应的其他三个点互换值即可实现旋转

实现不使用额外变量

通过:
A = A+B
B = A-B
A = A-B
这三步之后我们就实现了不占用额外的内存空间,交换了 AB 的值

使用位运算
通过使用两次^完成交换

代码实例

有临时变量(自己太菜了)

class Solution {
    public void rotate(int[][] matrix) {
        int length = matrix.length;
        int [][] temp = new int[length][length];
        for(int i = 0;i < length;i++){
        for(int j = length - 1;j >= 0;j--){
            temp[i][length - j - 1] = matrix[j][i];
        }
        }
        for(int i = 0;i < length; i++){
            matrix[i] = Arrays.copyOf(temp[i],length)
        }
    }
}

位运算版本

class Solution {
    public void rotate(int[][] matrix) {
        for (int i = 0; i < matrix.length/2; i++) {
            for (int j = i; j < matrix.length - (i + 1); j++) {
                // 相邻交换一次
                matrix[i][j] = matrix[i][j]^matrix[j][matrix.length-1-i];
                matrix[j][matrix.length-1-i] = matrix[i][j]^matrix[j][matrix.length-1-i];
                matrix[i][j] = matrix[i][j]^matrix[j][matrix.length-1-i];
                // 对角交换一次
                matrix[i][j] = matrix[i][j]^matrix[matrix.length-1-i][matrix.length-1-j];
                matrix[matrix.length-1-i][matrix.length-1-j] = matrix[i][j]^matrix[matrix.length-1-i][matrix.length-1-j];
                matrix[i][j] = matrix[i][j]^matrix[matrix.length-1-i][matrix.length-1-j];
                // 相邻交换一次
                matrix[i][j] = matrix[i][j]^matrix[matrix.length-1-j][i];
                matrix[matrix.length-1-j][i] = matrix[i][j]^matrix[matrix.length-1-j][i];
                matrix[i][j] = matrix[i][j]^matrix[matrix.length-1-j][i];
            }
        }
    }
}

旋转遍历法

    public static void rotate(int[][] matrix) {
        for (int i = 0; i < matrix.length - 1; i++) {
            for (int j = i; j < matrix.length - 1 - i; j++) {
                exchange(matrix, i, j, j, matrix.length - 1 - i);
                exchange(matrix, i, j, matrix.length - 1 - i, matrix.length - 1 - j);
                exchange(matrix, i, j, matrix.length - 1 - j, i);
            }
        }
    }

    public static void exchange(int[][] matrix, int i1, int j1, int i2, int j2) {
        matrix[i1][j1] = matrix[i1][j1] + matrix[i2][j2];
        matrix[i2][j2] = matrix[i1][j1] - matrix[i2][j2];
        matrix[i1][j1] = matrix[i1][j1] - matrix[i2][j2];
    }
}

旋转遍历法(位运算)

public static void rotate(int[][] matrix) {
        for (int i = 0; i < matrix.length - 1; i++) {
            for (int j = i; j < matrix.length - 1 - i; j++) {
                exchange(matrix, i, j, j, matrix.length - 1 - i);
                exchange(matrix, i, j, matrix.length - 1 - i, matrix.length - 1 - j);
                exchange(matrix, i, j, matrix.length - 1 - j, i);
            }
        }
    }

    public static void exchange(int[][] matrix, int i1, int j1, int i2, int j2) {
        matrix[i1][j1] = matrix[i1][j1] ^ matrix[i2][j2];
        matrix[i2][j2] = matrix[i1][j1] ^ matrix[i2][j2];
        matrix[i1][j1] = matrix[i1][j1] ^ matrix[i2][j2];
    }

相关推荐

  1. 数组旋转变换分析

    2024-03-31 19:38:01       51 阅读
  2. 搜索旋转排序数组

    2024-03-31 19:38:01       43 阅读
  3. 搜索旋转排序数组

    2024-03-31 19:38:01       32 阅读
  4. 33.搜索旋转排序数组

    2024-03-31 19:38:01       66 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-03-31 19:38:01       94 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-03-31 19:38:01       100 阅读
  3. 在Django里面运行非项目文件

    2024-03-31 19:38:01       82 阅读
  4. Python语言-面向对象

    2024-03-31 19:38:01       91 阅读

热门阅读

  1. 【Webflux】实现全局返回Long转String

    2024-03-31 19:38:01       37 阅读
  2. 面试中会被问到的GIT问题解答(含答案)

    2024-03-31 19:38:01       33 阅读
  3. 在数据开发项目中使用Hive的场景和风险

    2024-03-31 19:38:01       35 阅读
  4. python基础练习题6

    2024-03-31 19:38:01       39 阅读
  5. 组件递归和动态component

    2024-03-31 19:38:01       40 阅读
  6. Product of Binary Decimals(搜索,暴力枚举,打表预处理)

    2024-03-31 19:38:01       29 阅读
  7. 2024系统架构师---解释器架构风格的概念与应用

    2024-03-31 19:38:01       37 阅读