程序员面经集

Java开发面试高频考点学习笔记(每日更新)

2021-04-24  本文已影响0人  JAVA架构师的圈子

Java开发面试高频考点学习笔记(每日更新)

Java:

1.深拷贝和浅拷贝

内存中有栈区和堆区,基本类型数据直接存在栈中,而引用类型(new出来的)是在堆中存储,在栈中保存堆中的地址。也就是说引用类型中在栈中存的不是数据,而是地址。赋值其实就是拷贝。

在基本类型数据赋值的时候,没有深浅拷贝的区别,因为直接赋予的是数据。

但在引用类型数据赋值的时候,实际上是把原来的地址复制给了新的,并没有实际复制其中的数据,所以这是一个浅拷贝(拷贝的深度不够),当使用新的变量操作地址中的值的时候,旧变量对应的值也会发生改变。Java中Objectclone方法默认是浅拷贝。

深拷贝会创造另外一个一模一样的对象,新对象和原来的对象不共享内存,修改新对象不会影响旧对象。

参考文章

2.接口和抽象类的区别

参考文章

3.java的内存是怎么分配的

内存分配分为在栈上分配和在堆上分配,大多数都是引用类型,所以堆空间用的较多。

对象根据存活时间分为年轻代、年老代、永久代(方法区)

年轻代:对象被创建时,首先分配在年轻代。年轻代有三个区域:Eden区,survivor 0区和survive 1区,Eden区大多数对象消亡速度很快,Eden是连续的内存空间,分配内存很快。Eden区满的时候执行Minor GC,清理消亡对象,将存活的对象放在survivor 0区中,每次执行Minor GC的时候,将剩余存活对象都放在非空的survivor区中,survivor区满之后,就会清理并转移到另一个survivor区,也就是说总有一个survivor区是空的。HotSpot虚拟机中默认切换15次之后,仍然存活的对象放在年老代中。


年老代:年老代的空间一般比年轻代大,存放更多的对象,年老代内存不足的时候,执行Major GC(Full GC),如果对象比较大的情况,可能直接放在老年代上。有可能出现老年代引用新生代对象的情况,java维护一个512 byte的块“card table”,记录引用映射,进行Minor GC的时候直接查card table就可以了。

参考文章

4.java中的泛型是什么?类型擦除是什么?

java源代码要运行,首先要经过编译器编译出字节码,字节码存储着能被JVM解释运行的指令。java的泛型在运行时,无法获得类型参数的真正类型,因为编译器编译生成的字节码不包括类型参数的具体类型。
泛型是java 1.5之后引入的,其本质是参数化类型,也就是说变量的类型是一个参数,在使用的时候再指定为具体类型,泛型可以用于类、接口和方法。

public class User<T> {
    
    private T name;
}//泛型实际上就是把类型当作参数传入了
而类型擦除机制使得Java的泛型实际上是伪泛型,类型参数只存在于编译期,运行时,JVM并不知道泛型的存在。


public class ErasedTypeEquivalence {
  public static void main(String[] args) {
    Class c1 = new ArrayList<String>().getClass();
    Class c2 = new ArrayList<Integer>().getClass();
 System.out.println(c1 == c2); //代码输出是true
  }
}

在C++、C#这些支持真泛型的语言中,它们代表着不同的类,但在JVM看来他们是同一个类。无论何时定义一个泛型,相应的原始类型都会被自动提供,类型变量擦除,并使用其限定类型(无限定的变量用 Object)替换。Java 编译器是通过先检查代码中泛型的类型,然后在进行类型擦除,再进行编译。当具体的类型确定后,泛型提供了一种类型检测的机制,只有相匹配的数据才能正常的赋值,否则编译器就不通过。

参考文章

5.Java中的反射是什么

java反射就是把类中的各个成分映射成一个个java对象,在运行期间,对于任意一个类,都能够知道这个类的属性和方法,是一种动态获取信息、动态调用对象的方法。

我们使用的Spring/hibernate中使用了反射机制,在使用JDBC连接数据库使用class.forName()通过反射加载数据库的驱动程序。
Spring框架的IOC(动态加载管理bean)创建对象,AOP(动态代理)都和反射有关系。

