面试集合

java基础

==与equals的区别

== : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象。(基本数据类型比较的是值,引用数据类型比较的是内存地址)

equals() : 它的作用也是判断两个对象是否相等。但它一般有两种使用情况:

  • 情况 1:类没有覆盖 equals()方法。则通过 equals()比较该类的两个对象时,等价于通过“==”比较这两个对象。
  • 情况 2:类覆盖了 equals()方法。一般,我们都覆盖 equals()方法来两个对象的内容相等;若它们的内容相等,则返回 true(即,认为这两个对象相等)。

为什么重写 equals 时必须重写 hashCode 方法?

如果两个对象相等,则 hashcode 一定也是相同的。两个对象相等,对两个对象分别调用 equals 方法都返回 true。但是,两个对象有相同的 hashcode 值,它们也不一定是相等的 。因此,equals 方法被覆盖过,则 hashCode 方法也必须被覆盖。

hashCode()的默认行为是对堆上的对象产生独特值。如果没有重写 hashCode(),则该 class 的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)

final 在 Java 中有什么作用

  • final 修饰的类叫最终类,该类不能被继承。
  • final 修饰的方法不能被重写。
  • final 修饰的变量叫常量,常量必须初始化,初始化之后值就不能被修改。

Java 中操作字符串都有哪些类?它们之间有什么区别?

操作字符串的类有:String、StringBuffer、StringBuilder。

String 和 StringBuffer、StringBuilder 的区别在于 String声明的是不可变的对象,每次操作都会生成新的 String 对象,然后将指针指向新的 String 对象,而StringBuffer、StringBuilder 可以在原有对象的基础上进行操作,所以在经常改变字符串内容的情况下最好不要使用String。

StringBuffer 和 StringBuilder 最大的区别在于,StringBuffer 是线程安全的,而StringBuilder 是非线程安全的,但StringBuilder 的性能却高于 StringBuffer,所以在单线程环境下推荐使用StringBuilder,多线程环境下推荐使用 StringBuffer。

String str="i"与 String str=new String("i")一样吗?

不一样,因为内存的分配方式不一样。String str="i"的方式,Java 虚拟机会将其分配到常量池中;而 String str=new String("i") 则会被分到堆内存中。

String为什么是不可变的

String类是利用了final修饰的char类型数组存储字符

如何将字符串反转

使用 StringBuilder 或者 stringBuffer 的 reverse() 方法。

示例代码:

Copy// StringBuffer reverse
StringBuffer stringBuffer = new StringBuffer();
stringBuffer. append("abcdefg");
System. out. println(stringBuffer. reverse()); // gfedcba
// StringBuilder reverse
StringBuilder stringBuilder = new StringBuilder();
stringBuilder. append("abcdefg");
System. out. println(stringBuilder. reverse()); // gfedcba

String 类的常用方法都有那些?

indexOf():返回指定字符的索引。
charAt():返回指定索引处的字符。
replace():字符串替换。
trim():去除字符串两端空白。
split():分割字符串,返回一个分割后的字符串数组。
getBytes():返回字符串的 byte 类型数组。
length():返回字符串长度。
toLowerCase():将字符串转成小写字母。
toUpperCase():将字符串转成大写字符。
substring():截取字符串。
equals():字符串比较。

常用方法

int length():返回字符串的长度:return value.length

char charAt(int index):返回某索引处的字符return value[index]

boolean  isEmpty():判断是否是空字符串:return value.length == 0

String toLowerCase():使用默认语言环境,将 String 中的所字符转换为小写

String toUpperCase():使用默认语言环境,将 String 中的所字符转换为大写

String trim():返回字符串的副本,忽略前导空白和尾部空白

boolean equals(Object obj):比较字符串的内容是否相同

boolean equalsIgnoreCase(String anotherString):与equals方法类似,忽略大小写

String concat(String str):将指定字符串连接到此字符串的结尾。 等价于用“+”

int compareTo(String anotherString):比较两个字符串的大小

String substring(int beginIndex):返回一个新的字符串,它是此字符串的从beginIndex开始截取到最后的一个子字符串。

String substring(int beginIndex, int endIndex) :返回一个新字符串,它是此字符串从beginIndex开始截取到endIndex(不包含)的一个子字符串。

boolean endsWith(String suffix):测试此字符串是否以指定的后缀结束

boolean startsWith(String prefix):测试此字符串是否以指定的前缀开始

boolean startsWith(String prefix, int toffset):测试此字符串从指定索引开始的子字符串是否以指定前缀开始

boolean contains(CharSequence s):当且仅当此字符串包含指定的 char 值序列时,返回 true

int indexOf(String str):返回指定子字符串在此字符串中第一次出现处的索引

int indexOf(String str, int fromIndex):返回指定子字符串在此字符串中第一次出现处的索引,从指定的索引开始

int lastIndexOf(String str):返回指定子字符串在此字符串中最右边出现处的索引

int lastIndexOf(String str, int fromIndex):返回指定子字符串在此字符串中最后一次出现处的索引,从指定的索引开始反向搜索

注:indexOf和lastIndexOf方法如果未找到都是返回-1

### *替换:*

String replace(char oldChar, char newChar):返回一个新的字符串,它是通过用 newChar 替换此字符串中出现的所 oldChar 得到的。

String replace(CharSequence target, CharSequence replacement):使用指定的字面值替换序列替换此字符串所匹配字面值目标序列的子字符串。

String replaceAll(String regex, String replacement):使用给定的 replacement替换此字符串所匹配给定的正则表达式的子字符串。

StringreplaceFirst(String regex, String replacement):使用给定的 replacement替换此字符串匹配给定的正则表达式的第一个子字符串。

### *匹配:*

boolean matches(String regex):告知此字符串是否匹配给定的正则表达式。

### *切片:*

String[] split(String regex):根据给定正则表达式的匹配拆分此字符串。

String[] split(String regex, int limit):根据匹配给定的正则表达式来拆分此字符串,最多不超过limit个,如果超过了,剩下的全部都放到最后一个元素中。

String实例化的不同方式

  1. 通过字面量定义的方式 数据是在方法区中的字符串常量池中
  2. 通过new+构造器的方式 是数据在堆空间中开辟空间以后对应的地址值
String s1 = "javaEE";
String s2 = "javaEE";
String s3 = new String("javaEE");
String s4 = new String("javaEE");

System.out.println(s1== s2);//true

System.out.println(s1== s3);//false

System.out.println(s1== s4);//false

System.out.println(s3== s4);//false

面试题:String s = new String("abc");方式创建对象,在内存中创建了几个对象
两个,一个是堆空间中new结构,另一个是char[]对应的常量池中的数据“abc”

字符串的拼接

  1. 常量与常量的拼接结果在常量池,且常量池中不会出现相同内容的常量
  2. 只要有一个是变量,结果就在堆中
  3. 如果拼接结果调用intern()方法,返回值就在常量池中
String s1 = "javaEE"; //常量池
String s2 = "hadoop"; //常量池

String s3 = "javaEEhadoop"; 
String s4 = "javaEE" + "hadoop"; //常量池
String s5 = s1 + "hadoop"; //堆
String s6 = "javaEE" + s2;
String s7 = s1 + s2;

System.out.println(s3 == s4);//true
System.out.println(s3 == s5);//false
System.out.println(s3 == s6);//false
System.out.println(s3 == s7);//false
System.out.println(s5 == s6);//false
System.out.println(s5 == s7);//false
System.out.println(s6 == s7);//false

String s8 = s6.intern();//返回值得到的s8使用的常量值中已经存在的“javaEEhadoop”
System.out.println(s3 == s8);//true
****************************
String s1 = "javaEEhadoop";
String s2 = "javaEE";
String s3 = s2 + "hadoop";
System.out.println(s1 == s3);//false

final String s4 = "javaEE";//s4:常量
String s5 = s4 + "hadoop";
System.out.println(s1 == s5);//true

String和其他类型结构的转换

基本类型,包装类型之间的转换

String —>基本数据类型,包装类:调用包装类的静态方法:parseXxx(str)

基本数据类型,包装类—>String: 调用String的ValueOf(xxx);

@Test
    public void test1(){
        String str1 = "123";
//        int num = (int)str1;//错误的
        int num = Integer.parseInt(str1);

        String str2 = String.valueOf(num);//"123"
        String str3 = num + "";

        System.out.println(str1 == str3);
    }

与字符数组之间的转换

String -->char[] 调用String的toCharArray()

char[] --> String 调用String的构造器

@Test
public void test2(){
    String str1 = "abc123";  

    char[] charArray = str1.toCharArray();
    for (int i = 0; i < charArray.length; i++) {
        System.out.println(charArray[i]);
    }

    char[] arr = new char[]{'h','e','l','l','o'};
    String str2 = new String(arr);
    System.out.println(str2);
}

与字节数组之间的转换

编码:String—>byte[]:调用String的getBytes()

解码:byte[]—>String: 调用String的构造器

说明:解码时,要求解码使用的字符集必须与编码使用的字符集一致,否则会出现乱码

@Test
public void test3() throws UnsupportedEncodingException {
    String str1 = "abc123中国";
    byte[] bytes = str1.getBytes();//使用默认的字符集,进行编码。
    System.out.println(Arrays.toString(bytes));

    byte[] gbks = str1.getBytes("gbk");//使用gbk字符集进行编码。
    System.out.println(Arrays.toString(gbks));

    System.out.println("******************");

    String str2 = new String(bytes);//使用默认的字符集,进行解码。
    System.out.println(str2);

    String str3 = new String(gbks);
    System.out.println(str3);//出现乱码。原因:编码集和解码集不一致!

    String str4 = new String(gbks, "gbk");
    System.out.println(str4);//没出现乱码。原因:编码集和解码集一致!

}

与StringBuffer、StringBuilder之间的转换

String—>StringBuffer/StringBuilder

调用StringBuffer/StringBuilder的构造器

StringBuffer/StringBuilder—>

String调用String的构造器 或 StringBufffer/StringBuilder的toString()

StringBuffer,StringBuilder

  1. String: 不可变的字符串,底层是char[]
  2. StringBuffer: 可变的字符序列,底层安全,效率低,底层使用char[]存储
  3. StringBuilder:可变的字符序列,线性不安全,效率高,底层使用char[]存储

StringBuffer 和StringBuilder内存解析

String str = new String(); //char[] value = new char[0]
String str1 = new String("abc"); //char[] value = new char[]{'a','b','c'};

StringBuffer sb1 = new StringBuffer();//char[] value = new char[16];底层创建了一个长度是16的数组。
System.out.println(sb1.length());//
sb1.append('a');//value[0] = 'a';
sb1.append('b');//value[1] = 'b';

StringBuffer sb2 = new StringBuffer("abc");//char[] value = new char["abc".length() + 16];

问题1. System.out.println(sb2.length()); //3

问题2. 扩容问题:如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组。

默认情况下,扩容为原来容量的2倍 + 2,同时将原数组中的元素复制到新的数组中。

StringBuilder也是这样扩容的

指导意义:开发中建议大家使用:带参数的StringBuffer(int capacity) 或 StringBuilder(int capacity)

对比String、StringBuffer、StringBuilder三者的执行效率

从高到底排列:StringBuilder>StringBuffer>String

StringBuffer、StringBuilder的常用方法

