1.并发和并行的区别
你吃饭吃到一半,电话来了,你一直到吃完了以后才去接,这就说明你不支持并发也不支持并行。你吃饭吃到一半,电话来了,你停了下来接了电话,接完后继续吃饭,这说明你支持并发。你吃饭吃到一半,电话来了,你一边打电话一边吃饭,这说明你支持并行。并发的关键是你有处理多个任务的能力,不一定要同时。并行的关键是你有同时处理多个任务的能力。
`
作者:龚昱阳 Dozer
链接:https://www.zhihu.com/question/33515481/answer/58849148
来源:知乎
桥接方法:
当实现具有泛型的类及类中具有表示泛型的方法时,Java虚拟机会自动在字节码中给该类生成一个用Object代替泛型类的桥接方法,以便于向后兼容。
java中什么是bridge method(桥接方法)
`
虚拟机类加载的过程
虚拟机类加载过程包括加载、验证、准备、解析、初始化、使用和卸载七个阶段。
加载:
加载的过程需要完成进行三个步骤:
①通过类的全限定名获取定义此类的二进制字节流
②将字节流代表的静态存储结构转化为方法区的运行时数据结构
③在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口
加载过程包括非数组类的加载和数组类的加载
非数组类的加载:执行以上三个加载步骤
数据类的加载:如果数组的组件类型是引用类型,则递归进行以上三个步骤。如果不是引用类型,则使用引导类加载器进行加载。
验证:
包括4个阶段的验证工作:
①文件格式验证:是否符合Class文件格式的规范,如是否以魔术开头。
②元数据验证:对字节码描述的信息进行语义分析,如类是否有父类。
③字节码验证:通过数据流和控制流分析,确认程序语义是合法的、符合逻辑的,如保证跳转执行不会跳转到方法体以外的字节码指令上。
④符号引用验证:对类自身以外的信息进行匹配性校验,如符号引用中通过字符串描述的全限定名是否能找到对应的类
准备:
在这个阶段正式为类变量分配内存并设置类变量初始值,进行内存分配的变量仅包含类变量(被static修饰的变量),而不包括实例变量。
初始值通常情况下是零值。如果Java代码中对类变量进行赋值,在准备阶段后,类变量的初始值依然是零值。但是,如果该类变量是由final进行修饰的基本数据类型或java.lang.String类型(如public static final String str = "string";),那么该类字段的字段表属性将存在ConstantValue属性,在准备阶段变量就会被初始化为ConstantValue属性指定的值。
解析:
将常量池内的符号引用替换为直接引用的过程。符号引用包括以下三类:①类和接口的全限定名。②字段的名称和描述符。③方法的名称和描述符。
虚拟机规范在16种操作符号引用的字节码指令之前(自己去查),会对他们所使用的符号引用进行解析。除了invokedynamic指令,虚拟机实现可以将第一次解析的结果进行缓存。
解析包括类和接口的解析、字段解析、类方法解析和接口方法解析。
初始化:
执行类构造器<clinit>()方法的过程。<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块中的语句合并产生的。和<init>()方法不同,并不需要显示地条用父类构造器。虚拟机会保证在子类的<clinit>()方法执行之前,父类的该方法已经执行完毕。如果一个类中没有静态语句块,也没有对变量的赋值操作,那么编译器可以不为这个类生成<clinit>方法。
和类不同,接口不能使用静态语句块,但存在变量初始化的赋值操作,而且执行接口的<clinit>()方法不需要先执行父接口的该方法,只有当父接口中定义的变量使用时,父接口才会初始化。接口的实现类在初始化时也不会执行接口的<clinit>()方法。
多个线程同时去初始化一个类,只会有一个线程去执行这个类的<clinit>()方法,其他线程会被阻塞。如果执行该方法的线程退出该方法后,其他线程唤醒之后不会再进入该方法。因为在同一个类加载器下,一个类型只会初始化一次。
双亲委派模型
系统提供了三种类加载器:启动类加载器、扩展类加载器和应用程序类加载器。双亲委派模型要求除了顶层的启动类加载器外,其他的类加载器都应当有自己的父类加载器。这里类加载器之间的父子关系一般不会继承的关系来实现,而是都使用组合关系来复用父加载器的代码。
如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把这个请求委派给父类加载器去完成,每一个层次的类加载器都是如此,因此所有的加载请求最终都应该传送到顶层的启动类加载器中,只有当附加在其反馈自己无法完成这个加载请求时,子加载器才会尝试自己去加载。
多态:
包括继承、重写或重载、和向上转型
面向对象的三个特性:
继承、封装和多态
方法调用:
方法调用阶段的唯一任务是确定被调用方法的版本,有两种方式:解析和分派
方法的解析:
调用目标在程序代码写好、编译器进行编译时就必须确定下来。这类方法的调用被称为解析。
符合“编译器可知,运行期不可变”要求的方法,主要包括静态方法和私有方法两大类
在Java中提供了5条方法调用字节码指令,其中invokestatic和invokespecial指令调用的方法,都可以在解析中确定唯一的调用版本,符合这个条件的有静态方法、私有方法、实力构造器、父类方法4类,它们和final修饰的方法可以称为非虚方法,与之相反的则是虚方法。
分派:
分派分配静态分派和动态分派。
Human man = new Man();Human为静态类型,Man为实际类型
静态分派:所有依赖静态类型来定位方法执行版本的分派动作称为静态分派。静态分派的典型应用的方法重载
字面量的强转过程:char->int->long->float->double->自动装箱的封装类型->封装类型实现的接口类型(Serialiable或者Comparable,两者同时出现需要自行制定)->父类(如Object)->变长参数
动态分派:在运行期根据实际类型确定方法执行版本的分派动作,典型应用是重写(@Override)。
单分派和多分派:
宗量:方法的接收者与方法的参数称为方法的总量
接收者:执行方法的所有者
根据分派基于多少种宗量,可以将分派分为单分派和多分派。静态分派属于多分派,动态分派属于单分派。
Java编译器的分类
三种:前端编译器、JIT编译器和AOT编译器。
volatile和static的区别: