哔哩哔哩秋招Java二面

前言

作者:晓宜

个人简介:互联网大厂Java准入职,阿里云专家博主,csdn后端优质创作者,算法爱好者

一面过后面试官叫我别走,然后就直接二面,二面比较简短,记录一下,希望可以帮助到你

jvm的内存结构

1、方法区

方法区主要用于存储虚拟机加载的类信息、常量、静态变量,以及编译器编译后的代码等数据。

在jdk1.7及其之前,方法区是堆的一个"逻辑部分"(一片连续的堆空间),但为了与堆做区分,方法区还有个名字叫"非堆",也有人用"永久代"(HotSpot对方法区的实现方法)来表示方法区。

在jdk1.8中,方法区已经不存在,原方法区中存储的类信息、编译后的代码数据等已经移动到了元空间(MetaSpace)中,元空间并没有处于堆内存上,而是直接占用的本地内存(NativeMemory)

2、程序计数器(PC寄存器)

由于在JVM中,多线程是通过线程轮流切换来获得CPU执行时间的,因此,在任一具体时刻,一个CPU的内核只会执行一条线程中的指令,

因此,为了能够使得每个线程都在线程切换后能够恢复在切换到之前的程序执行位置,每个线程都需要有自己独立的程序计数器,并且不能互相被干扰,

3、java栈

Java栈中存放的是一个个的栈帧,每个栈帧对应一个被调用的方法,在栈帧中包括局部变量表(Local Variables)、操作数栈(Operand Stack)、

指向当前方法所属的类的运行时常量池的引用(Reference to runtime constant pool)、

方法返回地址(Return Address)和一些额外的附加信息。当线程执行一个方法时,就会随之创建一个对应的栈帧,并将建立的栈帧压栈。当方法执行完毕之后,便会将栈帧出栈。

4、堆内存

堆内存主要用于存放对象和数组,它是JVM管理的内存中最大的一块区域,堆内存和方法区都被所有线程共享,在虚拟机启动时创建。在垃圾收集的层面上来看,由于现在收集器基本上都采用分代收集算法,因此堆还可以分为新生代(YoungGeneration)和老年代(OldGeneration),新生代还可以分为 Eden、From Survivor、To Survivor。

5、本地方法栈

本地方法栈与虚拟机栈的区别是,虚拟机栈执行的是 Java 方法,本地方法栈执行的是本地方法(Native Method),其他基本上一致,在 HotSpot 中直接把本地方法栈和虚拟机栈合二为一,这里暂时不做过多叙述。

oom如何调试

可能原因

1.无法在 Java 堆中分配对象

2.应用程序保存了无法被GC回收的对象。

3.应用程序过度使用 finalizer。

注:finalizer方法约定对象在被回收前需要调用,会减慢对象回收的速度,可能造成oom。

解决方案

1.查找关键报错信息,如:java.lang.OutOfMemoryError: Java heap space

2.使用内存映像分析工具(如Eclipsc Memory Analyzer或者Jprofiler)对Dump出来的堆储存快照进行分析,分析清楚是内存泄漏还是内存溢出。

3.如果是内存泄漏,可进一步通过工具查看泄漏对象到GC Roots的引用链,修复应用程序中的内存泄漏。

4.如果不存在泄漏,先检查代码是否有死循环,递归等,再考虑用 -Xmx 增加堆大小。

static的作用

static表示"全局"或者"静态"的意思,用来修饰成员变量和成员方法,也可以形成静态static代码块,但是Java语言中没有全局变量的概念。

被static修饰的成员变量和成员方法独立于该类的任何对象。也就是说,它不依赖类特定的实例,被类的所有实例共享。

只要这个类被加载,Java虚拟机就能根据类名在运行时数据区的方法区内定找到他们。因此,static对象可以在它的任何对象创建之前访问,无需引用任何对象。

简而言之:static表示不要实例化就可以使用

算法题sql 表中某一用户最近几天最早登录的日期

假设有一张login表,表中有三个字段,id,date,user_id,则答案如下:

bash 复制代码
select
    min(date) date,
from
    login
group by user_id

算法题 链表中的一段进行反转

题目链接:反转列表2

bash 复制代码
# Definition for singly-linked list.
# class ListNode:
#     def __init__(self, val=0, next=None):
#         self.val = val
#         self.next = next
class Solution:
    def reverseBetween(self, head: Optional[ListNode], left: int, right: int) -> Optional[ListNode]:
        dummy = ListNode(next = head)
        left_node,right_node,pre = dummy,dummy,dummy

        for i in range(left-1):
            pre = pre.next
        left_node = pre.next
        for i in range(right):
            right_node = right_node.next
        
        curr = right_node.next

        pre.next = None
        right_node.next = None
        for i in range(right-left+1):
            next = left_node.next
            left_node.next = curr
            curr = left_node
            left_node = next
        pre.next = right_node

        return dummy.next
相关推荐
TheITSea几秒前
云服务器宝塔安装静态网页 WordPress、VuePress流程记录
java·服务器·数据库
AuroraI'ncoding7 分钟前
SpringMVC接收请求参数
java
数据小爬虫@12 分钟前
利用Python爬虫获取淘宝店铺详情
开发语言·爬虫·python
Clarify30 分钟前
docker部署go游戏服务器(进阶版)
后端
九圣残炎32 分钟前
【从零开始的LeetCode-算法】3354. 使数组元素等于零
java·算法·leetcode
IT书架1 小时前
golang面试题
开发语言·后端·golang
天天扭码1 小时前
五天SpringCloud计划——DAY1之mybatis-plus的使用
java·spring cloud·mybatis
程序猿小柒1 小时前
leetcode hot100【LeetCode 4.寻找两个正序数组的中位数】java实现
java·算法·leetcode
编程修仙1 小时前
Collections工具类
linux·windows·python
芝麻团坚果2 小时前
对subprocess启动的子进程使用VSCode python debugger
linux·ide·python·subprocess·vscode debugger