增:append(xxx)
删:delete(int start,int end)
改:setCharAt(int n ,char ch) / replace(int start, int end, String str)
查:charAt(int n )
插:insert(int offset, xxx)
长度:length();
遍历:for() + charAt() / toString()

重载和重写的区别

重载

发生在一个类中,具有相同的方法名,参数类型不同,个数不同,顺序不同,方法返回值和访问修饰符可以不同

重写

重写是子类对父类的允许访问的方法的实现过程进行重新编写,发生在子类中,方法名,参数列表必须相同,返回值范围小于父类,抛出的异常范围小于父类,访问修饰符大于父类,另外,如果父类的访问修饰符为private,则子类就不能重写该方法,也就是说方法提供的行为发生了改变,方法的外貌并没有改变

封装 继承 多态

封装

封装把一个对象的属性私有化,同时提供一些可以被外界访问的属性和方法,如果属性不想被外界访问,我们大可不必提供方法给外界访问。但是一个类没有提供给外界访问的方法,那么这个类也就没有什么意义了

继承

继承是使用已存在类的定义作为基础建立新类的技术,新类的定义可以添加新的数据或新的功能,也可以用父类的功能,但不能选择性的继承父类。通过继承我们可以非常方便的复用之前的代码

关于继承的三点:

  1. 子类拥有父类的所有属性和方法(包括私有属性和方法),但是父类中的私有属性和方法是子类无法访问的,只是拥有。
  2. 子类可以拥有自己的属性和方法,即子类可以对父类进行扩展
  3. 子类可以用自己的方式实现父类的方法

多态

所谓多态就是程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行时确定的,即一个引用变量到底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类实现的方法,必须在程序运行时才能够决定

在java中有两种方式可以实现多态:继承(多个子类对同一个方法的重写),接口(实现接口并覆盖接口中同一方法)

抽象类必须要有抽象方法吗

不需要,抽象类不一定非要有抽象方法。

普通类和抽象类有哪些区别?

  • 普通类不能包含抽象方法,抽象类可以包含抽象方法
  • 抽象类不能直接实例化,普通类可以直接实例化

抽象类能使用final修饰吗?

不能,定义抽象类就是让其他类继承的,如果定义为final该类就不能被继承,这样彼此就会产生矛盾,所以不能修饰抽象类

接口和抽象类的区别

  • 实现:抽象类的子类使用extends类继承;接口必须使用implement来实现接口
  • 构造函数:抽象类可以有构造函数,接口不能
  • 实现数量:类可以实现很多接口,但只能继承一个抽象类
  • 访问修饰符:接口中的方法默认使用public修饰,抽象类中的方法可以是任意访问修饰符

java序列化中如果有些字段不想被进行序列化,怎么办

对于不想被序列化的变量,可以使用transient关键字修饰

transient关键字的作用是:阻止实例中那些用此关键字的变量序列化,但对象被反序列化时,被transient修饰的变量值不会被持久化和恢复,transient只能修饰变量,不能修饰类和方法。

获取用键盘输入常用的两种方法

  1. 使用scanner
Scanner input = new Scanner(System.in);
String s = input.nextLine();
input.close();
  1. 通过BuffferdReader

    BufferedReader input = new BufferedReader(new InputStreamReader(System.in));
    String s = input.readLine();
    

java中IO流分为几种?

  • 按功能来分:输入流(input),输出流(output)
  • 按类型来分:字节流和字符流
  • 字节流和字符流的区别:字节流按8位传输以字节为单位输入输出数据,字符流按16位传输以字符为单位输入输出数据

BIO,NIO,AIO有什么区别?

  • BIO:Block IO同步阻塞式IO,就是我们平常使用的传统IO,它的特点是模式简单使用方便,并发处理能力低。
  • NIO:Non IO同步非阻塞IO,是传统IO的升级,客户端和服务端通过Channel(通道)通讯,实现多路复用
  • AIO:Asynchronous IO是NIO的升级,也叫NIO2,实现异步非阻塞IO,异步IO的操作基于事件和回调机制

同步异步

当你同步执行某项任务时,你需要等待其完成才能继续执行其他任务。当你异步执行某些操作时,你可以在完成另一个任务之前继续进行。

  • 同步 :两个同步任务相互依赖,并且一个任务必须以依赖于另一任务的某种方式执行。 比如在A->B事件模型中,你需要先完成 A 才能执行B。 再换句话说,同步调用中被调用者未处理完请求之前,调用不返回,调用者会一直等待结果的返回。
  • 异步: 两个异步的任务完全独立的,一方的执行不需要等待另外一方的执行。再换句话说,异步调用种一调用就返回结果不需要等待结果返回,当结果返回的时候通过回调函数或者其他方式拿着结果再做相关事情,

阻塞和非阻塞

  • 阻塞: 阻塞就是发起一个请求,调用者一直等待请求结果返回,也就是当前线程会被挂起,无法从事其他任务,只有当条件就绪才能继续。
  • 非阻塞: 非阻塞就是发起一个请求,调用者不用一直等着结果返回,可以先去干其他事情。

Files的常用方法都有哪些?

  • Files. exists():检测文件路径是否存在。
  • Files. createFile():创建文件。
  • Files. createDirectory():创建文件夹。
  • Files. delete():删除一个文件或目录。
  • Files. copy():复制文件。
  • Files. move():移动文件。
  • Files. size():查看文件个数。
  • Files. read():读取文件。
  • Files. write():写入文件。

集合

java容器有哪些

Java容器分为Conllection和Map两大类,其下又有很多子类,Conllection下面有List和Set两大类,Map下面有HashMap,TreeMap,ConcurrentHashMap,Hashtable;而List下面又有ArrayList,LinkedList,Vector,Stack;Set下面又分为HashSet,LinkedHashSet,TreeSet.

Collection和Collections有什么区别

  • Collection 是一个集合接口,它提供了对集合对象进行基本操作的通用接口方法,所有集合都是它的子类,比如 List、Set 等。
  • Collections 是一个包装类,包含了很多静态方法,不能被实例化,就像一个工具类,比如提供的排序方法: Collections. sort(list)。

List,Set,Map之间的区别是什么?

  • List元素有序,允许元素重复
  • Set元素无序,不允许元素重复
  • Map中key值唯一但是value可以重复,Hashmap元素无序,要想有序的话可以用TreeMap

HashMap和HashTable有什么区别?

  • 存储上:HashMap运行key和value为null,而Hashtable不允许
  • 线程安全上:HashTable是线程安全的,而HashMap是非线程安全的。
  • 在 Hashtable 的类注释可以看到,Hashtable 是保留类不建议使用,推荐在单线程环境下使用 HashMap 替代,如果需要多线程使用则用 ConcurrentHashMap 替代。

如何决定使用 HashMap 还是 TreeMap?

对于在 Map 中插入、删除、定位一个元素这类操作,HashMap 是最好的选择,因为相对而言 HashMap 的插入会更快,但如果你要对一个 key 集合进行有序的遍历,那 TreeMap 是更好的选择。

说一下 HashMap 的实现原理

HashMap 基于 Hash 算法实现的,我们通过 put(key,value)存储,get(key)来获取。当传入 key
时,HashMap 会根据 key. hashCode() 计算出 hash 值,根据 hash 值将 value 保存在 bucket
里。当计算出的 hash 值相同时,我们称之为 hash 冲突,HashMap 的做法是用链表和红黑树存储相同 hash 值的 value。当hash 冲突的个数比较少时,使用链表否则使用红黑树。

说一下 HashSet 的实现原理

HashSet 是基于 HashMap 实现的,HashSet 底层使用 HashMap 来保存所有元素,因此 HashSet的实现比较简单,相关 HashSet 的操作,基本上都是直接调用底层 HashMap 的相关方法来完成,HashSet 不允许重复的值

ArrayList 和 LinkedList 的区别是什么

  • 数据结构实现:ArrayList 是动态数组的数据结构实现,而 LinkedList 是双向链表的数据结构实现。
  • 随机访问效率:ArrayList 比 LinkedList 在随机访问的时候效率要高,因为 LinkedList 是线性的数据存储方式,所以需要移动指针从前往后依次查找。
  • 增加和删除效率:在非首尾的增加和删除操作,LinkedList 要比 ArrayList 效率要高,因为 ArrayList 增删操作要影响数组内的其他数据的下标。

综合来说,在需要频繁读取集合中的元素时,更推荐使用 ArrayList,而在插入和删除操作较多时,更推荐使用 LinkedList。

如何实现数组和List之间的转换

  • 数组转List:使用Arrays.asList(array)进行转换
  • List转数组:使用List自带的toArray()方法

ArrayList 和 Vector 的区别是什么?

  • 线程安全:Vector 使用了 Synchronized 来实现线程同步,是线程安全的,而 ArrayList 是非线程安全的。
  • 性能:ArrayList 在性能方面要优于 Vector。
  • 扩容:ArrayList 和 Vector 都会根据实际的需要动态的调整容量,只不过在 Vector 扩容每次会增加 1 倍,而 ArrayList 只会增加 0.5倍

Array 和 ArrayList 有何区别

  • Array 可以存储基本数据类型和对象,ArrayList 只能存储对象。
  • Array 是指定固定大小的,而 ArrayList 大小是自动扩展的。
  • Array 内置方法没有 ArrayList 多,比如 addAll、removeAll、iteration 等方法只有 ArrayList 有。

在 Queue 中 poll()和 remove()有什么区别

  • 相同点:都是返回第一个元素,并在队列中删除返回的对象。
  • 不同点:如果没有元素 poll()会返回 null,而 remove()会直接抛出 NoSuchElementException 异常。
CopyQueue<String> queue = new LinkedList<String>();
queue. offer("string"); // add
System. out. println(queue. poll());
System. out. println(queue. remove());
System. out. println(queue. size());

哪些集合类是线程安全的

Vector、Hashtable、Stack 都是线程安全的,而像 HashMap 则是非线程安全的,不过在 JDK 1.5 之后随着
Java. util. concurrent 并发包的出现,它们也有了自己对应的线程安全类,比如 HashMap 对应的线程安全类就是ConcurrentHashMap。

迭代器 Iterator 是什么

Iterator 接口提供遍历任何 Collection 的接口。我们可以从一个 Collection 中使用迭代器方法来获取迭代器实例。迭代器取代了 Java 集合框架中的 Enumeration,迭代器允许调用者在迭代过程中移除元素。

Iterator怎么使用?有什么特点

Iterator 使用代码如下:

CopyList<String> list = new ArrayList<>();
Iterator<String> it = list. iterator();
while(it. hasNext()){
  String obj = it. next();
  System. out. println(obj);
}

Iterator 的特点是更加安全,因为它可以确保,在当前遍历的集合元素被更改的时候,就会抛出 ConcurrentModificationException 异常。

Itertor和ListIterator有什么区别

  • Iterator 可以遍历 Set 和 List 集合,而 ListIterator 只能遍历 List。
  • Iterator 只能单向遍历,而 ListIterator 可以双向遍历(向前/后遍历)。
  • ListIterator 从 Iterator 接口继承,然后添加了一些额外的功能,比如添加一个元素、替换一个元素、获取前面或后面元素的索引位置。

怎么确保一个集合不能被修改

可以使用Collections.unmodifiableCollection(Collection c)方法来创建一个只读集合,这样改变集合的任何操作都会抛出java.lang.UnsupportedOperationException异常

