真实面试经历(4)

问题1:关于SVN

SVN是Subversion的简称,是一个开放源代码的版本控制系统,相较于RCS、CVS,它采用了分支管理系统。

svn trunk和branches什么关系:

svn://proj/|+-trunk+-branches+-tags
这是一个标准的布局,trunk为主开发目录,branches为分支开发目录,tags为tag存档目录(不允许修改)。

SVN可以为一个版本库中的内容(主干)建立一个分支.分支和主干完全独立,就相当于把代码再复制一份,重新添加到版本库中。

如果要将主干内容合并到分支上,我们需要在分支的工作副本上操作。
如果要将分支的改变合并到主干上,我们需要在主干的工作副本上操作。

问题2:Redis

Redis本质上是一个Key-Value类型的内存数据库,因为是纯内存操作,Redis的性能非常出色。

Redis默认端口号:6379。

Redis支持的数据类型:

string,list,set,sorted set,hash。

支持事务,操作都是原子性,所谓的原子性就是对数据的更改要么全部执行,要么全部不执行

丰富的特性:可用于缓存,消息,按key设置过期时间,过期后将会自动删除

Redis的最大value可达1GB。

Redis持久化:两种方式AOF和RDB(默认RDB)。

问题3:对MAP的理解

map的底层实现:数组加链表。

数组到达一定长度之后就会扩容(16*0.75)。(0.75是负载因子。)两倍扩容。

链表到达一定长度之后(8),会变成红黑二叉树。

put方法原理:计算key的Hash值,key的Hash值和01111与运算得出1-15的数字,就是该node在数组中的位置,

如果该位置上存在元素,就创建链表,往下顺延。

当数组扩容之后,元素要迁移,避免17-32的位置空置。

迁移重新计算Hash值,如果Hash值的二进制从右数第五位为0,则不变。如为1,则在数组的位置+16。

Map详解链接:Map面试。

HashMap详解链接:HashMap面试。

问题4:try catch finally return执行顺序

1、不管有没有出现异常,finally块中代码都会执行;
2、当try和catch中有return时,finally仍然会执行;
3、finally是在return后面的表达式运算后执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,不管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的;
4、finally中最好不要包含return,否则程序会提前退出,返回值不是try或catch中保存的返回值。

详情链接:有return的情况下try catch finally的执行顺序

问题5:String

String, StringBuffer,StringBuilder的区别?

String是不可变类,每当我们对String进行操作的时候,总是会创建新的字符串。操作String很耗资源,所以Java提供了两个工具类来操作String - StringBuffer和StringBuilder。

StringBuffer和StringBuilder是可变类,StringBuffer是线程安全的,StringBuilder则不是线程安全的。所以在多线程对同一个字符串操作的时候,我们应该选择用StringBuffer。由于不需要处理多线程的情况,StringBuilder的效率比StringBuffer高。

String常用方法

length():求字符串的长度

indexOf():求某个字符在字符串中的位置

charAt():求一个字符串中某个位置的值

equals():比较两个字符串是否相同

replace():将字符串中的某些字符用别的字符替换掉。形如replace(“abc”,”ddd”);字符串中的abc将会被ddd替换掉。

split():根据给定正则表达式的匹配拆分此字符串。形如 String s = "The time is going quickly!"; str1=s.split(" ");

substring():输出一个新的字符串,它是此字符串中的子串,形如substring(3,7);它将字符串中的第四个第五个第六个输出。

trim():将字符串开头的空白(空格)和尾部的空白去掉。

format():使用指定的语言环境、格式字符串和参数返回一个格式化字符串。

toLowerCase():将字符串中所有的大写改变成小写

toUpperCase():将字符串中所有的小写改变为大写

@侵删。

问题6:jvm相关

(1)如果对象的引用被置为null,垃圾收集器是否会立即释放对象占用的内存?

不会,在下一个垃圾回收周期中,这个对象将是可被回收的。

也就是说当一个对象的引用变为 null 时,并不会被垃圾收集器立刻回收,而是在下一次垃圾回收时才会释放其占用的内存。

(2)什么是类加载器的双亲委派模型?

工作过程:如果一个类加载器收到了类加载的请求,它首先不会自己去加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求都应该传送到顶层的启动类加载器,只有当父加载器无法完成这个加载请求时,子加载器才会尝试自己去加载。

为什么要有双亲委派模型呢?原因是双亲委派模型可以保证 Java 程序的稳定性。比如你有一个类,在不采用双亲委派模型的情况下,可能会有不同的加载器去加载这个类,不同类加载器加载出来的 Class 文件必然不相同,这样就造成了不一致性。

(3)Java类加载器包括⼏种?它们之间的⽗⼦关系是怎么样的?

启动Bootstrap类加载、扩展Extension类加载、系统System类加载。

父子关系如下:

  • 启动类加载器 ,由C++ 实现,没有父类;
  • 扩展类加载器,由Java语言实现,父类加载器为null;
  • 系统类加载器,由Java语言实现,父类加载器为扩展类加载器;
  • 自定义类加载器,父类加载器肯定为AppClassLoader。

(4)JVM内存分配。

根据JVM规范,JVM内存共分为虚拟机栈,堆,方法区,程序计数器,本地方法栈五部分。

@侵删。

  • Java堆(Heap),是Java虚拟机所管理的内存中最大的一块。Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。
  • 方法区(Method Area),方法区(Method Area)与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
  • 程序计数器(Program Counter Register),程序计数器(Program Counter Register)是一块较小的内存空间,它的作用可以看做是当前线程所执行的字节码的行号指示器。
  • JVM栈(JVM Stacks),与程序计数器一样,Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法被执行的时候都会同时创建一个栈帧(Stack Frame)用于存储局部变量表、操作栈、动态链接、方法出口等信息。每一个方法被调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。
  • 本地方法栈(Native Method Stacks),本地方法栈(Native Method Stacks)与虚拟机栈所发挥的作用是非常相似的,其区别不过是虚拟机栈为虚拟机执行Java方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的Native方法服务。

(5)GC算法

GC最基础的算法有三种:标记 -清除算法、复制算法、标记-压缩算法,我们常用的垃圾回收器一般都采用分代收集算法。

  • 标记 -清除算法,“标记-清除”(Mark-Sweep)算法,如它的名字一样,算法分为“标记”和“清除”两个阶段:首先标记出所有需要回收的对象,在标记完成后统一回收掉所有被标记的对象。
  • 复制算法,“复制”(Copying)的收集算法,它将可用内存按容量划分为大小相等的两块,每次只使用其中的一块。当这一块的内存用完了,就将还存活着的对象复制到另外一块上面,然后再把已使用过的内存空间一次清理掉。
  • 标记-压缩算法,标记过程仍然与“标记-清除”算法一样,但后续步骤不是直接对可回收对象进行清理,而是让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存
  • 分代收集算法,“分代收集”(Generational Collection)算法,把Java堆分为新生代和老年代,这样就可以根据各个年代的特点采用最适当的收集算法。(java8取消了永久代,采用了Metaspace)

猜你喜欢

转载自blog.csdn.net/zaimeiyeshicengjing/article/details/81587746