java中list的用法 一道java面试题,20亿数字的文本排序,如何取前100?
一道java面试题,20亿数字的文本排序,如何取前100?
因为这是一个Java问题,所以这是典型的TOPK问题。首先取前100个数字构建一个最小堆,然后依次从堆的顶部插入剩余的数字,同时调整堆。堆中最后100个元素就是结果。空间复杂度是k,时间复杂度是nlogk
我们知道,在编程中,如果你想让一个业务重复执行,一般有两种实现方法:递归和循环。在实际的编码过程中,我们不建议使用递归,而是建议使用循环。为什么?
事实上,不仅仅是Java,任何编程语言,如果递归写入错误,都可能导致内存溢出
!学习过Java的朋友一定或多或少听说过并理解了堆栈内存和堆内存。程序运行时,计算机操作系统会给每个进程分配堆内存和堆栈内存,分配的堆栈内存有一个上限。一旦超过上限,就会导致内存溢出。
为什么递归操作容易导致内存溢出?主要原因如下:
在递归方法中,如果终止递归的条件写得不正确,可能导致无限递归,最终导致内存溢出;
即使递归方法和退出递归条件正常,如果递归深度太深(递归次数太多),也会导致堆栈内存溢出!因为栈入栈出的规则是先入后出(先入后出),如果递归次数过多,就会导致只入不出栈,最后导致栈内存溢出。
将递归写入方式改为循环写入方式的优点是不会在短时间内出现只进不出栈的现象,避免了栈内存溢出的现象。
java递归改为循环后为什么不会导致栈内存溢出?
实际上,JVM堆栈是以“堆栈帧”为单位的顺序访问结构。在JVM中,方法调用将在JVM堆栈上分配(put)和取消(put)相应的堆栈帧。当方法离开(正常返回或异常)时,撤消堆栈帧(即堆栈外)。
关于java堆栈的问题,在编译的时候入栈的顺序是怎么样的?
方法区域(非堆):它是所有线程共享的内存区域,用于存储类信息、常量、静态变量、编译器编译的代码以及虚拟机加载的其他数据。Java堆:它是虚拟机管理的最大内存区域,也是所有线程共享的内存区域。它是在虚拟机启动时创建的。这个内存区域的唯一用途是存储对象实例,几乎所有的对象实例都分配内存。Java堆是垃圾收集器管理的主要领域。Java虚拟机栈:线程是私有的,其生命周期与线程相同。在执行每个方法时,将同时创建一个堆栈帧来存储局部变量表、操作数堆栈、动态链接和方法出口等信息。每个方法都会被调用,直到执行完成,这与虚拟机中从堆栈到堆栈输出进程的堆栈帧相对应。
java中list的用法 java list去重 java的堆和栈
版权声明:本文内容由互联网用户自发贡献,本站不承担相关法律责任.如有侵权/违法内容,本站将立刻删除。