作者:Tom哥
公众号:微观技术
博客:https://offercome.cn
人生理念:知道的越多,不知道的越多,努力去学
答案:
1、跨平台。JAVA 的 class 文件是运行在虚拟机上的,虚拟机在不同平台有不同版本,所以说 JAVA 是跨平台的,“一次编写,到处运行(Write Once,Run any Where)”
2、面向对象。具有封装、继承、多态特点
3、支持多线程。C++ 语言没有内置的多线程机制,因此必须调用操作系统的多线程功能来进行多线程 程序设计,而 Java 语言却提供了多线程支持;
4、网络编程方便。Java 语言诞生本身就是为简化网络编程设计的,因此 Java 语言不仅支 持网络编程而且很方便
答案:
1、面向过程:
2、面向对象:
答案:
答案
接口是抽象类的变体,接口中所有的方法都是抽象的。而抽象类是声明方法的存在而不去实现它的类。
接口可以多继承,抽象类不行。
接口定义方法,不能实现,默认是 public abstract,而抽象类可以实现部分方法。
接口中基本数据类型为 public static final 并且需要给出初始值,而抽类象不是的。
答案:
1、重写:
2、重载:
答案:
构造器不能被继承,因此不能被重写,但可以被重载。每一个类必须有自己的构造函数,负责构造自己 这部分的构造。子类不会覆盖父类的构造函数,相反必须一开始调用父类的构造函数
答案:
1、JDK(全称 Java Development Kit),拥有 JRE 所拥有的一切,还有编译器(javac)和工具(如 javadoc 和 jdb)。能独立创建、编译、运行程序。
JDK = JRE + java开发工具(javac.exe/java.exe/jar.exe)
2、JRE(全称 Java Runtime Environment),包括 Java 虚拟机(JVM),Java 类库,java 命令和其他的一些基础构件。能运行已编译好的程序,但不能创建程序
JRE = JVM + java核心类库
3、JVM (全称 Java Virtual Machine),java虚拟机。
答案:
Java之所以可以“一次编译,到处运行
”,一是因为 JVM 针对各种操作系统、平台都进行了定制。二是因为无论在什么平台,都可以编译成固定格式的字节码(.class文件)供JVM使用。因此,也可以看出字节码对于Java生态的重要性。
字节码文件由十六进制值组成,而JVM以两个十六进制值为一组,即以字节为单位进行读取。在Java中一般是用javac命令
编译源代码为字节码文件。
Java语言通过字节码的方式,在一定程度上解决了传统解释型语言执行效率低的问题,同时又保留了解释型语言
可移植的特点。所以Java程序运行时比较高效,而且,由于字节码并不专对一种特定的机器, 因此,Java程序无须重新编译便可在多种不同的计算机上运行。
答案:
Java中,可以使用访问控制符来保护对类、变量、方法和构造方法的访问。Java 支持 4 种不同的访问权 限。
1、default (即默认,什么也不写): 在同一包内可见,不使用任何修饰符。使用对象:类、接口、变量、方法。 2、private : 在同一类内可见。使用对象:变量、方法。注意:不能修饰类(外部类)
3、public : 对所有类可见。使用对象:类、接口、变量、方法
4、protected : 对同一包内的类和所有子类可见。使用对象:变量、方法。注意:不能修饰类(外部类)
答案:
1、break 跳出总上一层循环,不再执行循环(结束当前的循环体)
2、continue 跳出本次循环,继续执行下次循环(结束正在执行的循环 进入下一个循环条件)
3、return 程序返回,不再执行下面的代码(结束当前的方法,直接返回)
答案:
1、new关键字
Person p1 = new Person();
2、Class.newInstance
Person p1 = Person.class.newInstance();
3、Constructor.newInstance
Constructor<Person> constructor = Person.class.getConstructor();
Person p1 = constructor.newInstance();
4、clone
Person p1 = new Person();
Person p2 = p1.clone();
5、反序列化
Person p1 = new Person();
byte[] bytes = SerializationUtils.serialize(p1);
Person p2 = (Person)SerializationUtils.deserialize(bytes);
答案:
1、值传递:
指的是在方法调用时,传递的参数是按值的拷贝传递,传递的是值的拷贝,也就是说传递后就 互不相关了。
2、引用传递:
指的是在方法调用时,传递的参数是按引用进行传递,其实传递的是引用的地址,也就是变 量所对应的内存空间的地址。也就是说传递前和传递后都指向同一个引用(同一个内存空间)
基本类型作为参数被传递时肯定是值传递;引用类型作为参数被传递时也是值传递,只不过“值”为对应的引用
答案:
1、== :如果是基本数据类型,比较两个值是否相等;如果是对象,比较两个对象的引用是否相等,指向同一块内存区域
2、equals:用于对象之间,比较两个对象的值是否相等。
答案:
hashCode() 的作用是生成哈希码,也称为散列码,返回一个 int 整数。 哈希码用来确定该对象在哈希表中的索引位置。hashCode() 定义在JDK的Object.java中,每个类中都包含这个方法。 散列表存储的是键值对(key-value),它的特点:能根据“键”快速的检索出对应的“值”。这其中就利用到了散列码,可以快速找到所需要的对象。
答案:
以HashSet 如何检查重复
为例子来说明为什么要有 hashCode。
当你把对象加入 HashSet 时,HashSet 会先计算对象的 hashcode 值来判断对象加入的位置,同时也会与其他已经加入的对象的 hashcode 值作比较,如果没有相符的hashcode,HashSet会假设对象没有重复出现。
但是如果发现有相同 hashcode 值的对象,这时会调用 equals()方法来检查 hashcode 相等的对象是否真的相同。如果两者相同,HashSet 就不会让其添加成功。如果不同的话,就会重新散列到其他位置。这样我们就大大减少了 equals 的次数,大大提高执行速度
答案:
重要规范:
答案:
1、不变性:String 是只读字符串,是一个典型的 immutable 对象,对它进行任何操作,其实都是创建一个新的对象,再把引用指向该对象。不变模式的主要作用在于当一个对象需要被多线程共享并频繁访问时,可以保证数据的一致性;
2、常量池优化:String 对象创建之后,会在字符串常量池中进行缓存,如果下次创建同样的对象时,会直接返回缓存的引用
3、final:使用 final 来定义 String 类,表示 String 类不能被继承,提高了系统的安全性。
答案
1、String。采用 final
修饰,对象不可变,线程安全。如果对一个已经存在的String对象修改,会重新创建一个新对象,并把值放进去。
2、StringBuffer,采用 synchronized
关键字修饰,线程安全
3、StringBuilder,非线程安全,但效率会更高些,适用于单线程。
答案:
HashMap 通过 key 的 hashcode 来确定 value 的存储位置,因为字符串是不可变的,所以当创建字符串后,它的 hashcode 被缓存下来,不需要再次计算,所以相比于其他对象更快。
答案:
Integer的默认值是null;int的默认值是0
final 变量:被修饰的变量不可变,不可变分为 引用不可变 和 对象不可变 ,final 指的是 引用不可变 ,final 修饰的变量必须初始化,通常称被修饰的变量为常量 。
final 方法:被修饰的方法不允许被覆盖,子类可以使用该方法。
final 类:被修饰的类不能被继承,所有方法不能被重写。
2、finally 作为异常处理的一部分,它只能在 try/catch 语句中,并且附带一个语句块表示这段语句最终 一定被执行(无论是否抛出异常),经常被用在释放资源。
3、finalize 是在 java.lang.Object 里定义的方法,也就是说每一个对象都有这么个方法,这个方法在 gc 启动,该对象被回收的时候被调用。
一个对象的 finalize 方法只会被调用一次,finalize 被调用不一定会立即回收该对象,所以有可能调用 finalize 后,该对象又不需要被回收了,然后到了真正要被回收的时候,因为前面调用过一次,所以不会 再次调用 finalize 了,进而产生问题,因此不推荐使用 finalize 方法。
答案:
当然啦,会在return之前执行。
答案:
static 变量在Java中是属于类的,它在所有的实例中的值是一样的。当类被Java虚拟机载入的时候,会对static 变量进行初始化。如果你的代码尝试不用实例来访问非static的变量,编译器会报错,因为这些变量还没有被创建出来,还没有跟任何实例关联上。
答案:
1、final 用于修饰变量、方法和类。
答案:
可以,因为都是类初始化的时候加载的。
答案:
可以,非静态方法就是实例方法,那是new之后才产生的,那么属于类的内容它都认识。
答案:
JAVA 标准库内建了一些通用的异常,这些类以Throwable
为顶层父类。
Throwable又派生出 Error类 和 Exception 类 。
1、错误: Error 属于程序无法处理的错误 ,我们没办法通过 catch 来进行捕获。例如,系统崩溃,内存不足,堆栈溢出等,编译器不会对这类错误进行检测,一旦这类错误发生,通常应用程序会被终止,仅靠应用程序本身无法恢复。
2、异常: Exception以及他的子类,代表程序运行时发送的各种不期望发生的事件。可以被Java异常处理机制使用,是异常处理的核心。 Exception 又可以分为运行时异常 (RuntimeException,又叫非受检查异常) 和非运行时异常 (又叫受检查异常) 。
非受检查异常和受检查异常之间的区别:
是否强制要求调用者必须处理此异常,如果强制要求调用者必须进行处理,那么就使用受检查异常,否则就选择非受检查异常
答案:
非受检查异常:包括 RuntimeException
类及其子类,表示 JVM 在运行期间可能出现的异常。 Java 编译器不会检查运行时异常。例如: NullPointException(空指针) 、 NumberFormatException(字符串转换为数字) 、 IndexOutOfBoundsException(数组越界) 、 ClassCastException(类转换异常) 、ArrayStoreException(数据存储异常,操作数组时类型不一致) 等。
答案:
受检查异常:是Exception 中除 RuntimeException 及其子类之外的异常。 Java 编译器会检查受检查异常。
常见的受检查异常有:IO 相关的异常
、 ClassNotFoundException
、 SQLException
等。
答案:
答案:
反射是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为 Java 语言的反射机制。
答案:
1、优点:
能够运行时动态获取类的实例,提高灵活性;可与动态编译结合
Class.forName('com.mysql.jdbc.Driver.class'); // 加载MySQL的驱动类。
2、缺点:
使用反射性能较低,需要解析字节码,将内存中的对象进行解析。其解决方案是:通过 setAccessible(true)
,关闭JDK的安全检查来提升反射速度;多次创建一个类的实例时,有缓存会快很多;ReflflectASM工具类,通过字节码生成的方式加快反射速度。
答案:
1、Class.forName(“类的路径”);当你知道该类的全路径名时,你可以使用该方法获取 Class 类对象。
Class clz = Class.forName("java.lang.String");
2、类名.class。这种方法只适合在编译前就知道操作的 Class。
Class clz = String.class;
3、对象名.getClass()
String str = new String("Hello");
Class clz = str.getClass();
4、如果是基本类型的包装类,可以调用包装类的Type属性来获得该包装类的Class对象。
答案:
反射 API 用来生成 JVM 中的类、接口或则对象的信息。
答案 : 1、反射让开发人员可以通过外部类的全路径名创建对象,并使用这些类,实现一些扩展的功能。
2、反射让开发人员可以枚举出类的全部成员,包括构造函数、属性、方法。以帮助开发者写出正确的代码。
3、测试时可以利用反射 API 访问类的私有成员,以保证测试代码覆盖率
答案: 泛型是 JDK1.5 的一个新特性,泛型就是将类型参数化,其在编译时才确定具体的参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法
泛型是一种语法糖。泛型只存在于编译阶段,而不存在于运行阶段。在编译后的 class 文件中,是没有泛型这个概念的。
答案:
1、类型安全
2、消除强制类型转换
3、潜在的性能收益
答案:
1、限定通配符
对类型进行了限制。有两种限定通配符,一种是 <? extends T> 它通过确保类型必须是T的子类来设定类型的上界。另一种是 <? super T> 它通过确保类型必须是T的父类来设定类型的下界。泛型类型必须用限定内的类型来进行初始化,否则会导致编译错误。
2、非限定通配符 ?
可以用任意类型来替代。如 List<?> 的意思是这个集合是一个可以持有任意类型的集合,它可以是 List<A> ,也可以是 List<B> ,或者 List<C> 等等。
答案:
1、序列化:序列化是把对象转换成有序字节流,以便在网络上传输或者保存在本地文件中。核心作用是对象状态的保存与重建。我们都知道,Java对象是保存在JVM的堆内存中的,也就是说,如果JVM堆不存在了,那么对象也就跟着消失了。 而序列化提供了一种方案,可以让你在即使JVM停机的情况下也能把对象保存下来的方案。就像我们平时用的U盘一样。把Java对象序列化成可存储或传输的形式(如二进制流),比如保存在文件中。这样,当再次需要这个对象的时候,从文件中读取出二进制流,再从二进制流中反序列化出对象。
2、反序列化:客户端从文件中或网络上获得序列化后的对象字节流,根据字节流中所保存的对象状态及描述信息,通过反序列化重建对象。
答案:
serialVersionUID 用来表明类的不同版本间的兼容性。 Java 的序列化机制是通过在运行时判断类的 serialVersionUID 来验证版本一致性的。在进行反序列化时,JVM会把传来的字节流中的 serialVersionUID 与本地相应实体(类)的serialVersionUID进行比较,如果相同就认为是一致的,可以进行反序列化,否则就会出现序列化版本不一致的异常。
答案:
如果不显示指定serialVersionUID,JVM在序列化时会根据属性自动生成一个serialVersionUID, 然后与属性一起序列化, 再进行持久化或网络传输。在反序列化时,JVM会再根据属性自动生成一个新版serialVersionUID,然后将这个新版serialVersionUID与序列化时生成的旧版 serialVersionUID 进行比较,如果相同则反序列化成功,否则报错。
如果显示指定了,JVM在序列化和反序列化时仍然都会生成一个serialVersionUID,但值为我们显示指定的值,这样在反序列化时新旧版本的serialVersionUID就一致了。
在实际开发中,不显示指定serialVersionUID的情况会导致什么问题? 如果我们的类写完后不再修改,那当然不会有问题,但这在实际开发中是不可能的,我们的类会不断迭代, 一旦类被修改了, 那旧对象反序列化就会报错。所以在实际开发中, 我们都会显示指定一个serialVersionUID,值是多少无所谓, 只要不变就行。
答案: 《阿里巴巴Java开发手册》中有以下规定:
答案: 对于不想进行序列化的变量,使用 transient 关键字修饰。
答案:
1、进程:是一个程序的执行流程,是系统进行资源分配和调度的基本单位,作用是程序能够并发执行提高资源利用率。因为进程的创建、销毁、切换产生大量的时间和空间的开销,所以进程的数量不能太多
2、线程:是比进程更小的能独立运行的基本单位,他是进程的一个实体,可以减少程序并发执行时的时间和空间开销,使得操作系统具有更好的并发性。多个线程可以共享进程的系统资源。线程基本不拥有系统资源,只有一些运行时必不可少的资源,比如程序计数器、寄存器和栈,进程则占有堆。
答案:
AQS内部维护一个state状态位,尝试加锁的时候通过CAS(CompareAndSwap)修改值,如果成功设置为 1,并且把当前线程ID赋值,则代表加锁成功。 一旦获取到锁,其他的线程将会被阻塞进入阻塞队列自旋,获得锁的线程释放锁的时候将会唤醒阻塞队列中的线程,释放锁的时候则会把state重新置为0,同时当前线程ID置为空。
答案:
答案:
内部由数组和链表组成,非线程安全。JDK1.7和1.8的主要区别在于头插和尾插方式的修改,头插容易导致HashMap链表死循环,并且1.8之后加入红黑树对性能有提升。
红黑树的时间复杂度 O(logn);链表的时间复杂度 O(n),当链表过长时,红黑树能大大提高查询性能。
答案:
HashMap 1.7 中扩容时,因为采用的是头插法,所以会可能会有循环链表产生,导致数据有问题,在 1.8 版本已修复,改为了尾插法 在任意版本的 HashMap 中,如果在插入数据时多个线程命中了同一个槽,可能会有数据覆盖的情况发生,导致线程不安全。
答案: ConcurrentHashmap在JDK1.7和1.8的版本改动比较大。
答案:
1、Arraylist
2、LinkedList
答案: java ReentrantLock 意为可重入锁,其底层使用 AQS 实现。有两种模式,一种是公平锁,一种是非公平锁。
公平锁:
非公平锁:
答案:
答案:
ThreadLocal有一个静态内部类ThreadLocalMap,ThreadLocalMap又包含了一个Entry数组,Entry本身是一个弱引用,他的key是指向ThreadLocal的弱引用,Entry具备保存key -- value键值对的能力。 在使用完之后调用remove方法删除Entry对象,避免出现内存泄露。
答案:
答案:
答案:
答案: New、Runnable、Running、Blocked、Waiting、Timed Waiting、Terminated
答案:
lock锁机制
Lock lock = new ReentrantLock();
lock. lock();
try {
System. out. println("获得锁");
} catch (Exception e) {
} finally {
System. out. println("释放锁");
lock. unlock();
}
答案:
1、同步阻塞IO。当 应用B 发起读取数据申请时,如果内核数据没有准备好,应用B会一直处于等待数据状态,直到内核把数据准备好了交给应用B才结束。
2、非阻塞IO。当应用B发起读取数据申请时,如果内核数据没有准备好会即刻告诉应用B,不会让B在这里等待。
3、IO复用模型。进程通过将一个或多个fd传递给select,阻塞在select操作上,select帮我们侦测多个fd是否准备就绪,当有fd准备就绪时,select返回数据可读状态,应用程序再调用recvfrom读取数据。
4、信号IO。信号驱动IO不是用循环请求询问的方式去监控数据就绪状态,而是在调用sigaction时候建立一个SIGIO的信号联系,当内核数据准备好之后再通过SIGIO信号通知线程数据准备好后的可读状态,当线程收到可读状态的信号后,此时再向内核发起recvfrom读取数据的请求,因为信号驱动IO的模型下应用线程在发出信号监控后即可返回,不会阻塞,所以这样的方式下,一个应用线程也可以同时监控多个fd。
5、异步IO。解决了应用程序需要先后查看数据是否就绪、发送接收数据请求两个阶段的模式,在异步IO的模式下,只需要向内核发送一次请求就可以完成状态查询和数据拷贝的所有操作。
答案:
如果数据没有就绪,在查看数据是否就绪的这个阶段是一直等待?还是直接返回一个标志信息。
答案:
<img src="https://offercome.cn/images/article/interview/java/java-basic-7.jpg" width="800px">
JMM 就是 Java内存模型(java memory model)。因为在不同的硬件生产商和不同的操作系统下,内存的访问有一定的差异,所以会造成相同的代码运行在不同的系统上会出现各种问题。所以java内存模型(JMM)屏蔽掉各种硬件和操作系统的内存访问差异,以实现让java程序在各种平台下都能达到一致的并发效果。
Java内存模型规定所有的变量都存储在主内存中,包括实例变量,静态变量,但是不包括局部变量和方法参数。每个线程都有自己的工作内存,线程的工作内存保存了该线程用到的变量和主内存的副本拷贝,线程对变量的操作都在工作内存中进行。线程不能直接读写主内存中的变量。
每个线程的工作内存都是独立的,线程操作数据只能在工作内存中进行,然后刷回到主存。这是 Java 内存模型定义的线程基本工作方式。
答案:
// 懒汉式
public class Singleton {
// 延迟加载保证多线程安全
Private volatile static Singleton singleton;
private Singleton(){}
public static Singleton getInstance(){
if(singleton == null){
synchronized(Singleton.class){
if(singleton == null){
singleton = new Singleton();
}
}
}
return singleton;
}
}
答案:
答案:
1、对象头: 对象头又分为 MarkWord 和 **Class Pointer ** 两部分。
**2、Length **: 只在数组对象中存在,用来记录数组的长度,占用 4 字节
3、Instance data: 对象实际数据,对象实际数据包括了对象的所有成员变量,其大小由各个成员变量的大小决定。(这里不包括静态成员变量,因为其是在方法区维护的)
4、Padding: Java 对象占用空间是 8 字节对齐的,即所有 Java 对象占用 bytes 数必须是 8 的倍数,因为当我们从磁盘中取一个数据时,不会说我想取一个字节就是一个字节,都是按照一块儿一块儿来取的,这一块大小是 8 个字节,所以为了完整,padding 的作用就是补充字节,保证对象是 8 字节的整数倍