如果当时这16道题能答好,现在应该已经被录取了(记一次面试的亲身经历 2020-9-9)

前情提要

9月9日参加面试遇到的问题,10天时间,历经14次编辑,今天(9月19日)才整理出来,知识的匮乏,不言而喻。

今天无意之间看到了这个,果断申请了,希望CSDN能给我一次机会,让我更多的参与进来,提高自己,成为博客专家,成为高薪全栈工程师,公司顶梁柱。

之前的CSDN首页无外乎三点

  1. 标题党,所谓的面试题,今天阿里,明天字节跳动的,也不知道你到底去哪里了。
  2. Python,也不知道有多少干Python的,java都是hello world,你们呢?直接爬美女图片,吸引眼球?目的达到了,你赢了。
  3. 一些大佬的博客,这才是重点,CSDN所需要的,但是感觉有一个弊端,也许同样的博客,大佬的就访问量破万,小菜的基本全都石沉大海。

之前我也有一段时间,做起了标题党,效果不是很好;

我也插入一些美女图片,你还真别说,效果还是有的;

但,这不应该是一个高手所为,我也反思了自己,觉得还是应该坚持本心,写博客是为了学习,不是为了哗众取宠。

《如果当时》系列博客,是基于自己真实的面试经历,全部都是面试真题,血和泪的教训,分享给大家,希望能给那些和我有同样困扰的人,提供一些帮助,努力一定就会有收获,加油。

一、ArrayList 和 Vector 的区别是什么?

  1. Vector 使用了 Synchronized 来实现线程同步,是线程安全的,而 ArrayList 是非线程安全的。
  2. ArrayList 和 Vector 都会根据实际的需要动态的调整容量,只不过在 Vector 扩容每次会增加 1 倍,而 ArrayList 只会增加 50%。
  3. 在不考虑并发问题的时候还是推荐使用Arraylist。

二、Array 和 ArrayList 有何区别?

  1. Array可以包含基本类型和对象类型,ArrayList只能包含对象类型。
  2. Array大小是固定的,ArrayList的大小是动态变化的。
  3. ArrayList提供了更多的方法和特性,比如:addAll(),removeAll(),iterator()等等。
  4. 对于基本类型数据,ArrayList 使用自动装箱来减少编码工作量;而当处理固定大小的基本数据类型的时候,这种方式相对比较慢,这时候应该使用Array。

三、除了 ReetrantLock,你还接触过 JUC 中的哪些并发工具?

我的答案:

ReetrantWriteReadLock:读写锁,包含读取和写入两种类型的锁,当进行读取操作时允许多线程访问,当进行写操作时,只允许一个线程访问;
ConcurrentHashMap:HashMap的同步版本,HashTable的升级版本,HashTable会锁住整个集合,而ConcurrentHashMap只是锁住当前访问的键值对;
Vector:Vector中加入了Synchronized锁,是线程安全的;相对于Arraylist,因为加锁的原因Vector效率要低一点;vector是1倍扩容,ArrayList是50%扩容;
Volatile:只能修饰变量,不能修饰代码块和方法,保证了可见性和有序性,不能保证原子性;

百度答案:

juc下常用的五个高并发工具:

  1. CountDownLatch:同步计数器
  2. CyclicBarrier: 线程屏障的功能
  3. Exchanger:用来使两个线程交换数据。
  4. Semaphore:控制信号量的个数,构造时传入个数。总数就是控制并发的数量。
  5. 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 粘包是怎么产生的?

  1. 发送方需要等缓冲区满才能发送出去,造成粘包;
  2. 接收方不及时接收缓冲区的包,造成粘包;

七、举一个用 Java 实现的装饰模式(decorator design pattern)?它是作用于对象层次还是类层次?

在Java IO中运用了装饰器模式,inputStream作为抽象类,其下有几个实现类,表示从不同的数据源输入:

  1. byteArrayInputStream
  2. fileInputStream
  3. StringBufferInputStream
  4. PipedInputStream,从管道产生输入;
  5. SequenceInputStream,可将其他流收集合并到一个流内;

FilterInputStream作为装饰器在JDK中是一个普通类,其下面有多个具体装饰器比如BufferedInputStream、DataInputStream等。

FilterInputStream内部封装了基础构件:

protected volatile InputStream in;

而BufferedInputStream在调用其read()读取数据时会委托基础构件来进行更底层的操作,而它自己所起的装饰作用就是缓冲,在源码中可以很清楚的看到这一切。

八、请举例说明如何在 Spring 中注入一个 Java Collection?

Spring注入有四种方式,

  1. set注入;
  2. 构造器注入;
  3. 基于注解的注入;
  4. xml配置文件注入;

想要注入java collection,就是注入集合类:

  1. list
  2. set
  3. map
  4. 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 的内连接、左连接、右连接有什么区别?

  1. 内连接,显示两个表中有联系的所有数据;
  2. 左链接,以左表为参照,显示所有数据,右表中没有则以null显示
  3. 右链接,以右表为参照显示数据,,左表中没有则以null显示

十三、Redis支持的数据类型有哪些?

String、hash、list、set、zset(sorted set:有序集合)

十四、详细介绍一下 CMS 垃圾回收器?

CMS 垃圾回收器是Concurrent Mark Sweep,是一种同步的标记-清除,CMS分为四个阶段:

  1. 初始标记,标记一下GC Root能直接关联到的对象,会触发“Stop The World”;
  2. 并发标记,通过GC Roots Tracing判断对象是否在使用中;
  3. 重新标记,标记期间产生对象的再次判断,执行时间较短,会触发“Stop The World”;
  4. 并发清除,清除对象,可以和用户线程并发进行;

十五、新生代垃圾回收器和老生代垃圾回收器都有哪些?有什么区别?

新生代回收器:Serial、ParNew、Parallel Scavenge

老年代回收器:Serial Old、Parallel Old、CMS

新生代回收器一般采用的是复制算法,复制算法效率较高,但是浪费内存;

老生代回收器一般采用标记清楚算法,比如最常用的CMS;

十六、简述分代垃圾回收器是怎么工作的?

分代回收器分为新生代和老年代,新生代大概占1/3,老年代大概占2/3;

新生代包括Eden、From Survivor、To Survivor;Eden区和两个survivor区的 的空间比例 为8:1:1 ;

垃圾回收器的执行流程:

  1. 把 Eden + From Survivor 存活的对象放入 To Survivor 区;
  2. 清空 Eden + From Survivor 分区,From Survivor 和 To Survivor 分区交换;
  3. 每次交换后存活的对象年龄+1,到达15,升级为老年代,大对象会直接进入老年代;
  4. 老年代中当空间到达一定占比,会触发全局回收,老年代一般采取标记-清除算法;

上一篇:如果当时这20道题能答好,现在应该已经被录取了(记一次面试的亲身经历 2020-8-27)

下一篇:Java面试题总结(绝对经典)

  • 12
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 11
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 11
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

哪 吒

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值