6.序列化与反序列化

serializable接口是可以进行序列化的标志性接口,仅仅是告诉JVM该类对象可以进行序列化。
先让需要序列化的类实现serializable接口;序列化对象创建输出流ObjectOutputStream,然后调用writeObject()方法;反序列化对象创建输入流Obje ctInputStream,然后调用readObject()方法,得到一个object对象。最后关闭流。

7.Object有哪些方法?

equals:比较对象是否相等,这里实质是比较地址是否相等。
wait:调用wait方法会导致线程阻塞,释放该对象的锁
notify:调用对象的notify方法会随机解除该对象阻塞的线程,该线程重新获取该对象的锁
notifyAll:唤醒所有正在等待对象的线程,全部进入锁池竞争获取锁
wait,notify,notifyAll必须在synchronized方法块中使用。
toString:转换为字符串表示
getClass:返回对象运行时类,即反射机制。
hashCode: 对象在内存中的地址转换为int值。

8.JVM内存模型

程序计数器(PC register):线程执行的字节码行号指示器,线程私有,唯一一个没有内存超出错误的区域。

JVM调优参数

(1) -Xms:初始化堆内存。默认为物理内存的六十四分之一
(2) -Xmx: 最大堆内存。默认为物理内存的四分之一
(3) -Xss:单个线程栈的大小
(4) -Xmn:设置新生代的大小
(5) -XX:MetaspaceSize:设置元空间大小
(6) -XX:SurvivorRatio:调节新生代eden和S0、S1的空间比例 默认为8:1:1

JVM性能监控工具

(1)jps -l:查看进程号
(2)jstack:java堆栈跟踪工具 查看死锁和cpu占用过高的代码
(3)jinfo -flag查看运行的java程序参数属性的详情

9.类加载机制

类加载就是将类的数据从class文件加载到内存,并且进行校验解析和初始化,形成可以让虚拟机使用的java类型。

类的生命周期:加载,链接,初始化,使用,卸载。

  1. 加载:通过类名获取二进制字节流(通过类加载器),把静态数据结构放在方法区,内存中生成对应class对象,作为访问入口。
  2. 链接:确保当前字节流包含的信息符合虚拟机要求。正式分配内存,设置初始值(仅分配静态变量),虚拟机将常量池内的符号引用替换成直接引用。
  3. 初始化:按照代码逻辑,赋予属性真正的初始值,初始化阶段就是执行类构造器方法的过程。
  4. 类加载器:包括启动类加载器、扩展类加载器和应用程序类加载器。

10.对象的创建和对象的布局

对象创建的方法:

用new语句创建

调用clone方法,需要实现cloneable接口

反射:class的newInstance()

反序列化:从文件中获取一个对象的二进制流,使用ObjectInputStream的readObject方法。

对象创建的过程:

类加载检查:判断这个类是不是已经被加载链接初始化了。

为对象分配内存:如果内存规整,虚拟机使用碰撞指针法(指针向空闲区前移对象大小的距离);如果不规整则使用空闲列表法。并发安全:虚拟机维护一个列表记录哪些内存块可用,再分配的时候从列表中找到一块足够大的空间划分给对象实例,并更新列表内容。

初始化分配的空间:所有属性初始化为零,保证对象实例字段在不赋值的时候可以直接用

设置对象头信息

执行构造方法初始化

逃逸:方法体内创建的对象,方法体外被其他变量引用过。这样在方法执行完毕之后,该方法中创建的对象不能被GC回收。开启逃逸分析之后,如果对象的作用域仅在方法内,那对象可以创建在虚拟机栈上,随方法入栈创建,出栈销毁,减少GC回收压力。

对象的内存布局:包含三部分:对象头,实例数据和对齐填充。

对象头:运行时数据和类型指针。标记字段包含hashcodeGC分代年龄锁状态标志线程持有锁等信息类元数据的指针:可以知道这个对象是哪个类的实例。

实例数据:存储对象真正的数据,也包含父类的数据。

对齐填充:保证对象大小是8字节的整数倍。


11.Java的四种引用(强引用、软引用、弱引用和虚引用)

