文章目录
- 面试经典150题【31-40】
- 76.最小覆盖字串
- 36.有效的数独
- 54.螺旋矩阵
- 48.旋转图像
- 73.矩阵置零
- 289.生命游戏
- 383.赎金信
- 205.同构字符串
- 290.单词规律
- 242.有效的字母异位词
面试经典150题【31-40】
76.最小覆盖字串
基本思路很简单,就是先移动右边到合适位置。再移动左边到破戒,再移动位置到合适位置。
判断合适位置是 count0, 此时need[] 里面也都是0 或 <0 的元素。
public class LC76 {public String minWindow(String s, String t) {if(s == null || s.isEmpty() ||t==null || t.isEmpty()){return "";}int []need=new int[128];for(int i=0;i<t.length();i++){need[t.charAt(i)]++;}//l是当前左边界,r是当前右边界,size记录窗口大小,count是需求的字符个数,start是最小覆盖串开始的indexint l = 0, r = 0, size = Integer.MAX_VALUE, count = t.length(), start = 0;//如果need[]<0,说明无关紧要。如果need[]==0,则说明是被需要关注的元素while(r<s.length()){//对右边进行处理char c=s.charAt(r);if(need[c]>0){ //是值得关注的元素count--;}need[c]--;if(count==0){ //已经塞满了关注的元素了//移动左指针while(l<r && need[s.charAt(l)]<0){need[s.charAt(l)]++;l++;}//此时need==0,说明又满足了,开始统计if(r-l+1<size){size = r-l+1;start = l;}//又要向右移动,但是肯定和最小值无关了,开始破戒need[s.charAt(l)]++;l++;count++;}r++;}return size == Integer.MAX_VALUE ? "" : s.substring(start, start + size);}
}
36.有效的数独
一定是9×9的,
在第 i 个行中是否出现过
在第 j 个列中是否出现过
在第 j/3 + (i/3)*3个box中是否出现过.
定义三个数组,int [][]row, row[i][j], i代表第几行,j代表出现的元素。
box[i][j], i代表第几个小格子,j代表出现的元素。
class Solution {public boolean isValidSudoku(char[][] board) {int[][] row = new int[9][10];int[][] col = new int[9][10];int[][] box = new int[9][10];for (int i = 0; i < 9; i++) {for (int j = 0; j < 9; j++) {if (board[i][j] == '.') {continue;}int curNum = board[i][j] - '0';if (row[i][curNum] == 1) {return false;}if (col[j][curNum] == 1) {return false;}if (box[j / 3 + (i / 3) * 3][curNum] == 1) {return false;}row[i][curNum] = 1;col[j][curNum] = 1;box[j / 3 + (i / 3) * 3][curNum] = 1;}}return true;}
}
54.螺旋矩阵
class Solution {public List<Integer> spiralOrder(int[][] matrix) {int m = matrix.length;int n = matrix[0].length;int up = 0;int down = m - 1;int left = 0;int right = n - 1;List<Integer> ans = new ArrayList<>();while (true) {for (int i = left; i <= right; i++)ans.add(matrix[up][i]);if (++up > down)break;for (int i = up; i <= down; i++)ans.add(matrix[i][right]);if (--right < left)break;for (int i = right; i >= left; i--)ans.add(matrix[down][i]);if (--down < up)break;for (int i = down; i >= up; i--)ans.add(matrix[i][left]);if (++left > right)break;}return ans;}
}
48.旋转图像
关键是找到公式: matrix[i][j] -> matrix[j][n-1-i]
class Solution {public void rotate(int[][] matrix) {int n = matrix.length;for (int i = 0; i < n / 2; i++) {for (int j = 0; j < (n + 1) / 2; j++) {int tmp = matrix[i][j];matrix[i][j] = matrix[n - 1 - j][i];matrix[n - 1 - j][i] = matrix[n - 1 - i][n - 1 - j];matrix[n - 1 - i][n - 1 - j] = matrix[j][n - 1 - i];matrix[j][n - 1 - i] = tmp;}}}
}
73.矩阵置零
给定一个 m x n 的矩阵,如果一个元素为 0 ,则将其所在行和列的所有元素都设为 0 。请使用 原地 算法。
还是用两个辅助数组,一个判断某一行,一个判断某一列。舒服。
289.生命游戏
暴力模拟就行。dxdy代表你移动的旁边的8个位置。
int[] dx = { -1, 1, 0, 0, -1, -1, 1, 1 };
int[] dy = { 0, 0, -1, 1, -1, 1, -1, 1 };for (int k = 0; k < 8; k++) {int nx = x + dx[k];int ny = y + dy[k];if (nx < 0 || nx >= m || ny < 0 || ny >= n)continue;// 如果这个位置为 0,代表当前轮是死的,不需要算进去// 如果这个位置为 1,代表当前轮是活得,需要算进去// 如果这个位置为 2,代表当前轮是死的(状态10,下一轮是活的),不需要算进去// 如果这个位置为 3,代表是当前轮是活的(状态11,下一轮也是活的),需要算进去cnt += (board[nx][ny] & 1);
}
383.赎金信
int[] a = new int[26]; 用这个做个小哈希表就行
205.同构字符串
每个出现的字符都应当映射到另一个字符,同时不改变字符的顺序。不同字符不能映射到同一个字符上。
要满足最后一条,应该建立两个hash表, m ->t ; t->m
如果只建立一个hash表, abb -> rrr 会成立,但是其应该不成立。
290.单词规律
也是典型的双射问题,两个哈希表就解决了。
如果两个字符串完全不一样,也可以只用一个哈希表。利用hashmap的put操作的返回值的特性。
如果key不存在,插入成功,返回null;如果key存在,返回之前对应的value。
先将字符串截断 String[] words = str.split(" ");
然后用map.put(words[i],i) 与 map.put(s[i],i)判等就行
242.有效的字母异位词
也是hashmp,第一个字符串遍历去++,第二个字符串遍历去–。
最后判断是否所有元素均为0即可。