CopyList<String> list = new ArrayList<>();
list. add("x");
Collection<String> clist = Collections. unmodifiableCollection(list);
clist. add("y"); // 运行时此行报错
System. out. println(list. size());

多线程

并行并发有什么区别

并行:多个处理器或多核处理器同时处理多个任务

并发:多个任务在同一个cpu核上,按细分的时间片轮流(交替)执行,从逻辑上来看那些任务是同时执行。

线程和进程有什么区别

一个程序下至少有一个进程,一个进程至少有一个线程,一个进程下也可以有多个线程来增加程序的执行速度

守护进程是什么

守护进程是后台运行的一个特殊进程,它独立于控制终端周期性的执行某种任务或等待处理某些发生的事情。在java中垃圾回收线程就是特殊的守护进程

创建线程有哪几种方式

创建线程有3种方式

  • 继承Thread重写run方法
  • 实现Runnable接口
  • 实现Callable接口
  • 线程池

说一下 runnable 和 callable 有什么区别?

runnable 没有返回值,callable 可以拿到有返回值,callable 可以看作是 runnable 的补充

callable的call方法可以抛出异常,而Runnable的run方法不会抛出异常。

线程有哪些状态

  • new 尚未启动
  • runnable正在执行中
  • blocked阻塞的(被同步锁或者IO锁阻塞)
  • waiting 永久等待状态
  • timed_waiting等待指定的时间重新被唤醒的状态
  • terminated 执行完成

sleep()和wait()有什么区别?

  • 类不同:sleep()来自Thread,wait()来自Object
  • 释放锁:sleep()不释放锁,wait()释放锁
  • 用法不同:sleep()时间到了会自动恢复;wait()可以使用notify()/notifyAll()直接唤醒

notify()和 notifyAll()有什么区别?

notifyAll()会唤醒所有的线程,notify()之后唤醒一个线程。notifyAll()调用后,会将全部线程由等待池移到锁池,然后参与锁的竞争,竞争成功则继续执行,如果不成功则留在锁池等待锁被释放后再次参与竞争。而
notify()只会唤醒一个线程,具体唤醒哪一个线程由虚拟机控制。

线程的 run() 和 start() 有什么区别

start() 方法用于启动线程,run() 方法用于执行线程的运行时代码。run() 可以重复调用,而 start() 只能调用一次。

创建线程池有哪几种方式?

线程池创建有七种方式,最核心的是最后一种:

  • newSingleThreadExecutor():它的特点在于工作线程数目被限制为 1,操作一个无界的工作队列,所以它保证了所有任务的都是被顺序执行,最多会有一个任务处于活动状态,并且不允许使用者改动线程池实例,因此可以避免其改变线程数目;
  • newCachedThreadPool():它是一种用来处理大量短时间工作任务的线程池,具有几个鲜明特点:它会试图缓存线程并重用,当无缓存线程可用时,就会创建新的工作线程;如果线程闲置的时间超过 60 秒,则被终止并移出缓存;长时间闲置时,这种线程池,不会消耗什么资源。其内部使用 SynchronousQueue 作为工作队列;
  • newFixedThreadPool(intnThreads):重用指定数目(nThreads)的线程,其背后使用的是无界的工作队列,任何时候最多有 nThreads个工作线程是活动的。这意味着,如果任务数量超过了活动队列数目,将在工作队列中等待空闲线程出现;如果有工作线程退出,将会有新的工作线程被创建,以补足指定的数目 nThreads;
  • newSingleThreadScheduledExecutor():创建单线程池,返回 ScheduledExecutorService,可以进行定时或周期性的工作调度;
  • newScheduledThreadPool(intcorePoolSize):和newSingleThreadScheduledExecutor()类似,创建的是个
    ScheduledExecutorService,可以进行定时或周期性的工作调度,区别在于单一工作线程还是多个工作线程;
  • newWorkStealingPool(int parallelism):这是一个经常被人忽略的线程池,Java 8 才加入这个创建方法,其内部会构建ForkJoinPool,利用Work-Stealing算法,并行地处理任务,不保证处理顺序;
  • ThreadPoolExecutor():是最原始的线程池创建,上面1-3创建方式都是对ThreadPoolExecutor的封装。

线程池都有那些状态

  • RUNNING:这是最正常的状态,接受新的任务,处理等待队列中的任务。
  • SHUTDOWN:不接受新的任务提交,但是会继续处理等待队列中的任务。
  • STOP:不接受新的任务提交,不再处理等待队列中的任务,中断正在执行任务的线程。
  • TIDYING:所有的任务都销毁了,workCount 为 0,线程池的状态在转换为 TIDYING 状态时,会执行钩子方法 terminated()。
  • TERMINATED:terminated()方法结束后,线程池的状态就会变成这个。

线程池中submint()和execute()方法有什么区别

  • execute():只能执行 Runnable 类型的任务。
  • submit():可以执行 Runnable 和 Callable 类型的任务。

Callable类型的任务可以获取执行的返回值,而Runnable执行无返回值

在java程序中怎么保证多线程的运行安全

  • 方法一:使用安全类,比如 Java. util. concurrent 下的类。
  • 方法二:使用自动锁 synchronized。
  • 方法三:使用手动锁 Lock。

多线程中synchronized锁升级的原理是什么?

synchronized锁升级的原理:在锁对象的对象头里面有一个threadid字段,第一次访问的时候threadid为空,jvm让其持有偏向锁,并将threadid设置为其线程id,再次进入的时候会先判断threadid是否与其线程id一致,如果一致则可以直接使用此对象,如果不一致,则升级偏向锁为轻量级锁,通过自旋循环一定次数来获取锁,执行一定次数之后,如果还没有正常获取到要使用的对象,此时就会把锁从轻量级锁升级为重量级,此过程就构成了synchronized锁的升级。

锁的升级的目的:锁升级是为了减低了锁带来的性能消耗。在 Java 6 之后优化 synchronized 的实现方式,使用了偏向锁升级为轻量级锁再升级到重量级锁的方式,从而减低了锁带来的性能消耗。

什么是死锁

当线程 A 持有独占锁a,并尝试去获取独占锁 b 的同时,线程 B 持有独占锁 b,并尝试获取独占锁 a 的情况下,就会发生 AB 两个线程由于互相持有对方需要的锁,而发生的阻塞现象,我们称为死锁。

怎么防止死锁?

  • 尽量使用tryLock(long timeout,TimeUnit unit)的方法(ReentrantLock、ReentrantReadWriteLock),设置超时时间,超时可以退出防止死锁。
  • 尽量使用 Java. util. concurrent 并发类代替自己手写锁。
  • 尽量降低锁的使用粒度,尽量不要几个功能用同一把锁。
  • 尽量减少同步的代码块。

ThreadLocal 是什么?有哪些使用场景?

ThreadLocal 为每个使用该变量的线程提供独立的变量副本,所以每一个线程都可以独立地改变自己的副本,而不会影响其它线程所对应的副本。

ThreadLocal 的经典使用场景是数据库连接和 session 管理等。

说一下 synchronized 底层实现原理?

synchronized 是由一对 monitorenter/monitorexit 指令实现的,monitor对象是同步的基本实现单元。在 Java 6 之前,monitor的实现完全是依靠操作系统内部的互斥锁,因为需要进行用户态到内核态的切换,所以同步操作是一个无差别的重量级操作,性能也很低。但在 Java 6的时候,Java 虚拟机 对此进行了大刀阔斧地改进,提供了三种不同的 monitor 实现,也就是常说的三种不同的锁:偏向锁(Biased Locking)、轻量级锁和重量级锁,大大改进了其性能

synchronized 和 volatile 的区别是什么?

  • volatile 是变量修饰符;synchronized 是修饰类、方法、代码段。
  • volatile 仅能实现变量的修改可见性,不能保证原子性;而 synchronized 则可以保证变量的修改可见性和原子性。
  • volatile 不会造成线程的阻塞;synchronized 可能会造成线程的阻塞。

synchronized 和 Lock 有什么区别

  • synchronized 可以给类、方法、代码块加锁;而 lock 只能给代码块加锁。
  • synchronized 不需要手动获取锁和释放锁,使用简单,发生异常会自动释放锁,不会造成死锁;而 lock 需要自己加锁和释放锁,如果使用不当没有 unLock()去释放锁就会造成死锁。
  • 通过 Lock 可以知道有没有成功获取锁,而 synchronized 却无法办到。

synchronized 和 ReentrantLock 区别是什么

synchronized 早期的实现比较低效,对比 ReentrantLock,大多数场景性能都相差较大,但是在 Java 6 中对 synchronized 进行了非常多的改进。

主要区别如下:

  • ReentrantLock 使用起来比较灵活,但是必须有释放锁的配合动作;
  • ReentrantLock 必须手动获取与释放锁,而 synchronized 不需要手动释放和开启锁;
  • ReentrantLock 只适用于代码块锁,而 synchronized 可用于修饰方法、代码块等。
  • volatile 标记的变量不会被编译器优化;synchronized 标记的变量可以被编译器优化

说一下 atomic 的原理?

atomic 主要利用 CAS (Compare And Wwap) 和 volatile 和 native 方法来保证原子操作,从而避免 synchronized 的高开销,执行效率大为提升。

反射

什么是反射?

反射是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为 Java 语言的反射机制。

动态编译和静态编译

静态编译:在编译时确定类型,绑定对象

动态编译:在运行时确定类型,绑定对象

什么是 Java 序列化?什么情况下需要序列化

Java 序列化是为了保存各种对象在内存中的状态,并且可以把保存的对象状态再读出来。

以下情况需要使用 Java 序列化:

  • 想把的内存中的对象状态保存到一个文件中或者数据库中时候;
  • 想用套接字在网络上传送对象的时候;
  • 想通过RMI(远程方法调用)传输对象的时候。

动态代理是什么?有哪些应用

动态代理是运行时动态生成代理类。

动态代理的应用有 spring aop、hibernate 数据查询、测试框架的后端 mock、rpc,Java注解对象获取等。

怎么实现动态代理

JDK原生动态代理和cglib动态代理

jdk动态代理是基于接口实现的

cglib动态代理是基于继承当前类的子类实现的

对象克隆

为什么使用克隆

克隆的对象可能包括一些修改过的属性,而new出来的对象的属性都是初始化时候的值,当需要一个新对象来保存当前对象的状态就靠克隆方法了

如何实现对象克隆

实现cloneable接口,并重写Object类clone()方法

实现Serializable接口,通过对象的序列化和反序列化实现克隆,可以实现真正的深度克隆

深拷贝和浅拷贝的区别

浅拷贝:当对象被复制时只复制它本身和其中包含的值类型的成员变量,而引用类型的成员对象并没有复制

深克隆:除了对象本身被复制外,对象所包含的所有成员变量也将复制。

Linux

linux怎么关闭进程

通常用ps查看进程的pid,用kill命令终止进程。

ps命令用于查看当前正在运行的程序

ps -ef|grep java //表示查看所有进程里cmd是java的进程信息
ps -aux|grep java  //-aux 显示所有状态
kill -9 [pid] //kill命令用于终止进程 -9表示强迫进程立即停止

Java Web

什么是jsp

jsp本质上就是一个servlet,它是servlet的一种特殊形式,每个jsp页面都是一个servlet实例

JSP 和 servlet 有什么区别