在jdk1.2之前,Java对引用的定义很传统:如果reference类型的数据中存储的数值是另一块内存的起始地址,就称这块内存代表一个引用。

 Object obj = new Object();

参考文章

12.内存泄露和内存溢出

内存泄漏的原因

1.长生命周期的对象持有短生命周期对象的引用。

2.连接未正常关闭。

3.变量作用域设置过大

避免内存泄漏

1.避免在循环中创建对象

2.没有用的对象尽早释放

3.慎用静态变量

4.字符串的拼接使用Stringbuffer/StringBuilder

5.增大xmx和xms的值

内存溢出的原因

1.加载数据过大

2.死循环或过多循环

3.启动参数中内存值设定过小

栈溢出

原因:递归深度过大、局部变量过大

解决:递归不要太深,局部变量改为静态变量

如果排查内存问题
1.JConsole:能看到内存用量的趋势,确定是否有问题
2.GC日志:能看到年轻代和老年代等区域配置是否合理
3.代码中打印内存使用量
4.分析dump文件:针对性的看到发生OOM时候的内存使用量和线程情况

13.List、Set和Map三者的区别和其底层数据结构

List:有序的对象

(1)ArrayList:数组
(2)Vector:数组
(3)LinkedList:双向链表

Set:不允许重复的集合

(1)HashSet(无序且唯一):基于HashMap
(2)LinkedHashSet基于HashMap
(3)TreeSet(有序且唯一):基于红黑树

Map:使用键值对存储

(1)HashMap:Jdk1.8之前HashMap由数组+链表组成,之后再链表长度大于阈值(默认8)时将链表转换为红黑树以减少搜索时间。
(2)LinkedHashMap:继承自 HashMap,所以它的底层仍然是基于拉链式散列结构即由数组和链表或红黑树组成。另外,LinkedHashMap 在上面结构的基础上,增加了一条双向链表,使得上面的结构可以保持键值对的插入顺序。
(3)HashTable:数组+链表组成,数组是HashMap的主体,链表为了解决哈希冲突
(4)TreeMap:红黑树
ArrayList、LinkedList、Vector的区别

14.创建线程的四种方式

继承Thread类,重写run方法,继承Thread类的线程类不能再继承其他父类。

实现Runnable接口,重写run方法

通过Callable接口和Future接口创建线程,执行call方法,有返回值可以抛异常

线程池。前三种的线程如果创建关闭频繁的话会消耗系统资源影响性能,而使用线程池可以不用线程的时候放回线程池,用的时候再从线程池取。

15.NIO、AIO和BIO

BIO:传统的网络通讯模型,同步阻塞IO。服务器实现是一个连接一个线程,客户端有连接请求的时候,服务端就要启动一个线程去处理。线程数量可能会爆炸导致崩溃。适用于连接数目小且固定的架构。
NIO:同步非阻塞。服务器实现是一个请求一个线程,客户端发送的连接请求都会注册到多路复用器上,复用器轮询到连接有IO请求才启动线程。适用于连接数目多且连接比较短的架构,比如聊天服务器。
AIO:异步非阻塞。用户进程只需要发起一个IO操作然后立即返回,等IO操作真正完成之后,应用程序会得到IO操作完成的通知。适用于连接数目多且连接长的架构。

16.重写和重载

重写(Override):重写是子类对父类允许访问的方法实现过程进行重新编写,返回值和形参都不能改变。重写的好处是子类可以根据特定需要,定义特定行为。异常范围可以减少,但是不能抛出新的或更广的异常。

class Animal{
   public void move(){
      System.out.println("动物可以移动");
   }
}
//加入Java开发交流君样:756584822一起吹水聊天
class Dog extends Animal{
   public void move(){
      System.out.println("狗可以跑和走");
   }
}
 
public class TestDog{
   public static void main(String args[]){
      Animal a = new Animal(); // Animal 对象
      Animal b = new Dog(); // Dog 对象
//加入Java开发交流君样:756584822一起吹水聊天
      a.move();// 执行 Animal 类的方法
      b.move();//执行 Dog 类的方法
   }
}

虽然b属于Animal类型,但是它运行的是Dog类的move方法。因为在编译阶段,只是检查参数的引用类型,运行时JVM指定对象的类型并运行该对象的方法。
方法重写规则

