目录
[1. 旋转数组](#1. 旋转数组)
[1.1. 题目描述](#1.1. 题目描述)
[1.2. 解题思路](#1.2. 解题思路)
[2. 螺旋矩阵](#2. 螺旋矩阵)
[2.1. 题目描述](#2.1. 题目描述)
[2.2. 解题思路](#2.2. 解题思路)
[3. 顺时针旋转矩阵](#3. 顺时针旋转矩阵)
[3.1. 题目描述](#3.1. 题目描述)
[3.2. 解题思路](#3.2. 解题思路)
[4. 设计LFU缓存结构](#4. 设计LFU缓存结构)
[4.1. 题目描述](#4.1. 题目描述)
[4.2. 解题思路](#4.2. 解题思路)
[5. 顺时针打印矩阵](#5. 顺时针打印矩阵)
[5.1. 题目描述](#5.1. 题目描述)
[5.2. 解题思路](#5.2. 解题思路)
[6. 验证图书取出顺序](#6. 验证图书取出顺序)
[6.1. 题目描述](#6.1. 题目描述)
[6.2. 解题思路](#6.2. 解题思路)
1. 旋转数组
1.1. 题目描述
data:image/s3,"s3://crabby-images/6483a/6483ab158320b6b81fe445ecd58b0274953afa2f" alt=""
1.2. 解题思路
方法一:三次翻转(推荐使用)
data:image/s3,"s3://crabby-images/5adce/5adce8112a73a15b7086359769eddc100c8556f9" alt=""
data:image/s3,"s3://crabby-images/3b681/3b681f751a0740e698a6eecb7264aaf1b1481be6" alt=""
Java代码实现:
public class Solution {
public int[] solve (int n, int m, int[] a) {
//取余,因为每次长度为n的旋转数组相当于没有变化
m = m % n;
//第一次逆转全部数组元素
reverse(a, 0, n - 1);
//第二次只逆转开头m个
reverse(a, 0, m - 1);
//第三次只逆转结尾m个
reverse(a, m, n - 1);
return a;
}
//反转函数
public void reverse(int[] nums, int start, int end){
while(start < end){
swap(nums, start++, end--);
}
}
//交换函数
public void swap(int[] nums, int a, int b){
int temp = nums[a];
nums[a] = nums[b];
nums[b] = temp;
}
}
data:image/s3,"s3://crabby-images/69199/69199f8978f17158b64c09f992d73ad4c0ca0d48" alt=""
2. 螺旋矩阵
2.1. 题目描述
data:image/s3,"s3://crabby-images/a4387/a43879d02e8492d0f05d48e150185320fa57e75f" alt=""
2.2. 解题思路
方法一:边界模拟法(推荐使用)
data:image/s3,"s3://crabby-images/f3e4c/f3e4c7bbd7916a8cbb6618d37b5d16f8e9b7b5ed" alt=""
data:image/s3,"s3://crabby-images/29cef/29cefc16240d93f4106655161fd14fe2af2d609b" alt=""
Java代码实现:
import java.util.ArrayList;
public class Solution {
public ArrayList<Integer> spiralOrder(int[][] matrix) {
ArrayList<Integer> res = new ArrayList<>();
//先排除特殊情况
if(matrix.length == 0) {
return res;
}
//左边界
int left = 0;
//右边界
int right = matrix[0].length - 1;
//上边界
int up = 0;
//下边界
int down = matrix.length - 1;
//直到边界重合
while(left <= right && up <= down){
//上边界的从左到右
for(int i = left; i <= right; i++)
res.add(matrix[up][i]);
//上边界向下
up++;
if(up > down)
break;
//右边界的从上到下
for(int i = up; i <= down; i++)
res.add(matrix[i][right]);
//右边界向左
right--;
if(left > right)
break;
//下边界的从右到左
for(int i = right; i >= left; i--)
res.add(matrix[down][i]);
//下边界向上
down--;
if(up > down)
break;
//左边界的从下到上
for(int i = down; i >= up; i--)
res.add(matrix[i][left]);
//左边界向右
left++;
if(left > right)
break;
}
return res;
}
}
data:image/s3,"s3://crabby-images/853e5/853e54e6ae881faa254c232a33f25c54a7255330" alt=""
3. 顺时针旋转矩阵
3.1. 题目描述
data:image/s3,"s3://crabby-images/35fdc/35fdc7b395179ae264d1f0bfc729daca16689259" alt=""
3.2. 解题思路
方法一:倒置翻转(推荐使用)
data:image/s3,"s3://crabby-images/baf8a/baf8a03be3f383ac41c2c3e78679f3a781e83c0d" alt=""
data:image/s3,"s3://crabby-images/30e0f/30e0f6009c70b1c784f0b146226b0232f981f93c" alt=""
data:image/s3,"s3://crabby-images/28740/28740d8c7505871fba2c761ecf6bd64e41d571e5" alt=""
import java.util.*;
public class Solution {
public int[][] rotateMatrix(int[][] mat, int n) {
int length = mat.length;
//矩阵转置
for(int i = 0; i < length; ++i){
for(int j = 0; j < i; ++j){
//交换上三角与下三角对应的元素
int temp = mat[i][j];
mat[i][j] = mat[j][i];
mat[j][i] = temp;
}
}
//每行翻转
for (int i = 0; i < length; i++) {
for (int j = 0; j < length/2; j++){
int temp = mat[i][j];
mat[i][j] = mat[i][length - j - 1];
mat[i][length - j - 1] = temp;
}
}
return mat;
}
}
data:image/s3,"s3://crabby-images/0fd9d/0fd9d2bb64b3583ccb015f506615f88ab0745b01" alt=""
4. 设计LFU缓存结构
4.1. 题目描述
data:image/s3,"s3://crabby-images/22d17/22d17365e20246e0504a8120ee1469ea67674845" alt=""
4.2. 解题思路
方法一:双哈希表(推荐使用)
data:image/s3,"s3://crabby-images/df073/df073da63f9625d60db42c950af7c533ecf2570d" alt=""
data:image/s3,"s3://crabby-images/b8bbc/b8bbca892af1ed937a5608bcb76f56fda877123c" alt=""
data:image/s3,"s3://crabby-images/876ba/876ba4d1bed3f5e4a39667565174049c011f45cb" alt=""
data:image/s3,"s3://crabby-images/04ad8/04ad892f6ddbdf957edfb68ca6fd94705b704d16" alt=""
Java代码实现:
import java.util.*;
public class Solution {
//设置节点结构
static class Node{
int freq;
int key;
int val;
//初始化
public Node(int freq, int key, int val) {
this.freq = freq;
this.key = key;
this.val = val;
}
}
//频率到双向链表的哈希表
private Map<Integer, LinkedList<Node> > freq_mp = new HashMap<>();
//key到节点的哈希表
private Map<Integer, Node> mp = new HashMap<>();
//记录缓存剩余容量
private int size = 0;
//记录当前最小频次
private int min_freq = 0;
public int[] LFU (int[][] operators, int k) {
//构建初始化连接
//链表剩余大小
this.size = k;
//获取操作数
int len = (int)Arrays.stream(operators).filter(x -> x[0] == 2).count();
int[] res = new int[len];
//遍历所有操作
for(int i = 0, j = 0; i < operators.length; i++){
if(operators[i][0] == 1)
//set操作
set(operators[i][1], operators[i][2]);
else
//get操作
res[j++] = get(operators[i][1]);
}
return res;
}
//调用函数时更新频率或者val值
private void update(Node node, int key, int value) {
//找到频率
int freq = node.freq;
//原频率中删除该节点
freq_mp.get(freq).remove(node);
//哈希表中该频率已无节点,直接删除
if(freq_mp.get(freq).isEmpty()){
freq_mp.remove(freq);
//若当前频率为最小,最小频率加1
if(min_freq == freq)
min_freq++;
}
if(!freq_mp.containsKey(freq + 1))
freq_mp.put(freq + 1, new LinkedList<Node>());
//插入频率加一的双向链表表头,链表中对应:freq key value
freq_mp.get(freq + 1).addFirst(new Node(freq + 1, key, value));
mp.put(key, freq_mp.get(freq + 1).getFirst());
}
//set操作函数
private void set(int key, int value) {
//在哈希表中找到key值
if(mp.containsKey(key))
//若是哈希表中有,则更新值与频率
update(mp.get(key), key, value);
else{
//哈希表中没有,即链表中没有
if(size == 0){
//满容量取频率最低且最早的删掉
int oldkey = freq_mp.get(min_freq).getLast().key;
//频率哈希表的链表中删除
freq_mp.get(min_freq).removeLast();
if(freq_mp.get(min_freq).isEmpty())
freq_mp.remove(min_freq);
//链表哈希表中删除
mp.remove(oldkey);
}
//若有空闲则直接加入,容量减1
else
size--;
//最小频率置为1
min_freq = 1;
//在频率为1的双向链表表头插入该键
if(!freq_mp.containsKey(1))
freq_mp.put(1, new LinkedList<Node>());
freq_mp.get(1).addFirst(new Node(1, key, value));
//哈希表key值指向链表中该位置
mp.put(key, freq_mp.get(1).getFirst());
}
}
//get操作函数
private int get(int key) {
int res = -1;
//查找哈希表
if(mp.containsKey(key)){
Node node = mp.get(key);
//根据哈希表直接获取值
res = node.val;
//更新频率
update(node, key, res);
}
return res;
}
}
data:image/s3,"s3://crabby-images/bd6f3/bd6f3fbf32073abce39193d458ae535340fffa58" alt=""
5. 顺时针打印矩阵
5.1. 题目描述
data:image/s3,"s3://crabby-images/2699e/2699e572e0a6f294d485b82814feeaf23c9a1208" alt=""
5.2. 解题思路
方法一:模拟
data:image/s3,"s3://crabby-images/20149/20149577fcde510438b665cf8a00ed59a45616fa" alt=""
class Solution {
public int[] spiralArray(int[][] array) {
if (array == null || array.length == 0 || array[0].length == 0) {
return new int[0];
}
int rows = array.length, columns = array[0].length;
boolean[][] visited = new boolean[rows][columns];
int total = rows * columns;
int[] order = new int[total];
int row = 0, column = 0;
int[][] directions = {{0, 1}, {1, 0}, {0, -1}, {-1, 0}};
int directionIndex = 0;
for (int i = 0; i < total; i++) {
order[i] = array[row][column];
visited[row][column] = true;
int nextRow = row + directions[directionIndex][0], nextColumn = column + directions[directionIndex][1];
if (nextRow < 0 || nextRow >= rows || nextColumn < 0 || nextColumn >= columns || visited[nextRow][nextColumn]) {
directionIndex = (directionIndex + 1) % 4;
}
row += directions[directionIndex][0];
column += directions[directionIndex][1];
}
return order;
}
}
data:image/s3,"s3://crabby-images/3fa02/3fa027385fb74a70eb4014be3045e984d1484549" alt=""
方法二:按层模拟
data:image/s3,"s3://crabby-images/cd4af/cd4af8476211d5d53ad367a84a8c7d236d7dff55" alt=""
data:image/s3,"s3://crabby-images/d9def/d9def117f4cdbd2d49e9120f096e7a4e56ac3685" alt=""
class Solution {
public int[] spiralArray(int[][] array) {
if (array == null || array.length == 0 || array[0].length == 0) {
return new int[0];
}
int rows = array.length, columns = array[0].length;
int[] order = new int[rows * columns];
int index = 0;
int left = 0, right = columns - 1, top = 0, bottom = rows - 1;
while (left <= right && top <= bottom) {
for (int column = left; column <= right; column++) {
order[index++] = array[top][column];
}
for (int row = top + 1; row <= bottom; row++) {
order[index++] = array[row][right];
}
if (left < right && top < bottom) {
for (int column = right - 1; column > left; column--) {
order[index++] = array[bottom][column];
}
for (int row = bottom; row > top; row--) {
order[index++] = array[row][left];
}
}
left++;
right--;
top++;
bottom--;
}
return order;
}
}
data:image/s3,"s3://crabby-images/9b61f/9b61fd69d7179fa912527e9e1b8fbb3139a60267" alt=""
6. 验证图书取出顺序
6.1. 题目描述
data:image/s3,"s3://crabby-images/384eb/384ebe9062e690d4d17e85e35bfcbbdaa3df1452" alt=""
6.2. 解题思路
方法一:栈
data:image/s3,"s3://crabby-images/95e2f/95e2fe41c1d1540aa52c5fd3f78bae5623fa03cc" alt=""
data:image/s3,"s3://crabby-images/7b5de/7b5de6f076cbf0b121708cde19a689f9c62166b8" alt=""
data:image/s3,"s3://crabby-images/0c9c3/0c9c30693d29f9dd1e7afca1e8aa6726bdeb08c7" alt=""
class Solution {
public boolean validateBookSequences(int[] putIn, int[] takeOut) {
Stack<Integer> stack = new Stack<>();
int i = 0;
for(int num : putIn) {
stack.push(num); // num 入栈
while(!stack.isEmpty() && stack.peek() == takeOut[i]) { // 循环判断与出栈
stack.pop();
i++;
}
}
return stack.isEmpty();
}
}
data:image/s3,"s3://crabby-images/2740d/2740ded38b6deb5aea08617d1c14f4fff88b55ec" alt=""