JSP 是 servlet 技术的扩展,本质上就是 servlet 的简易方式。servlet 和 JSP最主要的不同点在于,servlet 的应用逻辑是在 Java 文件中,并且完全从表示层中的 html 里分离开来,而 JSP 的情况是 Java 和 html 可以组合成一个扩展名为 JSP 的文件。JSP 侧重于视图,servlet 主要用于控制逻辑。

谈一下Servlet的生命周期

Servlet运行在Servlet容器中,其生命周期由容器来管理,Servlet的生命周期通过javax.servlet.Servlet接口中的init(),service和destroy()方法来表示。Servlet的生命周期包含了下面4个阶段

  1. 加载和实例化
  2. 初始化
  3. 请求处理
  4. 服务终止

Servlet是线程安全的吗

  • Servlet是单例的,对于所有请求都使用一个实例
  • 如果有全局变量被多线程使用的时候,就会出现线程安全问题
  • 解决这个问题有三种方案:singThreadModel synchronized,不使用全局变量

tomcat是如何创建servlet类实例,用到了什么原理

当容器启动时,会读取在webapps目录下所有的web应用的web.xml文件,然后对xml文件进行解析,并读取servlet注册信息。然后,将每个应用中注册的servlet都进行加载,并通过反射的方式实例化(有时候也是在第一次请求实例化)

在servlet注册时加上1如果为正数,则在一开始就实例化,如果不写或为负数,则第一次请求实例化。

web.xml可以配置哪些内容

监听器,过滤器,相关参数,会话超时时间,安全验证方式,错误页面

Tomcat的省缺端口是多少,怎么修改

tomcat的主目录,conf/serve.xml文件下修改,把8080改为其他的

jsp四大作用范围

  • application
  • session
  • request
  • page

JSP有哪些内置对象?作用分别是什么

JSP 有 9 大内置对象:

  • request:封装客户端的请求,其中包含来自 get 或 post 请求的参数;
  • response:封装服务器对客户端的响应;
  • pageContext:通过该对象可以获取其他对象;
  • session:封装用户会话的对象;
  • application:封装服务器运行环境的对象;
  • out:输出服务器响应的输出流对象;
  • config:Web 应用的配置对象;
  • page:JSP 页面本身(相当于 Java 程序中的 this);
  • exception:封装页面抛出异常的对象。

JSP的4种作用域

  • page:代表与一个页面相关的对象和属性。
  • request:代表与客户端发出的一个请求相关的对象和属性。一个请求可能跨越多个页面,涉及多个 Web 组件;需要在页面显示的临时数据可以置于此作用域。
  • session:代表与某个用户与服务器建立的一次会话相关的对象和属性。跟某个用户相关的数据应该放在用户自己的 session 中。
  • application:代表与整个 Web 应用程序相关的对象和属性,它实质上是跨越整个 Web 应用程序,包括多个页面、请求和会话的一个全局作用域

get 和 post 请求有哪些区别?

  • 数据携带上,get 传递参数有大小限制,而 post 没有。
  • 请求参数的位置上,get方式,请求参数放在url地址后面,以?进行拼接;post方式,请求参数放在HTTP请求包内
  • 用途上,post参数传输更安全,所以get方式通常获取数据,post方式通常提交数据
  • get方式比post方式要快

froword和redirect有什么区别

  • forword是服务器行为,redirect是客户端行为

  • 地址栏的url显示:forword转发时,浏览器的url不会改变,也就是一次请求,redirect会发生改变,也就是说会发送两次请求

  • 数据共享方面:forword会共享request里的数据,redirect不能共享

  • 效率上:forword比redirect高

session和cookie有什么区别

  • 存储位置不同,seession存储在服务器端,cookie存储在浏览器端
  • 安全性不同:cookie在浏览器存储,可以被伪造和修改
  • 容量和个数的限制:cookie 有容量限制,每个站点下的 cookie 也有个数限制。
  • 存储的多样性:session 可以存储在 Redis 中、数据库中、应用程序中;而 cookie 只能存储在浏览器中。

session的工作原理

session的工作原理是客户端登录完成之后,服务器会创建对应的session,session创建完之后,会把session的id发送给客户端,客户端再存储到浏览器中,这样客户端每次访问服务器时,都会带着sessionid,服务器拿到sessionid之后,在内存找到与之对应的session这样就可以正常工作了。

如果客户端禁止cookie能实现session还能用吗

可以用,session只是依赖cookie存储sessionid,如果cooie被禁用了,可以使用url中添加sessionid的方法保证session的正常使用

springmvc和struts的区别是什么?

  • 拦截级别:struts2 是类级别的拦截;spring mvc 是方法级别的拦截。
  • 数据独立性:spring mvc 的方法之间基本上独立的,独享 request 和 response
    数据,请求数据通过参数获取,处理结果通过 ModelMap 交回给框架,方法之间不共享变量;而 struts2 虽然方法之间也是独立的,但其所有 action 变量是共享的,这不会影响程序运行,却给我们编码和读程序时带来了一定的麻烦。
  • 拦截机制:struts2 有以自己的 interceptor 机制,spring mvc 用的是独立的 aop 方式,这样导致struts2 的配置文件量比 spring mvc 大。
  • 对 ajax 的支持:spring mvc 集成了ajax,所有 ajax 使用很方便,只需要一个注解 @ResponseBody 就可以实现了;而 struts2 一般需要安装插件或者自己写代码才行。

如何避免 SQL 注入?

  • 使用预处理 PreparedStatement。
  • 使用正则表达式过滤掉字符中的特殊字符。

什么是 XSS 攻击,如何避免

XSS 攻击:即跨站脚本攻击,它是 Web 程序中常见的漏洞。原理是攻击者往 Web 页面里插入恶意的脚本代码(css代码、Javascript 代码等),当用户浏览该页面时,嵌入其中的脚本代码会被执行,从而达到恶意攻击用户的目的,如盗取用户cookie、破坏页面结构、重定向到其他网站等。

预防 XSS 的核心是必须对输入的数据做过滤处理。

什么是 CSRF 攻击,如何避免?

CSRF:Cross-Site Request Forgery(中文:跨站请求伪造),可以理解为攻击者盗用了你的身份,以你的名义发送恶意请求,比如:以你名义发送邮件、发消息、购买商品,虚拟货币转账等。

防御手段:

  • 验证请求来源地址;
  • 关键操作添加验证码;
  • 在请求地址添加 token 并验证。

谈一下Ajax

Ajax是一种创建交互式网页应用的网页开发技术

Ajax的优势:

  1. 通过异步的模式,提升了用户的体验
  2. 优化了服务器和浏览器之间的传输,减少了不必要的数据往返,减少了宽带占用
  3. Ajax引擎在客户端运行,承担了一部分原本由服务器承担的压力,从而减少了大用户量下的服务器负载
  4. 可以实现局部刷新,在不更新整个页面的前提下维护数据,提升用户体验度

异常

throw和throws的区别

throw是抛出一个异常

throw是声明可能会抛出一个异常

final ,finally,finalize有什么区别

  • final:是修饰符,如果修饰类,此类不能被继承;如果修饰方法和变量,则表示此方法和此变量不能在被改变,只能使用。
  • finally:是 try{} catch{} finally{} 最后一部分,表示不论发生任何情况都会执行,finally 部分可以省略,但如果 finally 部分存在,则一定会执行 finally 里面的代码。
  • finalize: 是 Object 类的一个方法,在垃圾收集器执行的时候会调用被回收对象的此方法。

try-catch-finally 中哪个部分可以省略

try-catch-finally 其中 catch 和 finally 都可以被省略,但是不能同时省略,也就是说有 try 的时候,必须后面跟一个 catch 或者 finally。

try-catch-finally 中,如果 catch 中 return 了,finally 还会执行吗?

finally 一定会执行,即使是 catch 中 return 了,catch 中的 return 会等 finally 中的代码执行完之后,才会执行

常见的异常类有哪些

  • NullPointerException 空指针异常
  • ClassNotFoundException 指定类不存在
  • NumberFormatException 字符串转换为数字异常
  • IndexOutOfBoundsException 数组下标越界异常
  • ClassCastException 数据类型转换异常
  • FileNotFoundException 文件未找到异常
  • NoSuchMethodException 方法不存在异常
  • IOException IO 异常
  • SocketException Socket 异常

网络

http 响应码 301 和 302 代表的是什么?有什么区别

301:永久重定向。

302:暂时重定向。

它们的区别是,301 对搜索引擎优化(SEO)更加有利;302 有被提示为网络拦截的风险。

常用的HTTP相应状态码

1XX:指示信息--表示请求已接收,继续处理
2XX:成功-表示请求已成功接收,理解,接受
3XX:重定向-要完成请求必须进一步的操作
4XX:客户端错误-请求有语法错误或者请求无法实现
5XX:服务端错误--服务器未能实现合法的请求

  • 200:请求被正常处理
  • 204:请求被受理但没有资源可以返回
  • 206:客户端只是请求资源的一部分,服务器只对请求的部分资源执行GET方法,相应报文中通过Content-Range指定范围的资源。
  • 301:永久性重定向
  • 302:临时重定向
  • 303:与302状态码有相似功能,只是它希望客户端在请求一个URI的时候,能通过GET方法重定向到另一个URI上
  • 304:发送附带条件的请求时,条件不满足时返回,与重定向无关
  • 307:临时重定向,与302类似,只是强制要求使用POST方法
  • 400:请求报文语法有误,服务器无法识别
  • 401:请求需要认证
  • 403:请求的对应资源禁止被访问
  • 404:服务器无法找到对应资源
  • 500:服务器内部错误
  • 503:服务器正忙

Http和Https的区别

  1. HTTP 的URL 以http:// 开头,而HTTPS 的URL 以https:// 开头
  2. HTTP 是不安全的,而 HTTPS 是安全的
  3. HTTP 标准端口是80 ,而 HTTPS 的标准端口是443
  4. 在OSI 网络模型中,HTTP工作于应用层,而HTTPS 的安全传输机制工作在传输层
  5. HTTP 无法加密,而HTTPS 对传输的数据进行加密
  6. HTTP无需证书,而HTTPS 需要CA机构颁发的SSL证书

Http是如何保持长连接的?

长连接:指的是通过三次握手建立起一次连接,可以在此连接内发送多次request请求,http1.0是短链接,http1.1协议默认是长连接,响应头加入Connection:keep-alive

什么是http的无状态协议,怎么解决

无状态协议就是对事务没有记忆能力,缺少状态意味着如果后续处理需要前面的信息,也就是说,当用户第一次请求服务器,再一次发生请求的时候并不知道它是老用户。

可以用cookie来解决无状态的问题,Cookie相当于一个通行证,客户端第一次访问的时候给客户端发送一个cookie,当用户再一次请求的时候,带着这个cookie,那么服务器就知道这是老用户了。

什么是Cookie

因为网页之间的交互是通过http协议传输数据的,而Http协议是无状态协议,也就是说当用户数据提交完毕之后,浏览器和服务器就会关闭,再次交互的时候需要建立新的连接

服务器无法确认用户的信息,于是w3c就提出了:给每个用户都发一个通行证,无论是谁访问都需要携带通行证,这样服务器就可以在通行证上确定用户的消息了,这个通行证就是Cookie