当需要在子类中调用父类的被重写方法时,使用super关键字。

重载(Overload):是在一个类里面,方法名字相同,参数不同的两个方法。返回类型可以相同也可以不同。每个重载的方法(或者构造函数)必须有一个独一无二的参数类型列表。常用于构造器重载。

重载规则

(1)被重载的方法必须改变参数列表。
(2)被重载的方法可以改变返回类型,可以改变访问修饰符,可以声明新的或更广的异常检查。
(3)方法能够在同一个类中或者在一个子类中被重载。

public class Overloading {
    public int test(){
        System.out.println("test1");
        return 1;
    }
 
    public void test(int a){
        System.out.println("test2");
    }   
 //加入Java开发交流君样:756584822一起吹水聊天
    //以下两个参数类型顺序不同
    public String test(int a,String s){
        System.out.println("test3");
        return "returntest3";
    }   
 
    public String test(String s,int a){
        System.out.println("test4");
        return "returntest4";
    }   
 
    public static void main(String[] args){
        Overloading o = new Overloading();
        System.out.println(o.test());
        o.test(1);
        System.out.println(o.test(1,"test3"));
        System.out.println(o.test("test4",1));
    }
}

方法重载和方法重写是java多态的不同表现。
参考文章

17.final/finally/finalize与static

18.String、StringBuffer和StringBuilder的区别

String是java编程中广泛使用的,但它的底层实现实际是一个final类型的字符数组,其中的值不可变,每次对String进行操作就会生成一个新对象,造成内存浪费。

private final char value[];

StringBuffer/StringBuilder:它们的底层是可变的字符数组,都继承AbstractStringBuilder抽象类,所以在进行频繁的字符串操作的时候,尽量使用这两个类,它们的区别是:StringBuilder是线程不安全的,但执行速度较快;StringBuffer线程安全,但执行速度慢。StringBuffer使用synchronized关键字进行同步锁。
另外,String类型的比较,“==”是比较两个内存地址是否一样,而“equals”是比较两个字符串的值是不是一样的。
参考文章

19.如果判断一个对象是否该被回收?

引用计数算法:为对象增加一个引用计数器,当对象增加一个引用的时候+1,引用失效-1,引用计数为0的对象可以被回收。但是当两个对象循环引用的情况下,计数器永远不为0,因此JVM不使用引用计数算法。
可达性分析算法:以GC Roots为起点开始搜索,可达的对象都是存活的,不可达的对象可以被回收,JVM使用该算法进行判断。GC Roots中包含:虚拟机栈中引用的对象、本地方法栈中引用的对象,方法区中静态成员或常量引用的对象。


20.垃圾收集算法

标记-清除算法(Mark-Sweep)

标记阶段:标记的过程实际上就是可达性分析算法过程,遍历GC Roots对象,可达的对象都做好标记,在对象的header中将其记录为可达。

清除阶段:对堆进行遍历,如果发现有某个对象没有可达对象标记,则回收。

缺点:两次遍历,效率低;GC运行时需要停止整个程序;产生大量的碎片,需要维护一个空闲列表。

复制算法(Copying)
对象在Survivor区每经历一次Minor GC,就将对象年龄+1,当对象年龄达到某个值时,对象复制到老年代,默认为15。JVM中EdenSurvivor区的默认比例为8:1:1,保证内存利用率为90%,如果每次回收有多于10%的对象存活,Survivor空间可能就不够用了,此时借用老年代空间。

缺点:复制收集算法在对象存活率高的时候需要进行很多的复制操作,效率会变低,老年代一般不会用该算法。

标记-整理算法

第一阶段和标记-清楚算法一样,第二阶段将所有存活的对象压缩到内存的另一端,按顺序排放。之后,清理边界外所有的空间。
缺点:效率不高,不仅要标记存活对象,还要整理所有存活对象的引用地址;移动过程中,要全程暂停用户应用程序。
分代收集算法

新生代:使用复制算法,因为大量对象需要回收。
老年代:回收的对象很少,所以采用标记清除或者标记整理算法。

21.Double与Float

