如果当时这16道题能答好,现在应该已经被录取了(记一次面试的亲身经历 2020-9-9)
前情提要
9月9日参加面试遇到的问题,10天时间,历经14次编辑,今天(9月19日)才整理出来,知识的匮乏,不言而喻。
今天无意之间看到了这个,果断申请了,希望CSDN能给我一次机会,让我更多的参与进来,提高自己,成为博客专家,成为高薪全栈工程师,公司顶梁柱。
之前的CSDN首页无外乎三点
- 标题党,所谓的面试题,今天阿里,明天字节跳动的,也不知道你到底去哪里了。
- Python,也不知道有多少干Python的,java都是hello world,你们呢?直接爬美女图片,吸引眼球?目的达到了,你赢了。
- 一些大佬的博客,这才是重点,CSDN所需要的,但是感觉有一个弊端,也许同样的博客,大佬的就访问量破万,小菜的基本全都石沉大海。
之前我也有一段时间,做起了标题党,效果不是很好;
我也插入一些美女图片,你还真别说,效果还是有的;
但,这不应该是一个高手所为,我也反思了自己,觉得还是应该坚持本心,写博客是为了学习,不是为了哗众取宠。
《如果当时》系列博客,是基于自己真实的面试经历,全部都是面试真题,血和泪的教训,分享给大家,希望能给那些和我有同样困扰的人,提供一些帮助,努力一定就会有收获,加油。
一、ArrayList 和 Vector 的区别是什么?
- Vector 使用了 Synchronized 来实现线程同步,是线程安全的,而 ArrayList 是非线程安全的。
- ArrayList 和 Vector 都会根据实际的需要动态的调整容量,只不过在 Vector 扩容每次会增加 1 倍,而 ArrayList 只会增加 50%。
- 在不考虑并发问题的时候还是推荐使用Arraylist。
二、Array 和 ArrayList 有何区别?
- Array可以包含基本类型和对象类型,ArrayList只能包含对象类型。
- Array大小是固定的,ArrayList的大小是动态变化的。
- ArrayList提供了更多的方法和特性,比如:addAll(),removeAll(),iterator()等等。
- 对于基本类型数据,ArrayList 使用自动装箱来减少编码工作量;而当处理固定大小的基本数据类型的时候,这种方式相对比较慢,这时候应该使用Array。
三、除了 ReetrantLock,你还接触过 JUC 中的哪些并发工具?
我的答案:
ReetrantWriteReadLock:读写锁,包含读取和写入两种类型的锁,当进行读取操作时允许多线程访问,当进行写操作时,只允许一个线程访问;
ConcurrentHashMap:HashMap的同步版本,HashTable的升级版本,HashTable会锁住整个集合,而ConcurrentHashMap只是锁住当前访问的键值对;
Vector:Vector中加入了Synchronized锁,是线程安全的;相对于Arraylist,因为加锁的原因Vector效率要低一点;vector是1倍扩容,ArrayList是50%扩容;
Volatile:只能修饰变量,不能修饰代码块和方法,保证了可见性和有序性,不能保证原子性;
百度答案:
juc下常用的五个高并发工具:
- CountDownLatch:同步计数器
- CyclicBarrier: 线程屏障的功能
- Exchanger:用来使两个线程交换数据。
- Semaphore:控制信号量的个数,构造时传入个数。总数就是控制并发的数量。
- Future:接口,FutureTask是它的实现类,配合线程池来一起工作,将任务交给线程池去处理。
百度的和我想的完全不是一回事,有时间再看看JUC吧!
四、请谈谈 ReadWriteLock 和 StampedLock。
1、ReadWriteLock
ReadWriteLock 可以实现多个读锁同时进行,但是读与写和写于写互斥,只能有一个写锁线程在进行。
2、StampedLock
StampedLock是Jdk在1.8提供的一种读写锁,相比较ReentrantReadWriteLock性能更好,因为ReentrantReadWriteLock在读写之间是互斥的,使用的是一种悲观策略,在读线程特别多的情况下,会造成写线程处于饥饿状态,虽然可以在初始化的时候设置为true指定为公平,但是吞吐量又下去了,而StampedLock是提供了一种乐观策略,更好的实现读写分离,并且吞吐量不会下降。
StampedLock包括三种锁:
(1)写锁writeLock:
writeLock是一个独占锁写锁,当一个线程获得该锁后,其他请求读锁或者写锁的线程阻塞, 获取成功后,会返回一个stamp(凭据)变量来表示该锁的版本,在释放锁时调用unlockWrite方法传递stamp参数。提供了非阻塞式获取锁tryWriteLock。
(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,升级为老年代,大对象会直接进入老年代;
- 老年代中当空间到达一定占比,会触发全局回收,老年代一般采取标记-清除算法;
上一篇:如果当时这20道题能答好,现在应该已经被录取了(记一次面试的亲身经历 2020-8-27)
下一篇:Java面试题总结(绝对经典)
还没有评论,来说两句吧...