Session和Cookie的区别

  • 从存储上:Session是存储在服务端的,Cookie是存储在客户端的,并且Cookie只能存储字符串,如果要存储非ASCII码需要进行对其编码,Session可以存储任何类型的数据,可以把Seesion看成一个容器
  • 从安全上:Session的安全性要比Cookie高,因为cookie存储在客户端,对用户是可见的,而Session是存储在服务端。
  • 从有效期上,Cookie存储在本地硬盘,只要设置maxAge为比较大的数,即使关闭了浏览器,Cookie还是存在的,Session存储在服务器中,设置maxInactiveInterval(inactive 不活跃的,interval间隔)属性来确定Session的有效值,并且Session依赖于名为Jsessionid的Cookie,如果该cookie的maxage属性小于0,关闭浏览量,那么服务器中的session就会失效。
  • Cookie禁用的情况下,Session可以通过URL重写技术来进行会话跟踪。在访问路径url后面,带一个jsessionid的参数

URI和URL的区别

  1. URI统一资源标识符,Web上可用的每种资源如HTML文档,图像,视频片段,程序都是URI来定位的
    一般由三个部分组成
  • 访问资源的命名机制
  • 存放资源的主机名
  • 资源自身的名称,由路径表示,着重强调与资源
  1. URL是统一资源定位器,它是一种具体的URI,即URL可以用来标识一个资源,而且还指明了如何locate这个资源。采用URL可以用一种统一的格式来描述各种信息资源,包括文件、服务器的地址和目录等。URL一般由三部组成:
  • ①协议(或称为服务方式)
  • ②存有该资源的主机IP地址(有时也包括端口号)
  • ③主机资源的具体地址。如目录和文件名等

URL是URI的子集

TCP和UDP的区别

tcp和udp是OSI模型中运输层中的协议,tcp提供可靠的通信传输,而udp则常被用于让广播和细节控制交给应用的通信传输。

两者的区别大致如下:

  • tcp面向连接,udp面向非连接即发送数据时不需要建立连接
  • tcp提供可靠的服务,udp无法保证
  • tcp面向字节流,udp面向报文
  • tcp数据传输慢,udp数据传输慢

tcp 为什么要三次握手,两次不行吗?为什么?

如果采用两次握手,那么只要服务器发出确认数据包就会建立连接,但可能这时候客户端的接收功能坏了,客户端此时并未响应服务器端的请求,那此时服务器端就会一直在等待客户端,这样服务器端就白白浪费了一定的资源。若采用三次握手,服务器端没有收到来自客户端的再此确认,则就会知道客户端并没有要求建立请求,就不会浪费服务器的资源。

第一次:客户端发送请求到服务端,这样服务器知道了客户端发送正常,自己接受正常,SYN=1,seq=x
第二次:服务器发送给客户端,这样客户端知道自己发送,接收正常,服务器接收正常,发送正常,ACK=1,ack=x+1,SYN=1,seq=y
第三次:就剩服务器还不知道客户端接收是否正常和自己发送是否正常了,这样客户端继续发送给服务端,服务端知道客户端发送,接收正常,自己也发送接收正常了。seq=x+1,ACK=1,ack=y+1

四次挥手的过程

  • 第一次挥手:客户端发出释放FIN=1,自己序列号seq=u,进入FIN-WAIT-1状态
  • 第二次挥手:服务器收到客户端的后,发出ACK=1确认标志和客户端的确认号ack=u+1,自己的序列号seq=v,进入CLOSE-WAIT状态
  • 第三次挥手:客户端收到服务器确认结果后,进入FIN-WAIT-2状态。此时服务器发送释放FIN=1信号,确认标志ACK=1,确认序号ack=u+1,自己序号seq=w,服务器进入LAST-ACK(最后确认态)
  • 第四次挥手:客户端收到回复后,发送确认ACK=1,ack=w+1,自己的seq=u+1,客户端进入TIME-WAIT(时间等待)。客户端经过2个最长报文段寿命后,客户端CLOSE;服务器收到确认后,立刻进入CLOSE状态。

分析:
第一次:客户端请求断开FIN,seq=u
第二次:服务器确认客户端的断开请求ACK,ack=u+1,seq=v
第三次:服务器请求断开FIN,seq=w,ACK,ack=u+1
第四次:客户端确认服务器的断开ACK,ack=w+1,seq=u+1

为什么连接是三次握手,关闭时四次握手

在关闭连接时,当服务端收到FIN报文时,可能服务端还有些数据还没处理,并不会立即关闭socket,只能先回复一个ACK报文,告诉客户端,你的报文我收到了,只有等到我服务端所有的报文都发送完了才能发送FIN报文,因此不能一起发送,所有需要四次握手

说一下Request的结构

  1. 请求头
    • 请求方法
    • 请求资源
    • 协议/版本
  2. 请求报头
    • 每个头由一个域名,冒号和域值三部分组成
    • 告诉web服务器浏览器接收的语言版本
    • 请求web服务器的ip地址和端口号
    • Cookies等信息
  3. 空白行
    • 分割报头和请求体的专用行
  4. 请求体
    • get请求方法不传送任何数据,post请求可以传数据

说一下 tcp 粘包是怎么产生的?

tcp 粘包可能发生在发送端或者接收端,分别来看两端各种产生粘包的原因:

  • 发送端粘包:发送端需要等缓冲区满才发送出去,造成粘包;
  • 接收方粘包:接收方不及时接收缓冲区的包,造成多个包接收。

OSI 的七层模型都有哪些?

  • 物理层:利用传输介质为数据链路层提供物理连接,实现比特流的透明传输。
  • 数据链路层:负责建立和管理节点间的链路。
  • 网络层:通过路由选择算法,为报文或分组通过通信子网选择最适当的路径。
  • 传输层:向用户提供可靠的端到端的差错和流量控制,保证报文的正确传输。
  • 会话层:向两个实体的表示层提供建立和使用连接的方法。
  • 表示层:处理用户信息的表示问题,如编码、数据格式转换和加密解密等。
  • 应用层:直接向用户提供服务,完成用户希望在网络上完成的各种工作。

get 和 post 请求有哪些区别?

  • 数据携带上,get 传递参数有大小限制,而 post 没有。
  • 请求参数的位置上,get方式,请求参数放在url地址后面,以?进行拼接;post方式,请求参数放在HTTP请求包内
  • 用途上,post参数传输更安全,所以get方式通常获取数据,post方式通常提交数据
  • get方式比post方式要快

如何实现跨域

实现跨域有以下几种方案

  • 服务器端运行跨域 设置 CORS 等于 *;
  • 在单个接口使用注解 @CrossOrigin 运行跨域;
  • 使用 jsonp 跨域;

说一下 JSONP 实现原理

jsonp:JSON with Padding,它是利用script标签的 src 连接可以访问不同源的特性,加载远程返回的“JS 函数”来执行的。

JDBC

jdbc操作数据库的步骤

  1. 建立数据库驱动
  2. 建立数据库连接
  3. 创建一个Statement
  4. 执行sql语句
  5. 处理结果集
  6. 关闭数据库连接

为什么要使用PreparedStatement

  1. PreparedStatement接口继承Statement,PreparedStatement实例包含已编译的SQL语句,所以执行速度高于Statement
  2. 作为Statement的子类,PreparedStatement继承了Statement的所有功能,三种方法execut,executeQuery、executeUpdate
  3. 在jdbc应用中,任何时候都不要使用Statement,原因如下:
    1. 代码的可读性和可维护性,Statement需要不断的拼接,而preparedStatement不会
    2. PreparedStatement尽最大可能提高性能.DB有缓存机制,相同的预编译语句再次被调用不会再次需要编译。
    3. 最重要的一点是极大的提高了安全性。Statement容易被SQL注入,而PreparedStatement传入的内容不会和sql语句发生任何匹配关系。

关系型数据库的连接池的机制是什么?

前提:为数据库连接池建立一个缓存池

  1. 能够从连接池获取或创建可用的连接
  2. 使用完毕后,将连接返回给连接池
  3. 在系统关闭之前,关闭所有的连接并释放占用系统的资源
  4. 能够处理无效的连接,限制连接池中的连接总数不低于或不超过某个限定值

Mybatis

MyBatis中#{}和${}的区别是什么?

#{} 是预编译处理,${}是字符替换,在使用 #{}时,MyBatis 会将 SQL 中的 #{}替换成“?”,配合 PreparedStatement 的 set 方法赋值,这样可以有效的防止 SQL 注入,保证程序的运行安全。

通常一个 Xml 映射文件,都会写一个 Dao 接口与之对应,请问,这个 Dao 接口的工作原理是什么?Dao 接口里的方法,参数不同时,方法能重载吗?

dao接口,就是人们常说的Mapper接口,接口的全限名,就是映射文件中的 namespace 的值,接口的方法名,就是映射文件中MappedStatement的 id 值,接口方法内的参数,就是传递给 sql 的参数。Mapper接口是没有实现类的,当调用接口方法时,接口全限名+方法名拼接字符串作为 key 值,可唯一定位一个MappedStatement,举例:com.mybatis3.mappers.StudentDao.findStudentById,可以唯一找到 namespace 为com.mybatis3.mappers.StudentDao下面id = findStudentByIdMappedStatement。在 Mybatis 中,每一个<select><insert><update><delete>标签,都会被解析为一个MappedStatement对象。

Dao 接口里的方法,是不能重载的,因为是全限名+方法名的保存和寻找策略。

Dao 接口的工作原理是 JDK 动态代理,Mybatis 运行时会使用 JDK 动态代理为 Dao 接口生成代理 proxy 对象,代理对象 proxy 会拦截接口方法,转而执行MappedStatement所代表的 sql,然后将 sql 执行结果返回。

MyBatis有几种分页方式

分页方式:物理分页和逻辑分页

逻辑分页:使用mybatis自带的RowBounds进行分页,它是一次性查询很多数据,然后在数据中进行检索

物理分页:自己手写SQL分页或者使用分页插件PageHandler,去数据库查询指定条数的分页数据形式

Mybatis逻辑分页和物理分页的区别是什么?

  • 逻辑分页是一次性查询很多数据,然后再在结果中检索分页的数据。这样做弊端是需要消耗大量的内存、有内存溢出的风险、对数据库压力较大。
  • 物理分页是从数据库查询指定条数的数据,弥补了一次性全部查出的所有数据的种种缺点,比如需要大量的内存,对数据库查询压力较大等问题

RowBounds是一次性查询全部结果吗?为什么?

RowBounds 表面是在“所有”数据中检索数据,其实并非是一次性查询出所有数据,因为 MyBatis 是对 jdbc 的封装,在jdbc 驱动中有一个 Fetch Size 的配置,它规定了每次最多从数据库查询多少条数据,假如你要查询更多数据,它会在你执行next()的时候,去查询更多的数据。就好比你去自动取款机取 10000 元,但取款机每次最多能取 2500 元,所以你要取 4次才能把钱取完。只是对于 jdbc 来说,当你调用 next()的时候会自动帮你完成查询工作。这样做的好处可以有效的防止内存溢出。

Mybatis是否支持延迟加载,延迟加载的原理是什么?

MyBatis 支持延迟加载,设置 lazyLoadingEnabled=true 即可。

延迟加载的原理的是调用的时候触发加载,而不是在初始化的时候就加载信息。比如调用 a. getB(). getName(),这个时候发现a. getB() 的值为 null,此时会单独触发事先保存好的关联 B 对象的 SQL,先查询出来 B,然后再调用 a.setB(b),而这时候再调用 a. getB(). getName() 就有值了,这就是延迟加载的基本原理。

