Java面试练习题刷题记录
目录
我几乎每天都会刷题训练来使自己对各种算法随时保持一个清晰的状态。要知道眼过千遍不如手过一遍,想成为一名合格的开发工程师,更要逼迫自己养成动手的好习惯。
我们都知道,算法的训练对程序员来说及其重要,语言和开发平台不断变化,但是万变不离其宗的是那些算法和理论,刷算法最最最直白的原因就是找一个好的工作,那刷题一定是必不可少的。
现在算法刷题平台还是蛮多的,给大家介绍一个我认为与大厂关联最深的平台——牛客网
相较于其他平台,他们的题单更和工作,大厂靠拢,不光有面试必刷的101到题目,还有大量大厂真题,内容也全程免费,相较于其它会员费结算的来说 非常的友好。
牛客网还支持ACM模式,没有练习过的一定要提前适应!像某团、某为,都要求自己处理输入输出,如果不提前练习会很吃亏的!
牛客的题解更新迭代也很快,讨论区也有技巧的分享,能帮你把所有盲点扫清楚,整体来说还是非常推荐去练习的~
二阶魔方又叫小魔方,是2*2*2的立方形结构。每一面都有4个块,共有24个块。每次操作可以将任意一面逆时针或者顺时针旋转90°,如将上面逆时针旋转90°操作如下。
Nero在小魔方上做了一些改动,用数字替换每个块上面的颜色,称之为数字魔方。魔方上每一面的优美度就是这个面上4个数字的乘积,而魔方的总优美度就是6个面优美度总和。
现在Nero有一个数字魔方,他想知道这个魔方在操作不超过5次的前提下能达到的最大优美度是多少。
魔方展开后每一块的序号如下图:
输入一行包含24个数字,按序号顺序给出魔方每一块上面的数字。所有数大小范围为[-100,100]。
输出一行包含一个数字,表示最大优美度。
输入:
2 -3 -2 3 7 -6 -6 -7 9 -5 -9 -3 -2 1 4 -9 -1 -10 -5 -5 -10 -4 8 2输出:
8281
题解:
- import java.util.*;
-
- public class Main {
- /**
- * 记录三种旋转需要变动的下标,共三组下标
- * 每组下标里,每4个下标形成一个轮回
- */
- private static int[][] map = new int[][]{
- {0,1,3,2,7,5,22,9,6,4,23,8},
- {6,7,13,12,2,8,17,11,3,14,16,5},
- {4,5,11,10,0,6,16,20,2,12,18,22}
- };
-
- /**
- * main
- * @param args 命令行参数
- */
- public static void main(String[] args) {
- // 读取输入
- Scanner sc = new Scanner(System.in);
- int[] n = new int[24];
- for (int i = 0; i < 24; i++) {
- n[i] = sc.nextInt();
- }
-
- // 输出结果
- System.out.println(resolve(n, -1, 5));
- }
-
- /**
- * 分析当前状态
- * @param n 当前状态
- * @param type 达到当前状态时的最后一个操作类型
- * @param remain 剩余操作数
- * @return 可以达到的最大优美度
- */
- private static int resolve(int[] n, int type, int remain) {
- // 非法状态
- if (remain < 0) return Integer.MIN_VALUE;
- // 当前状态优美度
- int max = compute(n);
- // 尝试转动,不可与type类型相同
- if (type != 0) max = Math.max(max, turn(n, 0, remain));
- if (type != 1) max = Math.max(max, turn(n, 1, remain));
- if (type != 2) max = Math.max(max, turn(n, 2, remain));
- return max;
- }
-
- /**
- * 按照类型转动魔方多次并分析
- * @param n 当前状态
- * @param type 转动类型
- * @param remain 剩余操作数
- * @return 转动可以达到的最大优美度
- */
- private static int turn(int[] n, int type, int remain) {
- // 转动一次
- turn(n, type);
- int max = resolve(n, type, remain-1);
- // 转动两次
- turn(n, type);
- max = Math.max(max, resolve(n, type, remain-2));
- // 转动三次,相当于逆向转动一次
- turn(n, type);
- max = Math.max(max, resolve(n, type, remain-1));
- // 转动四次,还原初始状态
- turn(n, type);
- return max;
- }
-
- /**
- * 转动操作
- * @param n 当前状态
- * @param type 转动类型
- */
- private static void turn(int[] n, int type) {
- // 每四个下标一轮回
- for (int i = 0, tmp = 0; i < 12; i++) {
- if (i % 4 == 0) tmp = n[map[type][i]];
- if (i % 4 == 3) n[map[type][i]] = tmp;
- else n[map[type][i]] = n[map[type][i+1]];
- }
- }
-
- /**
- * 计算当前状态的优美度
- * @param n 当前状态
- * @return 优美度
- */
- private static int compute(int[] n) {
- return n[0]*n[1]*n[2]*n[3] +
- n[6]*n[7]*n[12]*n[13] +
- n[4]*n[5]*n[10]*n[11] +
- n[8]*n[9]*n[14]*n[15] +
- n[16]*n[17]*n[18]*n[19] +
- n[20]*n[21]*n[22]*n[23];
- }
- }
有一个推箱子的游戏, 一开始的情况如下图:
上图中, '.' 表示可到达的位置, '#' 表示不可到达的位置,其中 S 表示你起始的位置, 0表示初始箱子的位置, E表示预期箱子的位置,你可以走到箱子的上下左右任意一侧, 将箱子向另一侧推动。如下图将箱子向右推动一格;
..S0.. -> ...S0.
注意不能将箱子推动到'#'上, 也不能将箱子推出边界;
现在, 给你游戏的初始样子, 你需要输出最少几步能够完成游戏, 如果不能完成, 则输出-1。
第一行为2个数字,n, m, 表示游戏盘面大小有n 行m 列(5< n, m < 50);
后面为n行字符串,每行字符串有m字符, 表示游戏盘面;
一个数字,表示最少几步能完成游戏,如果不能,输出-1;
输入:
3 6 .S#..E .#.0.. ......输出:
11
题解:
- import java.util.LinkedList;
- import java.util.Queue;
- import java.util.Scanner;
- public class Main {
-
- public static void main(String[] args){
- Scanner sc=new Scanner(System.in);
- int n=sc.nextInt();
- int m=sc.nextInt();
- char[][] chas=new char[n][m];
- // 人的起始位置S 以及箱子的位置
- int startX=0,startY=0,boxX=0,boxY=0;
- // 判断位置 ,即人要先找到箱子,再推着箱子往前走
- for(int i=0;i<n;i++){
- String string=sc.next();
- for(int j=0;j<m;j++){
- chas[i][j]=string.charAt(j);
- if(chas[i][j]=='S'){
- startX=i;
- startY=j;
- }
- if(chas[i][j]=='0'){
- boxX=i;
- boxY=j;
- }
- }
- }
- System.out.println(solve(chas,startX,startY,boxX,boxY));
- }
-
- public static class Node{
- int px; // 人的位置
- int py;
- int bx; //箱子的位置
- int by;
- int step; //从初始位置到现在节点所走的步数
- public Node(int px,int py,int bx,int by){
- this.px=px;
- this.py=py;
- this.bx=bx;
- this.by=by;
- }
- }
-
-
- private static int solve(char[][] chas,int startX, int startY,int boxX,int boxY) {
- Node start=new Node(startX, startY,boxX,boxY);
- int n=chas.length;
- int m=chas[0].length;
- // iswalked四维数组,可以用来存储走过的路径以防重复。
- int[][][][] iswalked=new int[n][m][n][m];
- // 每个节点都有dir4个方向的移动,分别是 下右上左
- int[][] movedir=new int[][]{{1,0},{0,1},{-1,0},{0,-1}};
-
- Queue<Node> que=new LinkedList<>(); //利用队列实现BFS
- start.step=0;
- que.add(start);
-
- //开始BFS广度搜索最短路径
- while(!que.isEmpty()){
- Node cur=que.poll();
- int newBx=cur.bx;
- int newBy=cur.by;
- for(int i=0;i<4;i++){ // 向4个方向走
-
- //人在箱子左边或右边
- if(cur.px==cur.bx){
-
- if (cur.py+movedir[i][1]==cur.by){
- newBy = cur.by+movedir[i][1];
- }else{
- newBy = cur.by;
- }
-
- }
-
- //人在箱子上面或下面
- if(cur.py==cur.by){
- if(cur.px+movedir[i][0]==cur.bx){
- newBx = cur.bx+movedir[i][0];
- }else{
- newBx = cur.bx;
- }
- }
-
-
- // 箱子找到了要随人往4个方向动;没找到则箱子不动人动
- Node next=new Node(cur.px+movedir[i][0], cur.py+movedir[i][1],newBx,newBy);
- //不能让人在没找到箱子之前出地图或者自己提前到目的地
- if(next.px<0||next.px>=n||next.py<0||next.py>=m||chas[next.px][next.py]=='#'
- ||next.bx<0||next.bx>=n||next.by<0||next.by>=m
- ||chas[next.bx][next.by]=='#'){
- continue;
- }
-
- // 0说明这条路径没有走过
- if(iswalked[next.px][next.py][next.bx][next.by]==0){
- iswalked[next.px][next.py][next.bx][next.by]=1;
- next.step=cur.step+1;
- if(chas[next.bx][next.by]=='E'){ //此时把箱子推到终点了
- return next.step; // 最先到终点的就是最短路径
- }
- que.add(next);
- }
- }
- }
- return -1;
- }
-
-
- }
有n个房间,现在i号房间里的人需要被重新分配,分配的规则是这样的:先让i号房间里的人全都出来,接下来按照 i+1, i+2, i+3, ... 的顺序依此往这些房间里放一个人,n号房间的的下一个房间是1号房间,直到所有的人都被重新分配。
现在告诉你分配完后每个房间的人数以及最后一个人被分配的房间号x,你需要求出分配前每个房间的人数。数据保证一定有解,若有多解输出任意一个解。
第一行两个整数n, x (2<=n<=10^5, 1<=x<=n),代表房间房间数量以及最后一个人被分配的房间号;
第二行n个整数 a_i(0<=a_i<=10^9) ,代表每个房间分配后的人数。
输出n个整数,代表每个房间分配前的人数。
输入:
3 1 6 5 1输出:
4 4 4
题解:
- import java.util.Scanner;
-
- public class Main {
- public static void main(String[] args) {
- Scanner scanner = new Scanner(System.in);
- int n = scanner.nextInt();
- int x = scanner.nextInt() - 1;
- long[] room = new long[n];
- long min = Long.MAX_VALUE;
- for (int i = 0; i < n; i++) {
- room[i] = scanner.nextInt();
- if (room[i] < min)
- min = room[i];
- }
- //get min_index
- int minIndex = x;
- while (room[minIndex] != min) {
- minIndex = minIndex > 0 ? minIndex - 1 : n - 1;
- }
- // remove the round number.
- for (int i = 0; i < n; i++) {
- room[i] -= min;
- }
- // remove the tail
- int remain = 0;
- for (int i = x; i != minIndex; i = i > 0 ? i - 1 : n - 1) {
- room[i] -= 1;
- remain += 1;
- }
- room[minIndex] += remain + n * min;
- //print the result
- for (int i = 0; i < n; i++) {
- System.out.print(room[i] + " ");
- }
- }
- }
【编码题】字符串S由小写字母构成,长度为n。定义一种操作,每次都可以挑选字符串中任意的两个相邻字母进行交换。询问在至多交换m次之后,字符串中最多有多少个连续的位置上的字母相同?
第一行为一个字符串S与一个非负整数m。(1 <= |S| <= 1000, 1 <= m <= 1000000)
一个非负整数,表示操作之后,连续最长的相同字母数量。
输入:
abcbaa 2输出:
2说明:
使2个字母a连续出现,至少需要3次操作。即把第1个位置上的a移动到第4个位置。 所以在至多操作2次的情况下,最多只能使2个b或2个a连续出现。
题解:
- import java.util.*;
-
- public class Main{
- static int res = 1;
- public static void main(String[] args){
-
- Scanner scanner = new Scanner(System.in);
- String input = scanner.nextLine();
- String str = input.split(" ")[0];
- int m = Integer.parseInt(input.split(" ")[1]);
-
- HashMap
> map = new HashMap<>(); - char[] chars = str.toCharArray();
- int len = chars.length;
-
- for(int i = 0; i < len; i++){
- List
list = map.getOrDefault(chars[i], new ArrayList<>()); - list.add(i);
- map.put(chars[i], list);
- }
-
- for(int i = 0; i < len; i++){
- char c = chars[i];
- List
list = map.get(c); - moveTogether(list, i, m);
- /*
- int lessCount = 0;
- for(int num : list){
- if(num < i){
- lessCount++;
- }
- else{
- break;
- }
- }
- int greatCount = list.size()-1-lessCount;
- for(int j = 0; j < lessCount; i++){
- int cur = list.get(j);
- // 现在这个节点的位置在cur,要移动到 (0 1 j 3 4 ... lessCount-1)i
- // lesscount-2的时候对应i-2
- // lesscount-1的时候对应i-1
- int target = i + j - lessCount;
- int times =
- }
- }
- */
-
-
-
- }
- System.out.println(res);
-
-
-
-
- }
- // 在times步内,尽可能把list里面的数字移动到mid周围
- public static void moveTogether(List
list, int mid, int times) { - int len = list.size();
- int mid_index = 0;
- for(int i = 0; i < len; i++){
- if(list.get(i) == mid){
- mid_index = i;
- break;
- }
- }
- int left_index = mid_index-1;
- int right_index = mid_index+1;
- int count = 1;
- // mid为中心的左右边界
- int left_mid = mid-1;
- int right_mid = mid+1;
- while(times > 0){
- if(left_index < 0 && right_index >= len){
- break;
- }
-
- // 看一下左边和右边哪个更近一点
- int left_dist;
- if(left_index < 0)
- left_dist = Integer.MAX_VALUE / 2;
- else
- left_dist = left_mid - list.get(left_index);
- int right_dist;
- if(right_index >= len) {
- right_dist = Integer.MAX_VALUE / 2;
- }
- else {
- right_dist = list.get(right_index) - right_mid;
- }
-
- if(left_dist > right_dist){
- // 右边靠更加近一点
- if(times < right_dist) break;
- times -= right_dist;
- right_mid++;
- count++;
- right_index++;
- }
- else{
- if(times < left_dist) break;
- times -= left_dist;
- left_mid--;
- left_index--;
- count++;
- }
- }
- res = Math.max(count, res);
- }
-
- }
点击链接 进行跳转注册,开始你的保姆级刷题之路吧!
另外这里不仅仅可以刷题,你想要的这里都会有,十分适合小白和初学者入门学习~
1、算法篇(398题):面试必刷100题、算法入门、面试高频榜单
2、数据结构篇(300题):都是非常经典的链表、树、堆、栈、队列、动态规划等
3、语言篇(500题):C/C++、java、python入门算法练习
4、SQL篇(82题):快速入门、SQL必知必会、SQL进阶挑战、面试真题
5、大厂笔试真题:字节跳动、美团、百度、腾讯…掌握经验不在惧怕面试!