如果当时这 16 道题能答好,现在应该已经被录取了(记一次面试的亲身经历 2020-9-9
(2)悲观读锁 readLock:
readLock 是一个共享读锁,在没有线程获取写锁情况下,多个线程可以获取该锁。如果有写锁获取,那么其他线程请求读锁会被阻塞。悲观读锁会认为其他线程可能要对自己操作的数据进行修改,所以需要先对数据进行加锁,这是在读少写多的情况下考虑的。请求该锁成功后会返回一个 stamp 值,在释放锁时调用 unlockRead 方法传递 stamp 参数。提供了非阻塞式获取锁方法 tryWriteLock。
(3)乐观读锁 tryOptimisticRead:
tryOptimisticRead 相对比悲观读锁,在操作数据前并没有通过 CAS 设置锁的状态,如果没有线程获取写锁,则返回一个非 0 的 stamp 变量,获取该 stamp 后在操作数据前还需要调用 validate 方法来判断期间是否有线程获取了写锁,如果是返回值为 0 则有线程获取写锁,如果不是 0 则可以使用 stamp 变量的锁来操作数据。由于 tryOptimisticRead 并没有修改锁状态,所以不需要释放锁。这是读多写少的情况下考虑的,不涉及 CAS 操作,所以效率较高,在保证数据一致性上需要复制一份要操作的变量到方法栈中,并且在操作数据时可能其他写线程已经修改了数据,而我们操作的是方法栈里面的数据,也就是一个快照,所以最多返回的不是最新的数据,但是一致性得到了保证。
五、说一下 session 的工作原理?
当客户端登录完成后,会在服务端产生一个 session,此时服务端会将 sessionid 返回给客户端浏览器。客户端将 sessionid 储存在浏览器的 cookie 中,当用户再次登录时,会获得对应的 sessionid,然后将 sessionid 发送到服务端请求登录,服务端在内存中找到对应的 sessionid,完成登录,如果找不到,返回登录页面。
六、说一下 tcp 粘包是怎么产生的?
发送方需要等缓冲区满才能发送出去,造成粘包;
接收方不及时接收缓冲区的包,造成粘包;
七、举一个用 Java 实现的装饰模式(decorator design pattern)?它是作用于对象层次还是类层次?
在 Java IO 中运用了装饰器模式,inputStream 作为抽象类,其下有几个实现类,表示从不同的数据源输入:
byteArrayInputStream
fileInputStream
StringBufferInputStream
PipedInputStream,从管道产生输入;
SequenceInputStream,可将其他流收集合并到一个流内;
FilterInputStream 作为装饰器在 JDK 中是一个普通类,其下面有多个具体装饰器比如 BufferedInputStream、DataInputStream 等。
FilterInputStream 内部封装了基础构件:
protected volatile InputStream in;
而 BufferedInputStream 在调用其 read()读取数据时会委托基础构件来进行更底层的操作,而它自己所起的装饰作用就是缓冲,在源码中可以很清楚的看到这一切。
八、请举例说明如何在 Spring 中注入一个 Java Collection?
Spring 注入有四种方式,
set 注入;
构造器注入;
基于注解的注入;
xml 配置文件注入;
想要注入 java collection,就是注入集合类:
list
set
map
props:该标签支持注入键和值都是字符串类型的键值对。
list 和 set 都使用 value 标签;map 使用 entry 标签;props 使用 prop 标签;
九、什么是 Swagger?你用 Spring Boot 实现了它吗?
Swagger 是用于生成 RestFul Web 服务的可视化表示工具,它使文档和服务器可视化更新;
当定义好 Swagger 后,可以调用服务端接口,来查看接口的返回值,验证返回数据的正确性;
十、什么是 Spring Profiles?
Spring Profiles 允许用户根据配置文件(dev、test、prod)来判定加载哪些配置文件,完成注册 bean;
十一、在 hibernate 中使用 Integer 和 int 做映射有什么区别?
hibernate 是面向对象的 ORM,所以一般定义成封装类型,要看数据库中的定义,如果数据库中有对应字段存在 null 值,就要定义 Integer。也可以定义基本类型,在配置文件中写清楚即可。
十二、mysql 的内连接、左连接、右连接有什么区别?
内连接,显示两个表中有联系的所有数据;
左链接,以左表为参照,显示所有数据,右表中没有则以 null 显示
右链接,以右表为参照显示数据,,左表中没有则以 null 显示
十三、Redis 支持的数据类型有哪些?
String、hash、list、set、zset(sorted set:有序集合)
十四、详细介绍一下 CMS 垃圾回收器?
CMS 垃圾回收器是 Concurrent Mark Sweep,是一种同步的标记-清除,CMS 分为四个阶段:
初始标记,标记一下 GC Root 能直接关联到的对象,会触发“Stop The World”;
并发标记,通过 GC Roots Tracing 判断对象是否在使用中;
重新标记,标记期间产生对象的再次判断,执行时间较短,会触发“Stop The World”;
并发清除,清除对象,可以和用户线程并发进行;
十五、新生代垃圾回收器和老生代垃圾回收器都有哪些?有什么区别?
新生代回收器:Serial、ParNew、Parallel Scavenge
老年代回收器:Serial Old、Parallel Old、CMS
新生代回收器一般采用的是复制算法,复制算法效率较高,但是浪费内存;
老生代回收器一般采用标记清楚算法,比如最常用的 CMS;
十六、简述分代垃圾回收器是怎么工作的?
分代回收器分为新生代和老年代,新生代大概占 1/3,老年代大概占 2/3;
新生代包括 Eden、From Survivor、To?Survivor;Eden 区和两个 survivor 区的 的空间比例 为 8:1:1 ;
垃圾回收器的执行流程:
把 Eden + From Survivor 存活的对象放入 To Survivor 区;
清空?Eden + From Survivor 分区,From Survivor 和 To Survivor 分区交换;
每次交换后存活的对象年龄+1,到达 15,升级为老年代,大对象会直接进入老年代;
老年代中当空间到达一定占比,会触发全局回收,老年代一般采取标记-清除算法;
评论