说一下MyBatis的一级缓存和二级缓存

  • 一级缓存:基于 PerpetualCache 的 HashMap 本地缓存,它的声明周期是和 SQLSession一致的,有多个 SQLSession 或者分布式的环境中数据库操作,可能会出现脏数据。当 Session flush 或 close 之后,该Session 中的所有 Cache 就将清空,默认一级缓存是开启的。

  • 二级缓存:也是基于 PerpetualCache 的 HashMap 本地缓存,不同在于其存储作用域为 Mapper级别的,如果多个SQLSession之间需要共享缓存,则需要使用到二级缓存,并且二级缓存可自定义存储源,如Ehcache。默认不打开二级缓存,要开启二级缓存,使用二级缓存属性类需要实现 Serializable 序列化接口(可用来保存对象的状态)。

    开启二级缓存数据查询流程:二级缓存 -> 一级缓存 -> 数据库。

    缓存更新机制:当某一个作用域(一级缓存 Session/二级缓存 Mapper)进行了C/U/D 操作后,默认该作用域下所有 select 中的缓存将被 clear。

MyBatis 和 hibernate 的区别有哪些

  • 灵活性:MyBatis 更加灵活,自己可以写 SQL 语句,使用起来比较方便。
  • 可移植性:MyBatis 有很多自己写的 SQL,因为每个数据库的 SQL 可以不相同,所以可移植性比较差。
  • 学习和使用门槛:MyBatis 入门比较简单,使用门槛也更低。
  • 二级缓存:hibernate 拥有更好的二级缓存,它的二级缓存可以自行更换为第三方的二级缓存。

MyBatis 有哪些执行器(Executor)?

MyBatis 有三种基本的Executor执行器:

  • SimpleExecutor:每执行一次 update 或 select 就开启一个 Statement 对象,用完立刻关闭 Statement 对象;
  • ReuseExecutor:执行 update 或 select,以 SQL 作为 key 查找 Statement
    对象,存在就使用,不存在就创建,用完后不关闭 Statement 对象,而是放置于 Map 内供下一次使用。简言之,就是重复使用
    Statement 对象;
  • BatchExecutor:执行 update(没有 select,jdbc 批处理不支持 select),将所有 SQL
    都添加到批处理中(addBatch()),等待统一执行(executeBatch()),它缓存了多个 Statement 对象,每个
    Statement 对象都是 addBatch()完毕后,等待逐一执行 executeBatch()批处理,与 jdbc 批处理相同。

MyBatis 分页插件的实现原理是什么

分页插件的基本原理是使用 MyBatis 提供的插件接口,实现自定义插件,在插件的拦截方法内拦截待执行的 SQL,然后重写 SQL,根据 dialect 方言,添加对应的物理分页语句和物理分页参数。

MyBatis 如何编写一个自定义插件

自定义插件实现原理

MyBatis 自定义插件针对 MyBatis 四大对象(Executor、StatementHandler、ParameterHandler、ResultSetHandler)进行拦截:

  • Executor:拦截内部执行器,它负责调用 StatementHandler 操作数据库,并把结果集通过 ResultSetHandler 进行自动映射,另外它还处理了二级缓存的操作;
  • StatementHandler:拦截 SQL 语法构建的处理,它是 MyBatis 直接和数据库执行 SQL 脚本的对象,另外它也实现了 MyBatis 的一级缓存;
  • ParameterHandler:拦截参数的处理;
  • ResultSetHandler:拦截结果集的处理。

自定义插件实现关键

MyBatis 插件要实现 Interceptor 接口,接口包含的方法,如下:

Copypublic interface Interceptor {   
   Object intercept(Invocation invocation) throws Throwable;       
   Object plugin(Object target);    
   void setProperties(Properties properties);
}
  • setProperties 方法是在 MyBatis 进行配置插件的时候可以配置自定义相关属性,即:接口实现对象的参数配置;
  • plugin 方法是插件用于封装目标对象的,通过该方法我们可以返回目标对象本身,也可以返回一个它的代理,可以决定是否要进行拦截进而决定要返回一个什么样的目标对象,官方提供了示例:return Plugin. wrap(target, this);
  • intercept 方法就是要进行拦截的时候要执行的方法。

Redis

redis的好处

  1. 速度快,数据都存储在内存中
  2. 丰富的数据结构,有String,List,hash,Set,Zset
  3. 支持事务
  4. 丰富的特性,可用于缓存,消息,按key设置过期时间,过期后会自动删除

Redis使用场景

  1. String 想知道什么时候封锁一个ip地址,Incrby命令
  2. Hash存储用户的信息(id,name,age)
  3. List实现最新消息排行
  4. Set自动排重,比如微博中将每个人的好友存在集合Set中,求两个人的共通好友,我们只要求交集即可
  5. Zset以某个条件为权重,进行排序;京东:商品详情都有一个综合排序,还可以按照价格进行排名

Redis持久化机制

  1. RDB
    RDB方式是在指定的时间间隔内将内存中的数据集快照写入到磁盘中,会生成一个dump.rdb文件,方便持久化。恢复直接将快照文件直接读到内存中。
  2. AOF
    AOF方法是以日志的形式来记录每个写操作,也就是将redis的每个写的指令记录下来,不记录读的指令,只允许追加文件,不允许改写文件,会生成一个appendonly.aof文件,开启aof方式是在redis的配置文件中将appendonly设置为yes,要保证数据完全不丢失的话,可以将appendfsync设置为always,也就是只要发生变化就会持久化。

AOF的优点:

  1. 备份机制更稳健,丢失数据的概率更低
  2. 可读的日志文本,通过操作aof稳健,可以处理误操作
    AOF的缺点:
  3. 比RDB占用多的磁盘空间
  4. 恢复备份速度要慢
  5. 每次读写都同步的话,有一定的性能压力
  6. 存在个别bug,造成恢复不能

RDB的优点:节省磁盘空间,恢复数据快,虽然redis在fork时使用了写时拷贝技术,如果数据量大的话也是非常消耗性能的。

RDB的缺点:因为备份周期是在一定间隔时间做一次备份,如果redis意外down掉的话,会丢失最后一次快照后的所有修改

Redis是单进程还是多线程

redis是单进程单线程,redis利用队列技术将并发访问变为串行访问,消除传统数据库串行控制的开销

String类型的最大开销

512M

Redis过期键的删除操作

  • 惰性删除,放任键过期不管,但是每次从键空间获取键时,都检查取得的键是否过期,如果过期,就删除该键,如果没有过期,就返回该键。

  • 定期删除,每过一段时间,程序就对数据库进行一次检查,删除里面的过期键。

为什么redis需要把所有的数据放到内存中

为了达到最快的读写速度将数据都读到内存中,并通过异步的方式将数据写入磁盘,所以redis 具有快速和数据持久化的特征。如果不将数据放在内存中,磁盘I/O 速度为严重影响redis 的性能。在内存越来越便宜的今天, redis 将会越来越受欢迎。如果设置了最大使用的内存,则数据已有记录数达到内存限值后不能继续插入新值。

Redis的同步机制

Redis可以使用主从同步,从从同步,第一次同步时,主节点做一次bgsave,并同时将后续修改操作记录到内存buffer,待完成后将rdb文件全量同步到复制节点,复制节点接受完成后将rdb镜像加载到内存,加载完成后,在通知主节点将期间修改的操作记录同步到复制节点进行重放就完成了同步过程。

什么是缓存穿透,如何解决

一般是黑客故意去请求缓存中不存在的数据,导致所有的请求都落到数据库上,造成数据库短时间内承受大量的请求而崩掉。
解决方法

  1. 使用布隆过滤器,将数据库中所有可能存在的数据哈希到一个足够的大bitmap中,这样一个一定不存在的数据就会被这个bitmap给拦截掉,从而避免了对底层存储系统的查询压力。
  2. 如果一个查询返回结果为空,我们仍然将这个空结果进行缓存,但它的过期时间会很短,一般不会超过5分钟。

Redis雪崩

缓存同一时间大面积失效,所以,后面的请求都会落到数据库上,造成数据库短时间承受大量请求而崩掉。
解决

  1. 尽量保证整个redis集群的高可用,发现机器宕机尽快补上,
  2. 本地ehcache缓存+hystrix限流&降级,避免MySQL崩掉
  3. 利用redis持久化机制,保存的数据尽快恢复缓存。

双写一致性

数据库和缓存双缓存,双写,如何解决一致性问题

  1. 如果你的系统不是严格要求缓存+数据必须强一致性的话,可以采用Cache-Aside Pattern
  2. 如果不得不做强一致性时,那么可以使用读请求和写请求串行化,串到一个内存队列里去
  3. 使用队列串行化的方式会导致系统的吞吐量会大幅度的降低

主从复制和哨兵机制

主从复制的好处:

避免redis单点故障

构建读写分离的架构,满足读多写少的应用场景

复制的过程原理

  1. 当从库和主库建立MS关系后,会向主数据库发送sync命令
  2. 主库接收到sync命令后,会开始在后台fork保存快照(RDB持久化过程bgsave)并将期间收到的写命令缓存起来。
  3. 当快照完成后,主Redis会将快照文件和所有缓存的写命令发送到从Redis。
  4. 当从Redis接收到后,会载入快照文件并且执行收到的缓存命令
  5. 主Redis每当接收到的写命令都会发送给从Redis,从而保证数据的一致性

哨兵
哨兵的作用是对Redis系统运行情况进行监控,是一个独立的进程,他的功能有两个:

  1. 监控主数据库和从数据库是否运行正常
  2. 主数据库出现故障后自动将从服务器转为主服务器

既然有了主从了,为什么还需要集群

即使有了主从复制,每个数据库都需要保存整个集群中的所有数据,容易形成木桶效应,所以还需要集群。

集群hash槽和key的关系

key的有效使用CRC16算法计算出哈希值,再对哈希值对163284取余,得到插槽值。

如何解决集群一个节点下线,导致插槽区有空档

在Redis集群中可以使用主从模式实现一个节点的高可用
在该结点宕机后,集群会将该结点的slave转变为master继续完成集群服务。

MySQL

BLOB和TEXT有什么区别

BLOB是一个二进制的对象,可以容纳可变数量的数据,TEXT是一个不区分大小写的BLOB。BLOB和TEXT类型的唯一区别在于对BLOB值进行排序和比较时区分大小写,对TEXT值不区分大小写

now()和current_date()有什么区别?

now()命令用于显示当前年份,月份,日期,小时,分钟,和秒
current_date()仅显示当前年份,月份和日期

char和varchar的区别

char的长度是有限的,而varchar的长度是不可变的,当char值被存储时,它们被用空格补充到特定的长度,检索char值需要删除尾随空格。

如果一个表有一列定义为TIMESTAMP,将会发生什么

每当行被更改时,时间戳字段将获取当前时间戳

列设置为AUTO INCREMENT 时, 如果在表中达到最大值,会发生什么情况?

它会停止递增,任何进一步的插入都将产生错误, 因为密钥已被使用。

怎样才能找出最后一次插入时分配了哪个自动增量?

LAST_INSERT_ID 将返回由Auto_increment 分配的最后一个值,并且不需要指定表名称。

mysql中InnoDB支持的四种隔离级别名称

  1. read uncommited:读到未提交的数据
  2. read committed: 读已提交
  3. repeatable read:可重复读
  4. serializable :串行事物

简述mysql中myISAM和InnoDB