java语言支持两种基本的浮点类型:floatdouble。32位浮点数float用1位表示符号,8位表示指数,用23位表示尾数;64位浮点数double用一位表示符号,11位表示指数,52位表示尾数。在表示超过23位的时候,float就会自动四舍五入,这就是float的精度限制,所以会出现double可以表示而float会不精确的情况,如果要将这两个浮点数进行转型,java提供了Float.doubleValue()Double.floatValue()方法。使用这个方法在单精度转双精度的时候,会出现偏差。
浮点运算很少是精确的,只要超过精度表示范围就会产生误差。

解决方法:可以通过String结合BigDecimal或者通过使用long类型来转换。

参考文章

22.垃圾收集器

查看默认垃圾收集器:-XX:+PrintCommandLineFlags

(1)初始标记:标记GC Roots能直接关联到的对象,速度很快,需要停顿。
(2)并发标记:进行GC Roots Trancing的过程,不需要停顿。
(3)重新标记:修正并发标记期间因为用户程序继续运作而导致变动的那一部分对象重新进行标记,需要停顿。
(4)并发清除:不需要停顿。

G1垃圾收集器:它使得Eden、Survivor和Tenured等内存区域不再连续,而变成一个个大小一样的region,每个region从1M到32M不等。它不再采用CMS的标记清理算法,G1整体上使用标记整理算法,局部上看是基于复制算法。JVM参数:-XX:+UseG1GC。

降低停顿时间是G1和CMS共同的关注点,但G1除了追求低停顿外,还能建立可以预测的停顿时间模型,能让使用者明确指定在一个长度为M毫秒的时间片内。是因为G1收集器在后台维护了一个优先列表,每次根据允许的收集时间,优先选择回收价值最大的region。
另:JVM设置参数的方法(win10):环境变量中新建变量JAVA_OPTS,在里面设置。

23.线程池

我们使用线程的时候去创建一个线程,这种方法非常简便,但是会导致一个问题:如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了,这样频繁的创建线程会大大降低系统效率。



Java中引入了线程池来使得线程可以复用,执行完一个任务不会被立刻销毁,而是可以继续执行其他任务。

ThreadPoolExecutor类是线程池技术最核心的类:

其构造器中的参数意义

(1)ThreadPoolExecutor.AbortPolicy:丢弃任务抛出RejectedExecutionException异常;
(2)ThreadPoolExecutor.DiscardPolicy:丢弃任务,不抛异常
(3)ThreadPoolExecutor.DiscardOldestPolicy:丢弃队列最前面的任务,然后重新尝试执行任务(重复该过程)
(4)ThreadPoolExecutor.CallRunsPolicy:由调用线程处理该任务

ThreadPoolExecutor类的方法

