前言
作者:晓宜
个人简介:互联网大厂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