myISAM

  1. 不支持事务,但每次查询都是原子的
  2. 支持表级锁,即每次操作是对整个表加锁
  3. 存储表的总行数
    一个myISAM表有三个文件:索引文件,表结构文件,数据文件

InnoDB

  1. 支持ACID的事务,支持事务的四种隔离级别
  2. 支持行级锁及外键约束,因此可以支持写并发

SQL中on条件和where条件的区别

数据库在通过连接两张或多张表来返回记录时,都会生成一张中间的临时表,然后再将这张临时表返回给用户。
在使用left join时,on和where条件的区别如下:

  1. on条件是在生成临时表时使用的条件,它不管on中的条件是否为真,都会返回左边表中的记录
  2. where条件是在临时表生成好后,再对临时表进行过滤的条件,这时已经没有left join的含义(必须返回左边表的记录)了,条件不为真的就全部过滤掉。

四个特征及含义

四个基本要素:原子性,一致性,隔离性,持久性

  1. 原子性:事务是一个原子操作单元,其对数据的修改,要么全部执行,要么全部不执行。
  2. 一致性:在事务开始和完成时,数据都必须保持一致状态,这意味着所有相关的数据规则都必须应用于事务的修改,以保持数据的完整性;事务结束时,所有的内部数据结构(如B树索引或双向链表)也都必须是正确的。
  3. 隔离性:数据库系统提供着一定的隔离机制,保证事务在不受外部并发操作影响的独立环境执行,这意味着事务处理过程的中间状态对外部是不可见的,反之亦然。
  4. 持久性:事务完成之后,它对数据的修改时永久性的,即使出现系统故障也能够保持

drop,delete,truncate的区别

drop 直接删除表
truncate 删除表中的数据,在插入数据自增长id又从1开始 trun cate
delete 删除表中的数据,可以加where语句

索引

数据本身之外,数据库还维护着一个满足特定查找算法的数据结构,这些数据结构以某种方式指向数据,以帮助MySQL高效获取数据,这种数据结构就是索引。

优点

  1. 类似大学图书馆建书目索引,提高数据检索的效率,降低数据库的IO成本
  2. 通过索引列对数据进行排序,降低数据排序的成本,降低了CPU的消耗
  3. 创建唯一索引,可以保证每一行数据的唯一性
  4. 大大加快检索的速度
  5. 加速表之间的连接
  6. 使用分组和排序子句进行数据检索时,可以显著减少查询中分组和排序的时间

劣势

  1. 建立索引需要占用物理空间
  2. 对表的数据进行增删改查时,索引也要动态维护,降低了数据的维护速度
  3. 创建和维护索引耗费时间,时间随着数据量的增加而增加

不需要创建索引

  • 查询中很少使用或者是参考的列
  • 数据量很少的列
  • 对于那些定义text、image和bit数据类型

行级锁,表级锁,乐观锁,悲观锁

  • 表级锁:每次操作都锁定在整张表,开销少,加锁快,不会出现死锁,锁定粒度大,发生锁冲突的概率最高,并发度最低
  • 行级锁:每次操作锁定一行数据,开销大,加锁慢,会出现死锁,锁定粒度小,发生锁冲突的概率最低,并发度最高
  • 页面锁:开销和加锁时间介于两者之间,会出现死锁,并发度一般
  • 悲观锁:假定会发生并发冲突,屏蔽一切可能违反数据完整性的操作
  • 乐观锁:假设不会发生并发冲突,只在提交操作时检查是否违反数据完整性
    乐观锁更适合解决冲突概率极小的情况,而悲观锁则适合解决并发竞争激烈的情况,尽量使用行锁,缩小加锁粒度,以提高并发处理能力,即使加行锁的时间比表锁要长

并发事务带来的问题

相对于串行处理来说,并发事务处理能大大增加数据库资源的利用率,提高数据库系统的事务吞吐量,从而可以支持更多的用户

更新丢失

当两个或多个事务选择同一行,然后基于最初选定的值更新该行时,由于每个事务都不知道其他事务的存在,就会发生丢失更新问题——最后的更新覆盖了由其他事务所做的更新。
例如,两个程序员修改同一java文件,每个程序员独立地更改其副本,然后保存更改后的副本,这样就覆盖了原始文档,最后保存其更改副本的编辑人员覆盖前一个程序员的更改。
如果在一个程序员完成并提交事务之前,另一个程序员不能访问同一文件,则可避免此问题。

脏读 事务A读取到了事务B已修改但尚未提交的数据

一个事务正在对一条记录做修改,在这个事务完成并提交前,这条记录的数据就处于不一致状态;这时,另一个事务也来读取同一条记录,如果不加控制,第二个事务读取了这些脏数据,并据此做进一步的处理,就会产生未提交的数据依赖关系,这种现象被形象的叫做脏读。
一句话:事务A读取到了事务B已修改但尚未提交的数据,还在这个数据基础上做了操作,此时,事务B进行回滚,A读取的数据无效,不符合一致性要求。

不可重复读 事务A读取到了事务B已经提交的数据

一个数据在读取某些数据后的某个时间,再次读取以前读过的数据,却发现其读取出的数据已经发生了变化,或某些记录已经被删除了,这种现象就叫做不可重复读。
一句话:事务A读取到了事务B已经提交的修改数据,不符合隔离性。

幻读 事务A读取了事务B提交的新增数据

一个数据按相同的查询条件重新读取以前检索过的数据,却发现其他事务插入了满足其条件查询的新数据,这种现象就称为幻读。
一句话:事务A读取到了事务B提交的新增数据,不符合隔离性
多说一句;幻读和脏读有点类似,脏读是事务B里面修改了数据,幻读是事务B里面新增了数据。
更新丢失可以通过加锁来解决,“脏读”、“不可重复读”、“幻读”属于数据库一致性问题,由数据库提供一定的事务隔离机制来解决

事务隔离方式实现

  1. 在读取数据前,对其进行加锁,阻止事务对数据进行修改
  2. 不加任何锁,通过一定机制生成一个数据请求时间点的一致性数据快照,并用这个快照提供一定级别的一致性读取,这种方式也叫做多版本并发控制,简称MVCC或MCC

事务隔离度级别

数据库的隔离越严格,并发副作用越小,付出的代价也就越大,隔离的实质就是在一定程度上使事务串行化,这显然与并发是矛盾的。
未提交读(Read uncommitted)
已提交读(Read committed)
可重复读(Repeatable read)
可序列化(Serializable)

innodb和myisam的区别

  1. innodb支持事务,myisam不支持
  2. innodb支持行级锁,myisam支持表级锁
  3. innodb支持外键,myisam不支持
  4. innodb支持MVCC,myisam不支持
  5. innodb不支持全文索引,myisam支持
  6. 应用场景:
    MyISAM:做很多count计算、插入不频繁、查询非常频繁、没有事务、查询快
    InnoDB:对事务要求比较高、可靠性要求高、表更新相当频繁、并发写入高
  7. DELETE操作:
    MyISAM:先drop表,然后重建表
    InnoDB:一行一行删除
  8. 查询表的行数不同:
    MyISAM:只是简单的读出保存好的行数
    InnoDB:不保存具体行数,执行count(*)时要扫描一整个表来计算有多少行

innodb引擎的四大特征

  1. 插入缓存(insert buffer)
  2. 二次写(double write)
  3. 自适应哈希索引
  4. 预读

MyISAM会比InnoDB的查询要快

  1. 数据块:InnoDB要缓存,MyISAM只缓存索引块
  2. InnoDB寻址要映射到块再到行,MyISAM记录的直接是文件的OFFSET,定位比InniDB要快
  3. InnoDB还要维护MVCC一致

varchar和char的区别以及varchar(50)50的含义

区别:char是一种固定长度的类型,varchar则是一种可变长度的类型
含义:最多存放50个字符,varchar(50)和varchar(200)存储hello所占空间一样,但后者在排序时会消耗更多的内存,因为order by col采用fixed_length计算col长度
int(10)10的含义:是指显示字符的长度,最大为255,仍占4字节存储,存储范围不变

int范围

有符号的整型范围是-2147483648~2147483647
无符号的整型范围是0~4294967295(2^32)
int(10)的意思是假设有一个变量名为id,它的能显示的宽度能显示10位。在使用id时,假如我给id输入10,那么mysql会默认给你存储0000000010。当你输入的数据不足10位时,会自动帮你补全位数。假如我设计的id字段是int(20),那么我在给id输入10时,mysql会自动补全18个0,补到20位为止。

innodb的事务与日志实现方式

  1. 日志种类
    错误日志:记录出错信息,也记录一些警告或是正确的信息
    查询日志:记录所有对数据请求信息,不论是否得到正确执行
    慢查询日志:设置一个阈值,将运行时间超过该值的所有SQL语句都记录进去
    二进制日志:记录对数据库执行更改的所有操作
  2. 事务是如何通过日志实现的
    事务日志是通过redo和innodb的存储引擎日志缓冲来实现的,当开始一个事务的时候,
    会记录该事务的lsn号,当事务执行时,会往innodb存储引擎日志缓存里面插入事务日志,
    当事务提交时,必须将存储引擎的日志缓存写入磁盘,也就是写数据前,要先写日志

innodb引擎的行锁怎么实现的

是基于索引来完成行锁的

例如:select * from a_table where id = 1 for update;

for update 可以根据条件来完成行锁定,并且id是有索引键的列,如果id不是索引键,那么innodb将完成表锁

Spring

为什么要使用Spring

  1. spring提供ioc技术,容器会帮你管理依赖的对象,从而不需要自己创建和管理依赖了,更轻松的实现了程序的解耦
  2. spring提供了事务支持,使得事务操作变得更加方便了
  3. spring提供了面向切面编程,这样可以更轻松的处理某一类问题
  4. 更轻松的框架集成,spring可以很方便的集成其他框架,如mybatis,hibernate

解释一下什么是aop

aop是面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。

简单来说就是统一处理某一切面类的问题的编程思想,比如统一处理日志,异常等。

什么是ioc

ioc(控制反转)是spring的核心,对于spring框架来说,就是由spring来负责控制对象的生命周期和对象间的关系

简单来说,控制就是当前对象对内部成员的控制权,控制反转就是,这种控制权不由当前对象管理了,由第三方容器来管理。

spring有哪些主要模块

  • spring core:框架的最基础部分,提供 ioc 和依赖注入特性。
  • spring context:构建于 core 封装包基础上的 context 封装包,提供了一种框架式的对象访问方法。
  • spring dao:Data Access Object 提供了JDBC的抽象层。
  • spring aop:提供了面向切面的编程实现,让你可以自定义拦截器、切点等。
  • spring Web:提供了针对 Web 开发的集成特性,例如文件上传,利用 servlet listeners 进行 ioc 容器初始化和针对 Web 的 ApplicationContext。
  • spring Web mvc:spring 中的 mvc 封装包提供了 Web 应用的 Model-View-Controller(MVC)的实现。

Spring的作用域有什么区别

  1. singleton在SpringIOC容器中仅存在一个Bean实例,Bean以单例的方式存在
  2. prototype每次调用getBean()都会返回一个新实例
  3. request每次Http请求都会创建一个新的Bean,该作用域仅适用于WebApplicationContext环境
  4. session同一个HttpSession共享一个Bean,不同的HttpSession适用不同的Bean,该作用域仅适用于WebApplicationContext环境