execute()submit():都是提交任务,execute方法用于提交不需要返回值的任务,无法判断任务是不是被线程池执行成功;submit提交需要返回值的任务,线程池返回future类型的对象以判断是否执行成功,future对象具有的get()方法可以获取返回值。`

shutdown()shutdownNow():都是关闭线程池,他们的原理是遍历线程池中的工作线程,然后逐个调用线程的interrupt方法来中断线程,所以无法响应中断的任务可能永远无法终止。shutdownNow首先将线程池的状态设置成STOP,然后尝试停止所有正在执行或者暂停的线程,并返回等待执行任务的列表;shutdown只是将线程池的状态设置为SHUTDOWN,然后中断所有没有执行任务的线程。

如何合理分配线程池的大小:CPU密集型任务,一般公式为:最大线程数 = CPU核数+1;IO密集型的最大线程数 = CPU核数 * 2;

实现一个线程池:

public class Test {
     public static void main(String[] args) {   
         ThreadPoolExecutor executor = new ThreadPoolExecutor(5, 10, 200, TimeUnit.MILLISECONDS,new ArrayBlockingQueue<Runnable>(5));
          
         for(int i=0;i<15;i++){
            MyTask myTask = new MyTask(i);
            executor.execute(myTask);
            System.out.println("线程池中线程数目:"+executor.getPoolSize()+",队列中等待执行的任务数目:"+
            executor.getQueue().size()+",已执行完别的任务数目:"+executor.getCompletedTaskCount());
         }
         executor.shutdown();
     }
}

线程池不允许使用Executors的静态方法创建,必须通过ThreadPoolExecutor。

线程池的处理流程
当线程池提交一个任务的时候:
(1)线程池判断核心线程池中的线程是不是都在执行任务,如果不是则创建一个新的工作线程执行任务,否则进入流程(2)
(2)线程池判断工作队列是否已满,如果没有满则将新提交的任务存储在这个任务队列中,如果工作队列满了,则进入流程(3)
(3)线程池判断池中的线程是否都处在工作状态,如果没有则创建一个新的工作线程来执行任务,如果已经满了就交给拒绝策略(handler)来处理任务。
参考文章

四种线程池:
(1)newCachedThreadPool 创建一个可以缓存的线程池。
(2)newFixedThreadPool 创建一个定长线程池,可以控制线程最大并发数。
(3)newScheduledThreadPool 创建一个定长线程池,支持定时和周期性任务执行。
(4)newSingleThreadExecutor 创建一个单线程化的线程池,他只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序执行。

//可以缓存的线程池
ExecutorService cachedThreadPool = Executors.newCachedThreadPool(); //需要指定长度
ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);

详细实现代码

24.线程同步和线程通讯

线程同步的五种方式:synchronized的关键字修饰方法、静态资源或者代码块;Lock(必须放在try-catch-finally中执行,finally释放锁以防止死锁);waitnotify,必须在synchronized范围内,被synchronized锁住的对象就是wait和notify的调用对象;CAS;信号量(Semaphore)。

线程通讯的方式:

25.中断线程

调用一个线程的interrupt()方法来中断线程,如果该线程处于阻塞、限期等待或者无限期等待状态,那么就会抛出InterruptedException,从而提前结束该线程。

如果线程的run()执行一个死循环,并且没有执行sleep()等会抛出InterruptedException的操作,那么调用interrupt()方法无法使线程提前结束。但是调用interrupt方法会设置线程的中断标记,此时调用Thread.interrupted()Thread.currentThread().isInterrupted()方法会返回true。因此可以在循环体中使用interrupted()方法判断线程是否处于中断状态,从而提前结束线程。

26.Synchronized的用法

线程安全是Java并发编程中的重点,造成线程安全问题主要有两个原因:一是存在共享数据,二是存在多条线程共同操作共享数据。因此,当存在多个线程操作共享数据的时候,需要保证同一时刻有且只有线程在操作共享数据,其他线程必须等到该线程处理完才能进行,这种方式叫做互斥锁。Java中,关键字synchronized可以保证在同一时刻,只有一个线程可以执行某个方法或者某个代码块,同时它还可以保证一个线程(共享数据)的变化被其他线程所看到(可见性保证,完全可以替代Volatile功能)

synchronized是Java的关键字,是一种同步锁。

Java的内置锁(synchronized):每个java对象都可以用做一个实现同步的锁,这些锁称为内置锁。线程进入同步代码块或方法的时候会自动获得该锁,退出同步代码块的时候会释放该锁。获得内置锁的唯一途径就是进入锁保护的同步代码块/方法。

Java的对象锁和类锁:在锁的概念上与内置锁一致,但对象锁是用于对象实例方法或对象实例上的,类锁是用于类的静态方法或者一个类的class对象上的。

Java中每个对象都有一把锁和两个队列,一个队列用于挂起未获得锁的线程,一个队列用于挂起条件不满足而等待的线程。synchronized实际上是一个加锁和释放锁的集成。JVM负责跟踪对象被加锁的次数。如果一个对象被解锁,计数归零。线程第一次给对象加锁的时候,计数变成1。每当这个相同的线程在此对象上获得锁的时候,计数就会递增。每当任务离开一个synchronized方法,计数就会递减,为0的时候锁被完全释放。

Synchronized有三种应用方式:

修饰一个实例方法:被修饰的方法称为实例同步方法,其作用范围是整个方法,锁定的事该方法所属的对象(调用该方法的对象)。所有需要获得该对象锁的操作都会对该对象加锁。

  public synchronized void method(){}
  //等同于
  public void method(){
    synchronized(this){
    }
  }

如果一个对象有多个synchronized方法,只要一个线程访问了其中的一个synchronized方法,其他线程不能同时访问这个对象中任何一个synchronized方法。
当一个对象O1在不同的线程中执行这个同步方法的时候,会形成互斥。但是O1对象所属类的另一对象O2是可以调用这个被加了synchronized关键字的方法的。其他线程调用O2中的相同方法时不会造成同步阻塞。程序可能在这种情况下摆脱同步机制的控制,造成数据混乱。注意:

修饰一个静态方法:被修饰的方法被称为静态同步方法,其作用域是整个静态方法,锁是静态方法所属的类。

 public synchronized static void method(){}
修饰代码块:被修饰的代码块被称为同步语句块。synchronized的括号中必须传入一个对象作为锁,作用范围是大括号中的代码,锁是synchronized括号中的内容,可以分为类锁和对象锁

//锁对象为实例对象
 public void method(Object o){
  synchronized(o){
   ...
  }
 }//加入Java开发交流君样:756584822一起吹水聊天
//锁对象为类的Class对象 
 public class Demo{
   public static void method(){
      synchronized(Demo.class){
       ...
      }
   }
 }

27.Synchronized的原理

实际上是通过monitor(监视器)。Java中的同步代码块是使用monitorentermonitorexit指令实现的,其中monitorenter指令插入到同步代码块的开始位置,monitorexit指令插入同步代码块的结束位置。

JVM保证这两个指令成对出现。
当执行monitorenter指令的时候,线程试图获取锁也就是获取monitor对象的所有权,当计数器为0的时候就可以成功获取,获取后将计数器加一。在执行monitorexit指令之后,将锁计数器减一,表明锁被释放。
synchronized修饰方法的时候,没有monitorentermonitorexit指令,取而代之的是ACC_SYNCHRONIZED标识,这个标识指明这个方法是一个同步方法。

28.Synchronized的四种状态

无锁-->偏向锁-->轻量级锁-->重量级锁(过程不可逆)

偏向锁:大多数情况下,锁不存在多线程竞争,总是由同一线程多次获得;如果一个线程获得了锁,锁进入偏向模式,此时对象头的Mark Word结构也变为偏向锁结构。

对象头在第十章节中提到过,另外这篇文章讲的更详细。
当该线程再次请求锁的时候,只需要检查Mark Word锁标记为是否为偏向锁,以及当前线程ID是不是等于Mark Word的Thread Id即可,省去了大量有关锁申请的操作。
偏向锁只适用于只有一个线程访问同步块的场景。


轻量级锁:当锁是偏向锁的时候,被另外的线程所访问,偏向锁就会升级为轻量级锁,其他线程会通过自旋的形式尝试获取锁,不会阻塞,从而提高性能。适用于追求响应时间,同步快执行速度非常快的情况。

代码在进入同步块的时候,如果同步对象锁状态是无锁,虚拟机首先在当前线程的栈帧中创建锁记录(Lock Record)空间,拷贝对象头的Mark Word复制到锁记录中。

之后虚拟机使用CAS操作尝试将对象的Mark Word更新为指向Lock Record的指针,并将Lock Record的owner指针指向对象的Mark Word。如果这个动作成功了,那么这个线程就有了该对象的锁,对象的锁标记为设置为“00”,说明处于轻量级锁定状态。

如果这个动作失败了,JVM检查对象的Mark Word是否指向当前线程的栈帧,是则说明当前线程已经拥有了这个对象的锁,否则说明多个线程竞争锁。

如果有两个以上的线程竞争同一个锁,轻量级锁不再有效,膨胀为重量级锁。


重量级锁:多线程情况,线程阻塞响应时间缓慢,频繁的释放获取锁会带来巨大的性能损耗。适用于追求吞吐量,同步快执行速度较长的情景。

29.Synchronized与重入锁ReentrantLock的区别

相对与ReentrantLock而言,synchronized锁是重量级的,而且是内置锁,意味着JVM可以对synchronized锁做优化。
在synchronized锁上阻塞的线程是不可中断的,而ReentrantLock锁实现了可中断的阻塞。

synchronized锁释放是自动的,而ReentrantLock需要显式释放(在try-finally块中释放)\

线程在竞争synchronized锁的时候是非公平的:如果synchronized锁被线程A占有,线程B请求失败,被放入队列中,线程C此时来请求锁,恰好A在此时释放了,线程C会跳过队列中等待的线程B直接获得这个锁。但是ReentrantLock可以实现锁的公平性。

synchronized锁是读写和读读都互斥,ReentrankWriteLock分为读锁和写锁,读锁可以同时被多个线程持有,适合于读多写少的并发场景。

ReentrantLock只能锁代码块,但是synchronized可以锁方法和类。ReentrantLock可以知道线程有没有拿到锁,但是synchronized不行。

有关synchronized的参考文章

30.锁优化

在28章节中,我们提到过重量级锁,在重量级锁中,JVM会阻塞未获取到锁的线程,在锁被释放的时候唤醒这些线程,阻塞和唤醒依赖于操作系统,需要从用户态切换到内核态,开销很大。monitor调用了OS底层的互斥量(mutex),切换成本很高。因此JVM引入了自旋的概念。

自旋锁与自适应自旋锁,CAS实现:

public static String test04(String s1, String s2, String s3) {
        StringBuffer sb = new StringBuffer();
        sb.append(s1);
        sb.append(s2);
        sb.append(s3);
        return sb.toString();
    }

上述连续的append操作就属于这类情况,jvm检测到一连串操作都是对同一个对象加锁,就会把锁同步范围扩展(粗化)到整个一系列操作的外部,使得一连串append操作只需要加一次锁就可以了。

31.Java设计模式

设计模式是一套被反复使用,多数人知晓的,经过分类编目的,代码设计经验的总结。使用设计模式是为了可重用代码,让代码更容易被他人理解。实际上就是在某些场景下,针对某类问题的某种通用的解决方案。
设计模式分为三类:

单例模式:属于创建型模式,主要有三种写法:懒汉式、饿汉式和登记式。

单例模式的特点:

懒汉式:在第一次调用的时候就实例化自己。

  public class Singleton{
    private Singleton(){}
    private static Singleton single = null;
    //静态工厂方法
    private static Singleton getInstance(){
      if(single == null) single = new Singleton();
    }
    return single;
  }

懒汉式并不考虑线程安全问题,所以他是线程不安全的,并发情况下很可能出现多个Singleton实例,要实现线程安全,有以下三个方式:

getInstance方法上加同步关键字:在并发环境下,多个一起进入getInstance里,因为还没有实例化单例模式,single都是null,就会创建多个Singleton实例化对象,破坏了单例模式想要的结果。我们可以在getInstance方法上加synchronized锁。

 public static synchronized Singleton getInstance(){
   if(single == null) single = new Singleton();
   return single;
 }

双重校验锁定:

 public static Singleton getInstance(){
  if(singleton == null){
    synchronized (Singleton.class){
       if(singleton == null) singleton = new Singleton();
    }
  }
  return singleton;
 }

双重校验锁定的单例仍然需要再加上volatile确保线程安全。

静态同步类:即实现了线程安全,又避免了同步带来的性能影响。

 public class Singleton{
   private static class LazyHolder{
     private static final Singleton INSTANCE = new Singleton();
   }
   private Singleton(){}
   public static final Singleton getInstance(){
     return LazyHolder.INSTANCE;
   }
 }

饿汉式:饿汉式在类创建的同时就已经创建好了一个静态的对象供系统使用,以后不再改变,所以天生是线程安全的。

  public class Singleton1{
    private Singleton1(){}
    private static final Singleton1 single = new Singleton1();
    //静态工厂方法
    public static Singleton1 getInstance(){
       return single;
    }
  }

饿汉就是类一旦加载,就把单例初始化完成,保证getInstance的时候,单例已经存在了;而懒汉比较懒,只有用户调用getInstance的时候,才会初始化这个实例。

总结

生命不止坚毅鱼奋斗,有梦想才是有意义的追求
给大家推荐一个免费的学习交流君样:756584822
最后,祝大家早日学有所成,拿到满意offer,快速升职加薪,走上人生巅峰。

Java开发交流君样:756584822

上一篇 下一篇

猜你喜欢

热点阅读