一、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:有序集合)
最后
手绘了下图所示的kafka知识大纲流程图(xmind文件不能上传,导出图片展现),但都可提供源文件给每位爱学习的朋友
左链接,以左表为参照,显示所有数据,右表中没有则以null显示
- 右链接,以右表为参照显示数据,,左表中没有则以null显示
十三、Redis支持的数据类型有哪些?
String、hash、list、set、zset(sorted set:有序集合)
最后
手绘了下图所示的kafka知识大纲流程图(xmind文件不能上传,导出图片展现),但都可提供源文件给每位爱学习的朋友
[外链图片转存中…(img-d0jr7l1N-1718919302462)]