简单介绍一下SpringBean的生命周期

  1. bean的定义:在配置文件里进行定义
  2. bean的初始化:有两种初始化的方式:
    1. 在配置文件中通过指定的init-method属性来完成
    2. 实现initializingBean接口
  3. bean的调用:有三种方式可以得到bean的实例,并进行调用
  4. bean销毁:两种方式
    1. 配置文件指定destroy-method属性
    2. 实现DisposeableBean接口

spring常用的注入方式有哪些

setter注入

构造方法注入

注解方法注入

spring中的bean是线程安全的吗

spring中的bean是默认是单例模式,spring并没有对单例bean进行多线程的封装处理

实际上大部分时候springbean无状态的(比如dao类),某种程度来说,bean也是安全的,但如果bean有状态的话(比如view model),那么开发者就要保证线程安全了,最简单的就是改变bean的作用域,把singleton变更为prototype,这样请求bean相当于new Bean()了,所以就可以保证线程安全了。

有状态的就是有数据存储功能

无状态的就是不会保存数据

spring支持几种bean的作用域

spring支持5种作用域,如下:

  • singleton:spring ioc 容器中只存在一个 bean 实例,bean 以单例模式存在,是系统默认值;

  • prototype:每次从容器调用 bean 时都会创建一个新的示例,既每次 getBean()相当于执行 new Bean()操作;

    Web 环境下的作用域:

  • request:每次 http 请求都会创建一个 bean;

  • session:同一个 http session 共享一个 bean 实例;

  • global-session:用于 portlet 容器,因为每个 portlet 有单独的 session,globalsession 提供一个全局性的 http session。

注意使用prototype作用域需要慎重选择,因为频繁创建和销毁bean会带来很大的性能开销

自动装配bean有哪些方式

  • no:默认值,表示没有自动装配,应使用显式的bean引用进行装配
  • byName:它根据bean名称注入对象依赖项
  • byType:它根据类型注入对象依赖项
  • 构造函数:通过构造函数来注入依赖项,需要设置大量的参数
  • autodetect:容器首先通过构造函数使用autowire装配,如果不能,就通过byType装配

spring事务实现方式有哪些

声明式事务:声明式事务也分为两种,基于xml配置文件的方式和注解方式(在类上加@Transaction注解)

编码方式:提供编码的形式管理和维护事务

说一下spring的事务隔离

spring有5大事务隔离级别,默认值为isolation_default(使用数据库的配置),其他四个隔离级别和数据库一致

  1. ISOLATION_DEFAULT:底层数据库的设置隔离级别,数据库设置什么我就用什么
  2. ISOLATIONREADUNCOMMITTED:未提交读,最低隔离级别、事务未提交前,就可被其他事务读取(会出现幻读、脏读、不可重复读);
  3. ISOLATIONREADCOMMITTED:提交读,一个事务提交后才能被其他事务读取到(会造成幻读、不可重复读),SQL server 的默认级别;
  4. ISOLATIONREPEATABLEREAD:可重复读,保证多次读取同一个数据时,其值都和事务开始时候的内容是一致,禁止读取到别的事务未提交的数据(会造成幻读),MySQL 的默认级别;
  5. ISOLATION_SERIALIZABLE:序列化,代价最高最可靠的隔离级别,该隔离级别能防止脏读、不可重复读、幻读。

脏读 幻读 不可重复读

  • 脏读:表示一个事务还能读取另一个事务还未提交的数据;比如某个事务尝试插入记录 A,此时该事务还未提交,然后另一个事务尝试读取到了记录 A。
  • 不可重复读:是指在同一事务内,多次读同一数据
  • 幻读:指同一事务内多次查询返回的结果集不一样,比如同一个事务 A 第一次查询时候有 n 条记录,但是第二次同等条件下查询却有 n+1条记录,这就好像产生了幻觉。发生幻读的原因也是另外一个事务新增或者删除或者修改了第一个事务结果集里面的数据,同一个记录的数据内容被修改了,所有数据行的记录就变多或者变少了。

Spring MVC

说一下springmvc的运行流程

  1. springmvc先将请求发送给dispatcherServlet
  2. DispatcherServlet查询一个或多个HandlerMapping,找到处理请求的Controller
  3. DispatcherServlet再把请求提交给对应的Controller
  4. Controller进行相应的业务逻辑处理后,返回一个ModelAndView
  5. DiepatcherServlet查询一个或多个ViewRelover视图解析器,找到 ModelAndView 对象指定的视图对象。
  6. 视图对象负责渲染返回给客户端。

springmvc有哪些组件

  • 前置控制器 DispatcherServlet
  • 映射控制器:HandlerMapper
  • 处理器适配器(HandlerAdapter)
  • 模型和视图:ModelAndView
  • 视图解析器:ViewResolver

@RequestMapping的作用是什么

将http的请求映射到相应的类/方法上

@Autowired的作用是什么

@Autowired它可以对类成员变量,方法及构造函数进行标注,完成自动装配的工作,通过@Autowired的使用来消除set/get方法

SpringMVC常用注解

  1. requestMapping用于请求url映射
  2. RequestBody 注解实现接受http请求的json数据,将json数据转换为java对象
  3. ResponseBody 注解实现将Controller方法返回对象转化为json响应给客户

如何开启注解处理器和适配器

在springmvc-xml中通过开启mvc:annotation-driven来实现注解处理器和适配器的开启

如何解决get和post乱码问题

  1. 配置charactEncodingFilter过滤器为UTF-8可以解决post乱码
  2. 解决get请求乱码:修改tomcat配置文件server.xml编码与工程编码一致,即在Connector标签中添加URIEncoding=UTF-8
    对参数重新编码
CopyString username = New String(request.getParameter.getBytes("IOS-8859-1"),UTF-8);

SpringBoot

什么是springBoot

spingboot是为spring服务的,是用来简化新spring应用的初始化搭建以及开发过程

为什么要使用springboot

  1. 配置简单
  2. 独立运行
  3. 自动装配
  4. 无代码生成和xml配置
  5. 提供应用监控
  6. 易上手
  7. 提升开发效率

Spring boot的配置原理

在spring程序的main方法里有一个@SpingBootApplication,这个注解是和混合注解,里面还有个@EnableAutoConfiguration会自动去maven中读取每个starter中的spring.factories文件,该配置文件配置了所有需要被创建spring容器中的bean

spring boot 常用的starter有哪些

spring-boot-starter-web 嵌入tomcat和web开发需要servlet与jsp支持

spring-boot-starter-data-jpa 数据库支持

spring-boot-starter-data-redis redis数据库支持

spring-boot-starter-data-solr solr支持

mybatis-spring-boot-starter 第三方的mybatis集成starter

spring boot 核心配置文件是什么

  • bootstrap(.yml或.properties):bootstrap由父ApplicationContext加载的,比application优先加载,且bootstrap里面的属性不能覆盖
  • application (. yml 或者 . properties):用于 spring boot 项目的自动化配置。

spring boot 配置文件有哪几种类型?它们有什么区别

配置文件有 . properties 格式和 . yml 格式,它们主要的区别是书法风格不同。

. properties 配置如下:

Copyspring. RabbitMQ. port=5672

. yml 配置如下:

Copyspring:
    RabbitMQ:
        port: 5672

. yml 格式不支持 @PropertySource 注解导入。

spring boot有哪些方式实现热部署

  • 使用 devtools 启动热部署,添加 devtools 库,在配置文件中把 spring. devtools. restart. enabled 设置为 true;
  • 使用 Intellij Idea 编辑器,勾上自动编译或手动重新编译。

Jpa和Hibernate有什么区别

jap全称java Persistence API ,是Java持久化接口的规范,Hibernate属于具体的实现

Spring Cloud

什么是SpringCloud

spring cloud 是一系列框架的有序集合。它利用 spring boot 的开发便利性巧妙地简化了分布式系统基础设施的开发,如服务发现注册、配置中心、消息总线、负载均衡、断路器、数据监控等,都可以用 spring boot 的开发风格做到一键启动和部署。

什么是微服务

以前的模式是所有的代码在同一个工程中,部署在同一个服务器中,同一个模块不同的功能互相抢占资源

微服务 将工程根据不同的业务规则拆分成微服务 微服务部署在不同的机器上,服务之间互相调用

微服务的框架有dubbo(只能用来做微服务),SpringCloud(提供了服务发现,断路器)

在分布式架构中,断路器模式的作用也是类似的,当服务单元发生故障(类似用电器发生短路)之后,通过断路器的故障监控(类似熔断保险丝),向调用方返回一个错误响应码,而不是长时间的等待,这样就不会使得线程因调用故障服务被长时间占用不释放,避免了故障在分布式系统中的蔓延。

springcloud的核心组件有哪些

  • Eureka:服务注册于发现
  • Feign:基于动态代理机制,根据注解和选择的机制,拼接请求url,发起请求
  • Ribbon:实现负载均衡,从一个服务的多个机器中选择一台
  • Hystrix:提供线程池,不同的服务走不同的线程池,实现了不同服务调用的隔离,避免了服务雪崩的问题
  • Zuul:网关管理,由Zuul网关转发请求给对应的服务

Spring Cloud如何实现服务的注册和发现

服务在发布时,指定对应的服务名(服务名包括ip和端口)将服务注册到注册中心(eureka或者zookeeper)

这一过程是springcloud自动实现 只需要在main方法添加@EnableDisscoveryClient 同一个服务修改端口就可以启动多个实例

调用方法:传递服务名称通过注册中心获取所有的可用实例 通过负载均衡策略调用(ribbon和feign)对应的服务

Ribbon和Feign的区别

ribbon(略奔)

Ribbon添加maven依赖 spring-starter-ribbon 使用@RibbonClient(value="服务名称") 使用RestTemplate调用远程服务对应的方法

feign添加maven依赖 spring-starter-feign 服务提供方提供对外接口 调用方使用 在接口上使用@FeignClient("指定服务名")

Ribbon和Feign都是用于调用其他服务的,不过方式不同

  1. 启动类使用的注解不同,Rabbon使用的是@RibbonClient,Feign用的是@EnableFeignClients
  2. 服务指定的位置不同,Ribbon是在@RibbonClient注解上声明,Feign则是在定义抽象方法的接口中使用@FeignClient声明。
  3. 调用方式不同,Ribbon需要自己构建http请求,模拟http请求然后使用RestTemplate发送给其他服务,步骤相当繁琐。
  4. Feign则是在Ribbon的基础上进行一次改进,采用接口的方式,将需要调用的其他服务定义成抽象方法即可,不需要自己构建Http请求,不过要注意的是抽象方法的注解、方法签名要和提供服务的方法完全一致。

SpringCloud断路器的作用

当一个服务调用另一个服务由于网络原因或者自身原因出现问题,调用者就会等待被调用者的响应 当更多的服务请求到这些资源时导致更多的请求等待 这样就会发生连锁效应(雪崩效应) 断路器就是解决这一问题。

  • 断路器有完全打开状态

    ​ 一定时间内 达到一定的次数无法调用 并且多次检查没有恢复的迹象 断路器完全打开,那么下次请求就不会请求到该服务

  • 半开:

    短时间内 有恢复迹象 断路器会将部分请求发给该服务 当能正常调用时 断路器关闭

  • 关闭

    当服务一直处于正常状态,能正常调用,断路器关闭

posted @ 2020-08-23 23:39  柒丶月  阅读(132)  评论(0编辑  收藏  举报