基础记录学习

第一阶段

day01

java编译运行过程:

1) 编译期.java源文件,经过编译,生成.class字节码文件

2) 运行期JVM加载.class并运行.class

特点: 跨平台、一次编程到处使用

1) JVM: java虚拟机------------->专门翻译.class文件的软件

加载.class并运行.class

2) JRE: java运行环境------------>JRE Java Runtime Enviroment/ɪnˈvaɪrənmənt/的简称,即Java 运行时环境,它是Java程序运行所必须的环境集合,主要由Java虚拟机Java平台核心类和若干支持文件组成。比如rt.jar文件,rtruntime的缩写,包含了编程所使用的核心APIjava.lang等),也就是核心类库编译后的class文件。还有一些配置文件,像java.util.logging的日志配置文件logging.properties

除了包含JVM以外,还包含了运行java程序所必须的环境

JRE = JVM+java系统类库(小零件)

3) JDK: java开发工具包---------->JDK Java Development/dɪˈveləpmənt

/ ToolKit/ˈtuːlkɪt

/ 的简称,也就是 Java 开发工具包。JDK 是整个 Java 的核心,包括 Java 运行环境(Java Runtime Envirnment,简称 JRE),Java 工具(比如 javacjavajavap 等等),以及 Java 基础类库(比如 rt.jar)。

最主流的 JDK Oracle 公司发布的 JDK,除了 Oracle JDK(商业化,更稳定)之外,还有很多公司和组织开发了属于自己的 JDK,比较有名的有 IBM JDK(更适合 IBM OpenJDK(开源的)。每个 JDK 都有自己的优缺点,我们开发者只需要掌握 Oracle JDK 就好

JDK 有三种类型。

1J2SEStandard Edition,标准版,是我们通常用的一个版本,从 JDK 5.0 开始,改名为 Java SE

2J2EEEnterprise Edition,企业版,从 JDK 5.0 开始,改名为 Java EE

3J2MEMicro Edition,主要应用于移动设备、嵌入式设备,从 JDK 5.0 开始,改名为 Java ME

除了包含JRE以外,还包含了开发java程序所必须的命令工具

JDK=JRE+编译、运行等命令工具

结论:

1) 运行java程序的最小环境为JRE

2) 开发java程序的最小环境为JDK

SDK API 有什么区别?

在开发中如果对方给你提供一个接口,这就是API;

开发中如果是一个工程提供另一个工程接口,这就是SDK;

SDK java JDK有什么区别?

JAVA JDK sdk 的一个子集,sdk 更加的广泛包含的更大。

JDKJava Development Kit,中文译为Java开发工具包

SDK本质是什么?

开发工具的集合,SDK编程就是直接用windows API进行编程。

开发步骤:

2.1) 新建Java项目/ 工程---------------------------小区

2.2) 新建Java--------------------------------------+单元

2.3) 新建Java--------------------------------------房子

项目(包含)------->(包含)------->(包含)-------->代码

符合包含关系 , 举什么例子都可以

day02

八大基本类型从小到大:byte, short, int, long, float, double, char, boolean

int:整型,4个字节,-21个多亿到21个多亿

Java

1)整数直接量默认为int类型,但不能超出范围,若超范围则发生编译错误

2)两个整数相除,结果还是整数,小数位无条件舍弃(不会四舍五入)

3)运算时若超出范围,则发生溢出(溢出不是错误,但是需要避免)

long:长整型,8个字节

double:浮点型,8个字节

boolean: 布尔型,1个字节

char:字符型,2个字节

Java

1)采用的是Unicode字符集编码格式,一个字符对应一个码

 表现的形式为字符char,但本质上是码int(065535之间)

2)字符型直接量必须放在单引号中,只能有一个

3)特殊符号需要通过\来转义

(ASCII: 'a'--97  'A'--65  '0'--48)

命名

1)只能包含字母、数字、_$符,不能以数字开头

2)严格区分大小写

3)不能使用关键字

4)允许中文命名,但不建议,建议"英文的见名知意"

类型间的转换:

两种方式:

1)自动/隐式类型转换:小类型到大类型

2)强制类型转换:大类型到小类型

 ----语法: (要转换成为的数据类型)变量

 ----强转有可能发生溢出或丢失精度

两点规则:

1)整数直接量可以直接赋值给byteshortchar,但不能超范围

2)byte,short,char型数据参与运算时,系统将其一律先转为int再运算

day03

三目运算:

            1)语法:

                boolean?1:2

            2)执行过程:

              ---计算boolean的值:

                 2.1)若为true,则整个表达式的结果为?号后的数1

                 2.2)若为false,则整个表达式的结果为:号后的数2
       

  1.  int num = 5;
  2.         int flag = num>0?1:-1;
  3. System.out.println(flag); //1

       

if语句特殊写法:

  1.  if(true)
  2.             System.out.println(num + "是偶数");

 常见面试题:     

  1.   short s = 5;
  2.         s = s+10; 编译错误,需强转,改为:s=(short)(s+10);
  3.         s += 10; 正确,相当于:s=(short)(s+10);

 ++/--:自增1/自减1,可在变量前也可在变量后

            1)单独使用时,在前在后都一样

            2)被使用时,在前在后不一样

                    a++的值为a----------(a--的值为a)

                    ++a的值为a+1--------(--a的值为a-1)

经典面试题:

  1. char类型变量能不能储存一个中文的汉字,为什么?
  1. 能否在不进行强制转换的情况下将一个 double 值赋值给 long 类型的变量?若需要  谁强转谁?
  1. boolean b = (3*0.1 == 0.3);

System.out.println(b):

控制台结果是true还是false?

day04

循环三要素:

  1. 循环变量的初始化
  1. 循环的条件(以循环变量为基础)
  1. 循环变量的改变(向着循环的结束变)
  1. 循环变量:在整个循环过程中所反复改变的那个数
  1. 循环结构:

while结构:先判断后执行,有可能一次都不执行

do...while结构:先执行后判断,至少执行一次

要素1与要素3相同时,首选do...while

变量的作用域/范围:从变量的声明开始,到包含它最近的大括号结束

  1. Math.random()------------------0.00.9999999999999999...
  2. *1000--------------------------0.0999.99999999999999...
  3. +1-----------------------------1.01000.9999999999999...
  4. (int)--------------------------11000    
  5. (int)--------------------------0999
  6. +1-----------------------------11000

day05

  1. 循环结构:
  1. for结构: 应用率最高,与次数相关的循环

三种循环结构如何选择:

1)先判断循环是否与次数相关:

 1.1)若相关---------------------------直接上for

 1.2)若无关,再判断要素1与要素3是否相同:

     1.2.1)若相同---------------------直接上do...while

     1.2.2)若不同---------------------直接上while

break:跳出循环

continue:跳过循环体中剩余语句而进入下一次循环

  1. 经典基础面试题(全国):
  2.  1-100之间的质数(质数又称素数。一个大于1的自然数,除了1和它自身外,不能被其他自然数整除的数叫做质数;否则称为合数。)
  3. 思路:
  4. 1、外层循环作为被除数,内层循环作为除数。
  5. 2、定义一个Boolean,标记外层循环数是否为质数。默认为 true
  6. 3、内层循环结束,如果开关还为true。即被除数为质数,打印出来。
  7.   boolean flag;
  8. 在编程中,标记(flag)是一个预先确定的位,或具有二进制值的位序列。一般地,程序用标记来记忆某件事或者用它来为另一个程序留下记号。
  9.         for (int i = 2; i <= 100; i++) {
  10.             flag=true;
  11.             for (int j = 2; j < i; j++) {
  12.                 if (i % j == 0) {
  13.                     flag=false;
  14.                     break;
  15.                 }
  16.             }
  17.             if (flag) {
  18.                 System.out.println(i);
  19.             }
  20.         }
  21.     }
  22. }
  23.  水仙花数
  24.  水仙花数(Narcissistic number)也被称为超完全数字不变数(pluperfect digital invariant, PPDI)、自恋数、自幂数、阿姆斯壮数或阿姆斯特朗数(Armstrong number),水仙花数是指一个 3 位数,它的每个位上的数字的 3次幂之和等于它本身(例如:1^3 + 5^3+ 3^3 = 153)。
  25.   for(int num=100;num<1000;num++){
  26.             int a=num/100;
  27.             int b=num/10%10;
  28.             int c=num%10;
  29.             if(a*a*a+b*b*b+c*c*c==num){
  30.                 System.out.println(num);

day06

数组的复制:

是重新在内存中开辟一个存储空间

  1. System.arraycopy(a,1,b,0,4);
  2. int[] b = Arrays.copyOf(a,6);
  3.  a = Arrays.copyOf(a,a.length+1); 扩容

数组的排序:

Arrays.sort(arr); arr进行升序排列

方法的定义:五要素(很重要)

  1. 修饰词  返回值类型  方法名(参数列表){
  2.    方法体
  3. }

return

1)return ; //1.1)结束方法的执行 1.2)返回结果给调用方------写在有返回值的方法中

2)return;    //2.1)结束方法的执行-------------------------写在无返回值的方法中

return breakcontinue的区别和作用

1. return关键字并不是专门用于跳出循环的,return的功能是结束一个方法。 一旦在循环体内执行到一个return语句,return语句将会结束该方法,循环自然也随之结束。与continuebreak不同的是,return直接结束整个方法,不管这个return处于多少层循环之内。

2.continue的功能和break有点类似,区别是continue只是中止本次循环,接着开始下一次循环。而break则是完全中止循环。

3.break用于完全结束一个循环,跳出循环体。不管是哪种循环,一旦在循环体中遇到break,系统将完全结束循环,开始执行循环之后的代码。 break不仅可以结束其所在的循环,还可结束其外层循环。此时需要在break后紧跟一个标签,这个标签用于标识一个外层循环。Java中的标签就是一个紧跟着英文冒号(:)的标识符。且它必须放在循环语句之前才有作用

// 外层循环,outer作为标识符

    outer:

    for (int i = 0 ; i < 5 ; i++ ){

      // 内层循环

      for (int j = 0; j < 3 ; j++ ){

        System.out.println("i的值为:" + i + " j的值为:" + j);

        if (j == 1){

          // 跳出outer标签所标识的循环。

           break outer;

        }

      }

    }

拓展知识:

(主调函数, 被调函数 , 回调函数)

函数的英文是function,有功能的意思,函数的作用在于合理分配功能,增强程序的可读性。合理分解功能,降低程序的复杂性。隐藏函数内部的数据和实现,尽可能将问题局限于函数本身。

主调函数与被调函数是成对出现的。是主动与被动的关系。现在有AB两个函数,A函数调用了B函数,那么,A函数就是主调函数,B函数就是被调函数。这和现实生活中的打电话是一样的,一个是主叫,一个是被叫。

回调函数就是一个被作为参数传递的函数。所谓的回调,就是程序员A写了一段程序(程序a),其中预留有回调函数接口,并封装好了该程序。程序员B要让a调用自己的程序b中的一个方法,于是,他通过a中的接口回调自己b中的方法。

  1. 下面是例子。
  2.       1.  首先定义一个类Caller,按照上面的定义就是程序员A写的程序a,这个类里面保存一个接口引用。
  3. public class Caller {  
  4.     private MyCallInterface callInterface;  
  5.       
  6.     public Caller() {  
  7.     }  
  8.       
  9.     public void setCallFunc(MyCallInterface callInterface) {  
  10.         this.callInterface = callInterface;  
  11.     }  
  12.       
  13.     public void call() {  
  14.         callInterface.printName();  
  15.     }  
  16. }  
  17.        2.  当然需要接口的定义,为了方便程序员B根据我的定义编写程序实现接口。
  18.   
  19. public interface MyCallInterface {  
  20.     public void  printName();  
  21. }  
  22.       3.  第三是定义程序员B写的程序b
  23. public class Client implements MyCallInterface {  
  24.   
  25.     @Override  
  26.     public void printName() {  
  27.         System.out.println("This is the client printName method");  
  28.     }  
  29. }  
  30.        4.  测试如下
  31. public class Test {  
  32.     public static void main(String[] args) {  
  33.         Caller caller = new Caller();  
  34.         caller.setCallFunc(new Client());  
  35.         caller.call();  
  36.     }  
  37. }  
  38. 看到这脑子里大概会有一个模糊的框或者概念 什么是回调函数
  39.         5.  在测试方法中直接使用匿名类,省去第3
  40. public class Test {  
  41.     public static void main(String[] args) {  
  42.         Caller caller = new Caller();  
  43. //      caller.setCallFunc(new Client());  
  44.         caller.setCallFunc(new MyCallInterface() {  
  45.             public void printName() {  
  46.                 System.out.println("This is the client printName method");  
  47.             }  
  48.         });  
  49.         caller.call();  
  50.     }  
  51. }  

OopDay01

什么是类?什么是对象?

1)现实生活是由很多很多对象组成的,基于对象抽出了类

2)对象:软件中真实存在的单个个体/东西

 :类型/类别,代表一类个体

3)类是对象的模板,对象是类的具体的实例

类中可以包含:

1)对象所共有的属性/特征------------成员变量

2)对象所共有的行为/动作------------方法

一个类可以创建几个对象?( 多个 )

如何创建类?

Java中定义类的通用格式:修饰符 class 类名{成员,方法}

理解:修饰符是可选的,有public(公共)或不加;

class——关键词,必须有;

类名——首字母大写,且按驼峰命名规则起名,必须有;

成员——有成员属性和成员方法。

如何创建对象?

new语句创建对象,这是最常用的创建对象的方式。(目前你们知道这个就可以了)

如何访问成员?

Java中通过类的实例(即对象)(.)运算符来访问类的成员变量和方法。

OO:面向对象

OOA:面向对象分析

OOD:面向对象设计

OOAD:面向对象分析与设计

OOP:面向对象编程----------------你们所参与的

数据类型  引用类型变量  指向       对象

Student      zs       =    new Student();

OopDay02

方法的签名:方法名+参数列表

  1. 方法的重载(overload)---------------------大大简化方法的调用
  1. 发生在同一类中,方法名相同,参数列表不同
  1. 编译器在编译时会根据方法的签名自动绑定调用的方法
  1. 构造方法:构造函数、构造器、构建器
  1. 给成员变量赋初值
  1. 与类同名,没有返回值类型(void都没有)
  1. 在创建(new)对象时被自动调用

若自己不写构造方法,则编译器默认提供一个无参的构造方法,

  1. 若自己写了构造方法,则不再默认提供
  1. 构造方法可以重载

this:指代当前对象,哪个对象调用方法它指的就是哪个对象

只能用在方法中,方法中访问成员变量之前默认有个this.

this的用法:

1)this.成员变量名-----------访问成员变量(成员变量与局部变量同名时,访问成员变量时this不能省略)

2)this.方法名()-------------调用方法(一般不用)

3)this()-------------------调用构造方法(一般不用)

null:表示空,没有指向任何对象

若引用的值为null,则该引用不能再进行任何点操作了,

若操作则发生NullPointerException空指针异常

OopDay03

继承:

1)作用:代码复用

2)通过extends来实现继承

3)超类/父类:派生类所共有的属性和行为

 派生类/子类:派生类所特有的属性和行为

4)派生类可以访问:派生类的+超类的,但超类只能访问超类的

5)一个超类可以有多个派生类,一个派生类只能有一个超类-----单一继承

6)继承具有传递性

7)java规定:构造派生类之前必须先构造超类

 7.1)在派生类的构造方法中,若自己没有调用超类的构造方法

     ----则编译器默认super()调用超类的无参构造方法

 7.2)在派生类的构造方法中,若自己调用了超类的构造方法

     ----则不再默认提供

 注意:super()调用超类构造方法,必须位于派生类构造方法的第一行

super:指代当前对象的超类对象

1)super.成员变量名-------------------访问超类的成员变量

2)super.方法名()--------------------调用超类的方法----------明天下午讲

3)super()--------------------------调用超类的构造方法

OopDay04

1.向上造型:

 1)超类型的引用指向派生类的对象

 2)能点出来什么,看引用的类型-----------------这是规定,记住就可以了

2.方法的重写(override):重新写、覆盖

 1)发生在父子类中,方法名相同,参数列表相同

 2)重写方法被调用时,看对象的类型--------------这是规定,记住就可以了

3.重写与重载的区别:

重写:发生在父子类中,方法名相同参数列表相同,方法体不同

重写遵循两同两小一大

1)两同:方法名相同,参数列表相同

2)两小:A.子类方法的返回值小于等于父类方法的返回值

a.void/基本类型返回值必须相同

b.引用类型的返回值小于等于父类的返回值

B.子类方法抛出的异常小于或等于超类的方法抛出的异常

3)一大:子类方法的访问权限大于或等于父类方法的

重载: 1. 在同一类中,方法名称相同,参数列表不同,方法体不同,但做的事是一个类型的

            2.与方法的返回值类型无关

OopDay05

final关键字

final是最终的意思,可以修饰变量,方法,

final修饰的变量是不能被改变

fianl修饰的方法不能被重写

final修饰的类不能被继承

static关键字

static是静态的意思,常用于修饰变量和方法,当然也可以修饰代码块和内部类.

静态的特点:

1.随着类的加载而加载

2.优先于对象存在

3.被所有对象所共享

4.可以被类名点直接调用

注意事项:

A为什么静态方法只能访问静态成员?

因为静态的内容是随着类的加载而加载,他是先进入内存中的

B.静态方法中不能使用this,super关键字(静态方法中没有隐式的this)

静态变量和实例变量的区别

1.调用方式:

静态变量也称为类变量,可以直接通过类名点调用,也可以通过对象名点(引用)调用[但是不建议].这个变

量是属于类的.

实例变量只能通过对象名(引用)点调用.这个变量属于对象的.

2.储存方式:

静态变量存储在方法区中

成员变量储存在堆中

3.生命周期

静态变量随着类的加载而存在,随着类的消失而消失,生命周期较长

成员变量随着对象的创建而存在,随着对象的消失而消失

4.与对象的相关性

静态变量是所有对象共享的数据

实例变量是每个对象所特有的数据

OopDay06

static

1)静态变量:

 1.1)static修饰

 1.2)属于类,存储在方法区中,只有一份

 1.3)常常通过类名点来访问

 1.4)何时用:所有对象所共享的数据(图片、音频、视频等)

2)静态方法:

 2.1)static修饰

 2.2)属于类,存储在方法区中,只有一份

 2.3)常常通过类名点来访问

 2.4)静态方法中没有隐式this传递,所以静态方法中不能直接访问实例成员

 2.5)何时用:方法的操作与对象无关

3)静态块:

 3.1)static修饰

 3.2)属于类,在类被加载期间自动执行,一个类只被加载一次,所以静态块也只执行一次

 3.3)何时用:加载/初始化静态资源(图片、音频、视频等)

static final常量:

1)必须声明同时初始化

2)常常通过类名点来访问,不能被改变

3)建议:常量名所有字母都大写,多个单词用_分隔

4)编译器在编译时会将常量直接替换为具体的值,效率高

5)何时用:数据永远不变,并且经常使用

抽象方法:

1)abstract修饰

2)只有方法的定义,没有具体的实现({}都没有)

抽象类:

1)abstract修饰

2)包含抽象方法的类必须是抽象类

3)抽象类不能被实例化(new对象)

4)抽象类是需要被继承的,派生类:

 4.1)重写所有抽象方法------------常用

 4.2)也声明为抽象类--------------不常用

5)抽象类的意义:

 5.1)封装共有的属性和行为----------代码复用

 5.2)为所有派生类提供统一的类型-----向上造型

 5.3)包含抽象方法,为所有派生类提供统一的入口(能点出来)

     派生类的具体实现不同,但入口是一致的

1.:抽象方法是不是多余的呢?(派生类总归还得重写step(),抽象方法达不到让派生类复用的效果)

 :不是的,抽象方法存在的意义在于,当向上造型时通过超类的引用能点出step()

     

2.:既然抽象方法的意义仅仅在于造型后能点出来,那设计为普通方法也能点,为何非要设计为抽象方法?

 :设计为普通方法,则派生类可以重写也可以不重写,而设计为抽象方法,可以强制派生类必须重写

    ----抽象方法可以达到强制必须重写的目的

OopDay07

  1. 成员内部类:应用率低类中套类,外面的称为外部类,里面的称为内部类
  2. 内部类通常只服务于外部类,对外不具备可见性
  3. 内部类对象通常在外部类中创建

内部类中可以直接访问外部类的成员(包括私有的)

----内部类中有个隐式的引用指向了创建它的外部类对象-------外部类名.this--------必须记住(API时用)

  1. 匿名内部类:-----------------大大便于访问数据(明天做案例时才能体会)
  2. 若想创建一个类(派生类)的对象,并且对象只被创建一次,此时该类不必命名,称为匿名内部类
  3. 匿名内部类中不能修改外面变量的值,因为在此处系统默认该变量为final---------必须记住(API时用)

面试题:

: 内部类有独立的.class吗?

  1. :

OopDay08

接口

引用类型interface定义,常量和抽象方法,不能被实例化,需要被实现,实现类必须重写所有抽象方法

  1. 一个类可以实现多个接口,用逗号分隔,若又继承又实现时,应先继承后实现,接口继承接口

多态

1) 意义:行为多态、对象多态

2) 向上造型:可以向上造型为: 超类+所实现的接口

3) 强制类型转换,成功的条件:

3.1) 引用所指向的对象,就是该类型

3.2) 引用所指向的对象,实现了该接口或继承了该类

  1. 4) 强转若不满足如上条件,则发生ClassCastException类型转换异常,建议强转之前先用instanceof判断

抽象类和接口的区别(面试题)

1.抽象类是由abstract class修饰的,接口是由interface修饰的

2.抽象类中可以有成员变量,也可以有常量,接口中都是常量

3.抽象类中可以有普通方法也可以有抽象方法,接口中都是抽象方法(Java8以后可以有普通方法,但是普 通方法必须由staticdefault修饰的)

4.抽象类中可以定义构造方法,接口中不能定义构造方法,但是都没有意思,因为不能被实例化 5.抽象类中任何访问控制修饰符都可以,接口中访问控制修饰符只能为public

OopDay09

  1. 内存管理:由JVM来管理的

堆:

1) 存储new出来的对象(包括实例变量)

2) 垃圾: 没有任何引用所指向的对象

垃圾回收器(GC)不定时到内存中清扫垃圾,回收过程是透明的(看不到的)

并非一看到垃圾就立即回收,通过调用System.gc()建议JVM尽快调度GC来回收

3) 实例变量的生命周期:

创建对象时存储在堆中,对象被回收时一并被回收

4) 内存泄漏:不再使用的对象没有被及时的回收,严重的泄漏会导致系统的崩溃

  1. 建议:不再使用的对象应及时将引用设置为null

栈:

1) 正在调用的方法中的所有局部变量(包括方法的参数)

2) 调用方法时,会在栈中为该方法分配一块对应的栈帧,栈帧中存储局部变量(包括方法参数)

方法调用结束时,栈帧自动被清除,局部变量一并被清除

3) 局部变量的生命周期:

  1. 调用方法时存储在栈中,方法调用结束时与栈帧一并被清除

方法区:

1) 存储.class字节码文件(包括静态变量、所有方法)

  1. 2) 方法只有一份,通过this来区分具体的访问对象

面向对象三大特征:

1.封装:

 1):封装的是对象的属性和行为

 2)方法:封装的是具体的业务逻辑实现

 3)访问控制修饰符:封装的是具体的访问权限

2.继承:

 1)作用:代码复用

 2)超类:所有派生类所共有的属性和行为

   接口:部分派生类所共有的属性和行为

   派生类:派生类所特有的属性和行为

 3)单一继承、多接口实现,传递性

3.多态:

 1)行为多态:所有抽象方法都是多态的(通过方法的重写来实现)

   对象多态:所有对象都是多态的(通过向上造型来实现)

 2)向上造型、强制类型转换、instanceof判断

JavaSE

Git快速入门及使用------->见网址

01. IDEA之Git工具 - SegmentFault 思否

面试题:

String

String  int 类型之间如何转换

int → String

1)String s = i + "";

2)String s = String.valueOf(i);

3)String s = Integer.toString(i);

String → int

1)int i = Integer.parseInt(s);

2)int i=Integer.valueOf(s).intValue();

String s = new String("abc");创建了几个对象

创建了一个或者两个String类型对象

1.new对象的时候,系统会先检测常量池中是否含有"abc"这个字符串对象,如果有,则不在常量池中创

,只在堆中开辟一个空间存放String对象.

2.如果字符串常量池中没有这个字符串对象,则在常量池中和堆中各创建一个对象

String s = new String("ab"+"c");创建了几个对象

创建了一个或者两个String类型对象

"ab"+"c" 在编译时 自动 转变成"abc"

String,StringBuilder,StringBuffer的区别

String,StringBuffer,StringBuilder这三者都是操纵字符串的类;

String 底层是由final修饰的char[],声明的对象不可以被改变;如果每次改变相当于重新生成一个

String对象,并将指针指向新的对象;

StringBuffer,StringBuilder 没有被final修饰,并且提供了一个自动扩容的机制(初始的默认

长度为16),会自动扩容,扩容的长度为原长度的2倍加2,所以在对于拼接字符串效率要比 String

;

StringBuffer,StringBuilder虽然都是可变的字符串,但是StringBuffer是由synchronized

饰的,线程安全,但是效率较低 

你都用过String里的那些方法

  1. 1length():获取字符串的长度,其实也就是字符个数
  2. String str = "adsfaxsdfas沙发上案发地方";
  3. System.out.println(str.length());
  4. 1
  5. 2
  6. 运行结果:
  7. 18
  8. 2charAt(int index):获取指定索引处的字符
  9. String str = "adsfaxsdfas沙发上案发地方";
  10.         
  11. char[] c = {'a','d','s','f','a'};
  12. System.out.println(str.charAt(12));
  13. 1
  14. 2
  15. 3
  16. 4
  17. 运行结果:
  18. 3indexOf(String str):获取str在字符串对象中第一次出现的索引
  19. String str = "adsfaxsdfas沙发上案发地方";
  20. System.out.println(str.indexOf('a',5));
  21. 1
  22. 2
  23. 运行结果:
  24. 9
  25. 4substring(int start):start开始截取字符串
  26. String str = "adsfaxsdfas沙发上案发地方";
  27. System.out.println(str.substring(1));
  28. 1
  29. 2
  30. 运行结果:
  31. dsfaxsdfas沙发上发地方
  32. 5String substring(int start,int end):start开始,到end结束截取字符串。包括start,不包括end
  33. String str = "adsfaxsdfas沙发上案发地方";
  34. System.out.println(str.substring(1, 12));
  35. 1
  36. 2
  37. 运行结果:
  38. dsfaxsdfas
  39. 3.String判断功能
  40. 1equals(Object obj):比较字符串的内容是否相同
  41. String str = "adsfaxsdfas沙发上案发地方";
  42. System.out.println(str.equals("adsfaxsdfas沙发上发地方"));
  43. System.out.println(str.equals("adsfaxsdfas"));
  44. 1
  45. 2
  46. 3
  47. 运行结果:
  48. true
  49. false
  50. 2equalsIgnoreCase(String anotherString):比较字符串的内容是否相同,忽略大小写
  51. String str = "adsfaxsdfas沙发上案发地方";
  52. System.out.println(str.equalsIgnoreCase("ADsfaxsdfAs沙发上发地方"));
  53. 1
  54. 2
  55. 运行结果:
  56. true
  57. 3startsWith(String prefix):判断字符串对象是否以指定的字符开头(区分大小写)
  58. String str = "adsfaxsdfas沙发上案发地方";
  59. System.out.println(str.startsWith("a"));
  60. System.out.println(str.startsWith("A"));
  61. 1
  62. 2
  63. 3
  64. 运行结果:
  65. true
  66. false
  67. 4startsWith(String prefix,int toffset):判断字符串对象是否以指定的字符开头,参数toffset为指定从哪个下标开始
  68. String str = "adsfaxsdfas沙发上案发地方";
  69. System.out.println(str.startsWith("f", 3));
  70. System.out.println(str.startsWith("f", 4));
  71. 1
  72. 2
  73. 3
  74. 运行结果:
  75. true
  76. false
  77. 5endsWith(String str):判断字符串对象是否以指定的字符结尾
  78. String str = "adsfaxsdfas沙发上案发地方";
  79. System.out.println(str.endsWith("x"));
  80. System.out.println(str.endsWith(""));
  81. 1
  82. 2
  83. 3
  84. 运行结果:
  85. false
  86. true
  87. 6isEmpty():判断指定字符串是否为空
  88. 4.String类中的转化方法:
  89. 1toCharArray():把字符串转换为字符数组
  90. public static void main(String[] args) {
  91.         
  92.         String str = "adsfaxsdfas沙发上发地方";
  93.         char arr[] = str.toCharArray();
  94.         printArray(arr);
  95. }
  96. public static void printArray(char a[]) {
  97.         
  98.         for(int i=0;i<a.length;i++) {
  99.             System.out.print(a[i]+"--");
  100.         }
  101. }
  102. 1
  103. 2
  104. 3
  105. 4
  106. 5
  107. 6
  108. 7
  109. 8
  110. 9
  111. 10
  112. 11
  113. 12
  114. 运行结果:
  115. a–d--s–f--a–x--s–d--f–a--s–------
  116. 2toLowerCase():把字符串转换为小写字符串
  117. String str1 = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
  118. System.out.println(str1.toLowerCase());
  119. 1
  120. 2
  121. 运行结果:
  122. abcdefghijklmnopqrstuvwxyz
  123. 3toUpperCase():把字符串转换为大写字符串
  124. String str1 = "abcdefghijklmnopqrstuvwxyz";
  125. System.out.println(str2.toUpperCase());
  126. 1
  127. 2
  128. 运行结果:
  129. ABCDEFGHIJKLMNOPQRSTUVWXYZ
  130. 5.其他常用方法
  131. 1trim():去除字符串两端空格
  132. String str3 = "    a  c  e x u a n x u a n    ";
  133. System.out.println(str3.trim());
  134. System.out.println(str3);
  135. 运行结果:
  136. a  c  e x u a n x u a n
  137.     a  c  e x u a n x u a n   
  138. 1
  139. 2
  140. 3
  141. 4
  142. 5
  143. 6
  144. 7
  145. 2split():去除字符串中指定的的字符,然后返回一个新的字符串
  146. public static void main(String[] args) {
  147.         
  148.         String str = "adsfaxsdfas沙发上发地方";
  149.         String array[] = str.split("a");
  150.         printString(array);
  151. }
  152. public static void printString(String a[]) {
  153.         for(int i=0;i<a.length;i++) {
  154.             System.out.print(a[i]);
  155.         }
  156. }
  157. 1
  158. 2
  159. 3
  160. 4
  161. 5
  162. 6
  163. 7
  164. 8
  165. 9
  166. 10
  167. 11
  168. 12
  169. 运行结果:
  170. dsfxsdfs沙发上发地方
  171. 3subSequence(int beginIndex,int endIndex ):截取字符串中指定位置的字符组成一个新的字符串
  172. String str = "adsfaxsdfas沙发上发地方";
  173. System.out.println(str.subSequence(1, 10));
  174. 1
  175. 2
  176. 运行结果:
  177. dsfaxsdfa
  178. 4replace(char oldChar, char newChar):将指定字符替换成另一个指定的字符
  179. String str = "adsfaxsdfas沙发上发地方";
  180. System.out.println(str.replace('a', 's'));
  181. 1
  182. 2
  183. 运行结果:
  184. sdsfsxsdfss沙发上发地方
  185. 5replaceAll(String regex,String replasement):用新的内容替换全部旧内容
  186. String str4 = "Hello,world!";
  187. System.out.println(str4.replaceAll("l", "&"));
  188. 1
  189. 2
  190. 运行结果:
  191. He&&o,wor&d!
  192. 6replaceFirst(String regex,String replacement):替换首个满足条件的内容
  193. String str = "adsfaxsdfas沙发上发地方";
  194. System.out.println(str.replaceFirst("", ""));
  195. 1
  196. 2
  197. 运行结果:
  198. adsfaxsdfas璇发上发地方
  199. 7lastIndexOf(String str):返回指定字符出现的最后一次的下标
  200. String str4 = "Hello,world!";
  201. System.out.println(str4.lastIndexOf("l"));
  202. 1
  203. 2
  204. 运行结果:
  205. 9
  206. 8contains(CharSequence s):查看字符串中是都含有指定字符
  207. String str4 = "Hello,world!";
  208. System.out.println(str4.contains("l"));
  209. 1
  210. 2
  211. 运行结果:
  212. true
  213. 9concat(String str):在原有的字符串的基础上加上指定字符串
  214. String str5 = "dr";
  215. System.out.println(str5.concat("eam"));
  216. 1
  217. 2
  218. 运行结果:
  219. dream

File

你都用过File里的那些方法?

  1. isDirectory() 是否为文件夹
  2. isFile() 是否为文件
  3. getPath() 得到file的路径
  4. getName() 得到最后一层的名字
  5. getParent() 得到去掉最后一层的路径
  6. getParentFile() 得到父类路径的新文件
  7. renameTo() 改名
  8. mkdir() 创建新文件夹,只能创建一层
  9. mkdirs() 创建新文件夹,可以多层
  10. createNewFile() 创建新文件,只能一层
  11. exists() 路径是否存在
  12. delete() 删除文件或者目录(为空的目录)
  13. list() 返回该路径下文件或者文件夹的名字数组
  14. listFiles() 返回该路径下文件或者文件夹组成的File数组
  15. separator 代替文件或文件夹路径的斜线或反斜线,防止跨平台出现错误
     

FileFilter

boolean accept(File pathname)
测试指定的抽象路径名是否应包含在路径名列表中。

IO

1IO的分类有哪些?

按流向分为输入流和输出流

按类型分为字节流和字符流

按功能分为节点流和处理流

2、字节流和字符流的区别?

字节流:处理除了文字之外的数据,无缓冲区,每传一次数据就会打开一次文件

字符流:处理文字和字符的数据,有缓存区,使用缓冲区打开文件次数少

3、节点流和处理流的区别?

节点流:可以从某节点读数据或向某节点写数据的流。如 FileInputStream

处理流:对节点流的包装,有比较多的功能

4、谈谈Java IO里面的常见类,字节流,字符流、接口、实现类、方法阻塞

答:输入流就是从外部文件输入到内存,输出流主要是从内存输出到文件。

IO里面常见的类,第一印象就只知道IO流中有很多类,IO流主要分为字符流和字节流。字符流中有抽象类InputStreamOutputStream,它们的子类FileInputStreamFileOutputStream,BufferedOutputStream等。字符流BufferedReaderWriter等。都实现了Closeable, Flushable, Appendable这些接口。程序中的输入输出都是以流的形式保存的,流中保存的实际上全都是字节文件。

java中的阻塞式方法是指在程序调用改方法时,必须等待输入数据可用或者检测到输入结束或者抛出异常,否则程序会一直停留在该语句上,不会执行下面的语句。比如read()readLine()方法。

5、字符流和字节流有什么区别?

要把一片二进制数据数据逐一输出到某个设备中,或者从某个设备中逐一读取一片二进制数据,不管输入输出设备是什么,我们要用统一的方式来完成这些操作,用一种抽象的方式进行描述,这个抽象描述方式起名为IO流,对应的抽象类为OutputStreamInputStream ,不同的实现类就代表不同的输入和输出设备,它们都是针对字节进行操作的。

在应用中,经常要完全是字符的一段文本输出去或读进来,用字节流可以吗?

计算机中的一切最终都是二进制的字节形式存在。对于中国这些字符,首先要得到其对应的字节,然后将字节写入到输出流。读取时,首先读到的是字节,可是我们要把它显示为字符,我们需要将字节转换成字符。由于这样的需求很广泛,人家专门提供了字符流的包装类。

底层设备永远只接受字节数据,有时候要写字符串到底层设备,需要将字符串转成字节再进行写入。字符流是字节流的包装,字符流则是直接接受字符串,它内部将串转成字节,再写入底层设备,这为我们向IO设别写入或读取字符串提供了一点点方便。

序列化

说说对序列化的理解,是怎么实现的,什么场景下需要它 ?

序列化是指将Java对象转换为字节序列的过程

反序列化则是将字节序列转换为Java对象的过程。

Java对象序列化必须实现Serializable接口,使得对象能通过网络传输或者文件储存等方式传输.

质上讲,序列化就是把实体对象状态按照一定的格式写入到有序字节流,反序列化就是从有序字节流重建

对象,恢复对象状态。

Java IO Sammary(总结)

1)明确源和目的。

数据source:就是需要读取,可以使用两个体系:InputStreamReader

数据destination:就是需要写入,可以使用两个体系:OutputStreamWriter

2)操作的数据是否是纯文本数据?

如果是:

           数据sourceReader

           数据destinationWriter 

   如果不是:

           数据sourceInputStream

           数据destinationOutputStream

3Java IO体系中有太多的对象,到底用哪个呢?

明确操作的数据设备。

数据source对应的设备:硬盘(File),内存(数组),键盘(System.in)

数据destination对应的设备:硬盘(File),内存(数组),控制台(System.out)

记住,只要一读取键盘录入,就用这句话。

BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in));

反之

BufferedWriter bufw = new BufferedWriter(new OutputStreamWriter(System.out));

异常处理机制

1.运行时异常

定义:RuntimeException及其子类都被称为运行时异常。

特点:Java编译器不会检查它。也就是说,当程序中可能出现这类异常时,倘若既"没有通过throws声明抛出它",也"没有用try-catch语句捕

获它",还是会编译通过。例如,除数为零时产生的ArithmeticException异常,数组越界时产生的IndexOutOfBoundsException异常,failfast机制产生的ConcurrentModi?cationException异常(java.util包下面的所有的集合类都是快速失败的,快速失败也就是fail-fast,它是

Java集合的一种错误检测机制。当多个线程对集合进行结构上的改变的操作时,有可能会产生fail-fast机制。记住是有可能,而不是一定。

例如:假设存在两个线程(线程1、线程2),线程1通过Iterator在遍历集合A中的元素,在某个时候线程2修改了集合A的结构(是结构上面的修改,而不是简单的修改集合元素的

内容),那么这个时候程序就会抛出ConcurrentModi?cationException 异常,从而产生fail-fast机制,这个错叫并发修改异常。Fail-safejava.util.concurrent包下面的所有的类都是安全失败的,在遍历过程中,如果已经遍历的数组上的内容变化了,迭代器不会抛出

ConcurrentModi?cationException异常。如果未遍历的数组上的内容发生了变化,则有可能反映到迭代过程中。这就是

ConcurrentHashMap迭代器弱一致的表现。ConcurrentHashMap的弱一致性主要是为了提升效率,是一致性与效率之间的一种权衡。要成为强一致性,就得到处使用锁,甚至是全局锁,这就与Hashtable和同步的HashMap一样了。)等,都属于运行时异常。

常见的五种运行时异常:

ClassCastException(类转换异常)

IndexOutOfBoundsException(数组越界)

NullPointerException(空指针异常)

ArrayStoreException(数据存储异常,操作数组是类型不一致)

Bu?erOver?owException

2.被检查异常

定义:Exception类本身,以及Exception的子类中除了"运行时异常"之外的其它子类都属于被检查异常。特点 : Java编译器会检查它。此类异常,要么通过throws进行声明抛出,要么通过try-catch进行捕获处理,否则不能通过编译。例如,CloneNotSupportedException就属于被

检查异常。当通过clone()接口去克隆一个对象,而该对象对应的类没有实现Cloneable接口,就会抛出CloneNotSupportedException异常。被检查异常通常都是可以恢复的。

如:

IOException

FileNotFoundException

SQLException

被检查的异常适用于那些不是因程序引起的错误情况,比如:读取文件时文件不存在引发的FileNotFoundException 。然而,不被检查的异

常通常都是由于糟糕的编程引起的,比如:在对象引用时没有确保对象非空而引起的 NullPointerException

3.错误

定义 : Error类及其子类。

特点 : 和运行时异常一样,编译器也不会对错误进行检查。当资源不足、约束失败、或是其它程序无法继续运行的条件发生时,就产生错误。程序本身无法修复这些错误的。例如,VirtualMachineError就属于错误。出现这种错误会导致程序终止运行。OutOfMemoryErrorThreadDeath

Java虚拟机规范规定JVM的内存分为了好几块,比如堆,栈,程序计数器,方法区等

1Java中异常分为哪两种?

编译时异常

运行时异常

2、异常的处理机制有几种?

异常捕捉:try…catch…finally,异常抛出:throws

3、如何自定义一个异常

继承一个异常类,通常是RumtimeException或者Exception

4try catch finallytry里有returnfinally还执行么?

执行,并且finally的执行早于try里面的return

结论:

1.不管有木有出现异常,finally块中代码都会执行;

2.trycatch中有return时,finally仍然会执行;

3.finally是在return后面的表达式运算后执行的(此时并没有返回运算后的值,而是先把要返回的值保存起来,管finally中的代码怎么样,返回的值都不会改变,任然是之前保存的值),所以函数返回值是在finally执行前确定的;

4.finally中最好不要包含return,否则程序会提前退出,返回值不是trycatch中保存的返回值。

5finalfianllyfianlize的区别?

1final用于修饰属性,方法,和类,分别表示属性不可变,方法不可覆盖,,类不可被继承(不能再派生出新的子类)

2finally作为异常处理的一部分,它只能用在try/catch语句中,并且附带一个语句块,表示这段语句最终一定要被执行,经常被用在需要释放资源的情况下。

3finalize Object类的一个方法,在垃圾回收器执行时会调用被回收对象的finalize()方法,可以覆盖此方法来实现对其他资源的回收,例如关闭文件等。需要注意的是,一旦垃圾回收器准备好释放对象占用的空间,将首先调用其finalize()方法,并且在下一次垃圾回收动作发生时,才会真正回收对象占用的内存。

6 ExcptionError包结构

Java可抛出(Throwable)的结构分为三种类型:被检查的异常(CheckedException),运行时异常

(RuntimeException),错误(Error)

7Thowthorws区别

位置不同

throws 用在函数上,后面跟的是异常类,可以跟多个;而 throw 用在函数内,后面跟的

是异常对象。

功能不同

throws 用来声明异常,让调用者只知道该功能可能出现的问题,可以给出预先的处理方

式;throw 抛出具体的问题对象,执行到 throw,功能就已经结束了,跳转到调用者,并

将具体的问题对象抛给调用者。也就是说 throw 语句独立存在时,下面不要定义其他语

句,因为执行不到。

throws 表示出现异常的一种可能性,并不一定会发生这些异常;throw 则是抛出了异常,

执行 throw 则一定抛出了某种异常对象。

两者都是消极处理异常的方式,只是抛出或者可能抛出异常,但是不会由函数去处理异

常,真正的处理异常由函数的上层调用处理。

8ErrorException区别?

ErrorException都是java错误处理机制的一部分,都继承了Throwable类。

Exception表示的异常,异常可以通过程序来捕捉,或者优化程序来避免。

Error表示的是系统错误,不能通过程序来进行错误处理。

线程

一、多线程基础基础知识
1. 并发编程
1.1 并发编程的优缺点
优点:

充分利用多核CPU的计算能力,通过并发编程的形式将多核CPU的计算能力发挥到极致,性能得到提升。
方面进行业务的拆分。提高系统并发能力和性能:高并发系统的开发,并发编程会显得尤为重要,利用好多线程机制可以大大提高系统的并发能力及性能;面对复杂的业务模型,并行程序会比串行程序更适应业务需求,而并发编程更适合这种业务拆分。cai
缺点:

并发编程的目的是为了提高程序的执行效率,提高程序运行速度,但并发编程并不是总能提高性能,有时还会遇到很多问题,例如:内存泄漏,线程安全,死锁等。
1.2 并发编程的三要素
并发编程的三要素:(也是带来线程安全所在)

原子性:原子是不可再分割的最小单元,原子性是指一个或多个操作要么全部执行成功,要么全部执行失败。
可见性:一个线程对共享变量的修改,另一个线程能看到(synchronized,volatile
有序性:程序的执行顺序按照代码的先后顺序
线程安全的问题原因有:

  1. 1. 线程切换带来的原子性问题
  2. 2. 缓存导致的可见性问题
  3. 3. 编译优化带来的有序性问题
  4. 1
  5. 2
  6. 3


解决方案:

JDK Atomic开头的原子类、synchronizedLOCK,可以解决原子性问题
synchronizedvolatileLOCK,可以解决可见性问题
Happens-Before 规则可以解决有序性问题
1.3 并发和并行有和区别
并发:多个任务在同一个CPU上,按照细分的时间片轮流交替执行,由于时间很短,看上去好像是同时进行的。
并行:单位时间内,多个处理器或多核处理器同时处理多个任务,是真正意义上的同时进行。
串行:有n个任务,由一个线程按照顺序执行。

1.4 什么是多线程,多线程的优劣?
定义:多线程是指程序中包含多个流,即在一个程序中可以同时进行多个不同的线程来执行不同的任务
优点:

可以提高CPU的利用率,在多线程中,一个线程必须等待的时候,CPU可以运行其它线程而不是等待,这样就大大提高了程序的效率,也就是说单个程序可以创建多个不同的线程来完成各自的任务。
缺点:
线程也是程序,线程也需要占内存,线程也多内存也占的也多。
多线程需要协调和管理,所以需要CPU跟踪线程。
线程之间共享资源的访问会相互影响,必须解决禁用共享资源的问题。


2. 线程与进程
2.1 什么是线程与进程


进程:内存中运行的运用程序,每个进程都有自己独立的内存空间,一个进程可以由多个线程,例如在Windows系统中,xxx.exe就是一个进程。
线程:进程中的一个控制单元,负责当前进程中的程序执行,一个进程至少有一个线程,一个进程可以运行多个线程,多个线程可以共享数据。

2.2 线程与进程的区别
根本区别:进程是操作系统资源分配的基本单元,而线程是处理器任务调度的和执行的基本单位。
资源开销:每个进程都有自己独立的代码和空间(程序上下文),程序之间的切换会有较大的开销;线程可以看作轻量级的进程,同一类线程共享代码和数据空间,每个线程都有自己独立的运行栈和程序计数器(PC),线程之间切换的开销小。
包含关系:如果一个进程内有多个线程,则执行的过程不是一条线的,而是多条线(多个线程),共同完成;线程是进程的一部分,可以把线程看作是轻量级的进程。
内存分配:同一进程的线程共享本进程的地址空间和资源,而进程之间的地址空间和资源是相互独立的。

2.3 用户线程与守护线程
用户(User)线程:运行在前台,执行具体任务,如程序的主线程,连接网络的子线程都是用户线程。
守护(Daemon)线程:运行在后台,为其它前台线程服务,也可以说守护线程是JVM非守护线程的佣人,一旦所有线程都执行结束,守护线程会随着JVM一起结束运行。
main函数就是一个用户线程,main函数启动时,同时JVM还启动了好多的守护线程,如垃圾回收线程,比较明显的区别时,用户线程结束,JVM退出,不管这个时候有没有守护线程的运行,都不会影响JVM的退出。

2.4 什么是线程死锁
死锁是指两个或两个以上进程(线程)在执行过程中,由于竞争资源或由于彼此通信造成的一种堵塞的现象,若无外力的作用下,都将无法推进,此时的系统处于死锁状态。
如图,线程A拥有的资源2,线程B拥有的资源1,此时线程A和线程B都试图去拥有资源1和资源2,但是它们的🔒还在,因此就出现了死锁。


2.5 形成死锁的四个必要条件
互斥条件:线程(进程)对所分配的资源具有排它性,即一个资源只能被一个进程占用,直到该进程被释放。
请求与保持条件:一个进程(线程)因请求被占有资源而发生堵塞时,对已获取的资源保持不放。
不剥夺条件:线程(进程)已获取的资源在未使用完之前不能被其他线程强行剥夺,只有等自己使用完才释放资源。
循环等待条件:当发生死锁时,所等待的线程(进程)必定形成一个环路,死循环造成永久堵塞。
2.6 如何避免死锁
我们只需破坏形参死锁的四个必要条件之一即可。
破坏互斥条件:无法破坏,我们的🔒本身就是来个线程(进程)来产生互斥
破坏请求与保持条件:一次申请所有资源
破坏不剥夺条件:占有部分资源的线程尝试申请其它资源,如果申请不到,可以主动释放它占有的资源。
破坏循环等待条件:按序来申请资源。

2.7 什么是上下文的切换
当前任务执行完,CPU时间片切换到另一个任务之前会保存自己的状态,以便下次再切换会这个任务时可以继续执行下去,任务从保存到再加载执行就是一次上下文切换。

3. 创建线程
3.1 创建线程的四种方式
继承Thread
实现Runnable接口
实现Callable接口
Executors工具类创建线程池
3.2 Runnable接口和Callable接口有何区别
相同点:

RunnableCallable都是接口
都可以编写多线程程序
都采用Thread.start()启动线程
不同点:

Runnable接口run方法无返回值,Callable接口call方法有返回值,是个泛型,和FutrueFutureTask配合用来获取异步执行结果。
Runable接口run方法只能抛出运行时的异常,且无法捕获处理;Callable接口call方法允许抛出异常,可以获取异常信息。
注:Callable接口支持返回执行结果,需要调用FutureTask.get()得到,此方法会堵塞主线程继续往下执行,如果不调用就不会堵塞。

3.2 run()方法和start()方法有和区别
每个线程都是通过某个特定的Thread对象对于的run()方法来完成其操作的,run方法称为线程体,通过调用Thread类的start方法来启动一个线程。
start()方法用于启动线程,run()方法用于执行线程的运行代码,run()可以反复调用,而start()方法只能被调用一次。

start()方法来启动一个线程,真正实现了多线程的运行。调用start()方法无需等待run()方法体代码执行结束,可以直接继续执行其它的代码;调用start()方法线程进入就绪状态,随时等该CPU的调度,然后可以通过Thread调用run()方法来让其进入运行状态,run()方法运行结束,此线程终止,然后CPU再调度其它线程。

3.3 为什么调用start()方法会执行run()方法,为什么不能直接调用run()方法
这是一个常问的面试题,new Thread,线程进入了新建的状态,start方法的作用是使线程进入就绪的状态,当分配到时间片后就可以运行了。start方法会执行线程前的相应准备工作,然后在执行run方法运行线程体,这才是真正的多线程工作。
如果直接执行了run方法,run方法会被当作一个main线程下的普通方法执行,并不会在某个线程中去执行它,所以这并不是多线程工作。
小结:
调用start方法启动线程可使线程进入就绪状态,等待运行;run方法只是thread的一个普通方法调用,还是在主线程里执行。

3.4 什么是CallableFuture
Callable接口也类似于Runnable接口,但是Runnable不会接收返回值,并且无法抛出返回结果的异常,而Callable功能更强大,被线程执行后,可有返回值,这个返回值可以被Future拿到,也就是说Future可以拿到异步执行任务的返回值。
Future接口表示异步任务,是一个可能没有完成的异步任务结果,所以说Callable用于产生结果,Future用于接收结果。

3.5 什么是FutureTask
FutureTask是一个异步运算的任务,FutureTask里面可以可以传入Callable实现类作为参数,可以对异步运算任务的结果进行等待获取,判断是否已经完成,取消任务等操作。只有当结果完成之后才能取出,如果尚未完成get方法将堵塞。一个Future对象可以调用CallableRunable的对象进行包装,由于FutureTask也是Runnable接口的实现类,所以FutureTask也可以放入线程池中。

4. 线程状态和基本操作
4.1 线程声明周期的6种状态
很多地方说线程有5种状态,但实际上是6中状态,可以参考Thread类的官方api

新创建:又称初始化状态,这个时候Thread才刚刚被new出来,还没有被启动。
可运行状态:表示已经调用Threadstart方法启动了,随时等待CPU的调度,此状态又被称为就绪状态。
被终止:死亡状态,表示已经正常执行完线程体run()中的方法了或者因为没有捕获的异常而终止run()方法了。
计时状态:调用sleep(参数)wait(参数)后线程进入计时状态,睡眠时间到了或wait时间到了,再或者其它线程调用notify并获取到锁之后开始进入可运行状态。另一种情况,其它线程调用notify没有获取到锁或者wait时间到没有获取到锁时,进入堵塞状态。
无线等待状态:获取锁对象后,调用wait()方法,释放锁进入无线等待状态
锁堵塞状态:wait(参数)时间到或者其它线程调用notify后没有获取到锁对象都会进入堵塞状态,只要一获取到锁对象就会进入可运行状态。

4.2 Java用到的线程调度算法是什么?
计算机通常只有一个CPU,在任意时刻只能执行一条机器指令,每个线程只有获取到CPU的使用权才能执行指令,所谓多线程的并发运行,其实从宏观上看,各线程轮流获取CPU的使用权,分别执行各自的任务。在运行池中,会有多个处于就绪状态的线程在等待,CPU的调度,JVM有一项任务就是负责CPU的调度,线程调度就是按照特定的机制为多个线程分配CPU的使用权。
有两种调度模型:分时调度和抢占式调度
分时调度:  就是让所有的线程轮流获得CPU的使用权,并且平均分配到各个线程占有CPU的时间片。
抢占式调度:Java虚拟机采用抢占式调度模型,是指优先让线程池中优先级高的线程首先占用CPU,如果线程池中优先级相同,那么随机选择一个线程,使其占有CPU,处于这个状态的CPU会一直运行,优先级高的分的CPU的时间片相对会多一点。

4.2 Java线程调度策略
线程调度优先选择优先级高的运行,但是如果出现一下情况,就会终止运行(不是进入死亡状态)

线程调用了yield方法让出CPU的使用权,线程进入就绪状态。
线程调用sleep()方法,使其进入计时状态线程由于IO受阻另一个更高的优先级线程出现在支持的时间片系统中,改线程的时间片用完。
4.3 什么是线程调度(Thread Scheduler)和时间分片(Time Slicing )
线程调度是一个操作系统服务,它负责为储在Runnable状态的线程分配CPU时间片,一旦我们创建一个线程并启动它,它的执行便依赖线程调度器的实现。
时间分片是指CPU可用时间分配给Runnable的过程,分配的时间可以根据线程优先级或线程等待时间。

4.4 Java线程同步和线程调度的相关方法
wait()调用后线程进入无限等待状态,并释放所持对象的锁
sleep()使一个线程进入休眠状态(堵塞状态),带有对象锁,是一个静态方法,需要处理InterruptException异常。
notify()唤醒一个处于等待状态的线程(无线等待或计时等待),如果多个线程在等待,并不能确切的唤醒一个线程,与JVM确定唤醒那个线程,与其优先级有关。
notityAll()唤醒所有处于等待状态的线程,但是并不是将对象的锁给所有的线程,而是让它们去竞争,谁先获取到锁,谁先进入就绪状态。
4.5 sleep()wait()有什么区别
两者都可以使线程进入等待状态

类不同:sleep()Thread下的静态方法,wait()Object类下的方法
是否释放锁:sleep()不释放锁,wait()释放锁
用处不同:wait()常用于线程间的通信,sleep()常用于暂停执行。
用法不同:wait()用完后,线程不会自动执行,必须调用notify()notifyAll()方法才能执行,sleep()方法调用后,线程经过过一定时间会自动苏醒,wait(参数)也可以传参数使其苏醒。它们苏醒后还有所区别,因为wait()会释放锁,所以苏醒后没有获取到锁就进入堵塞状态,获取到锁就进入就绪状态,而sleep苏醒后之间进入就绪状态,但是如果cpu不空闲,则进入的是就绪状态的堵塞队列中。
4.6 你是如何调用wait()方法的,使用if还是循环
        等待状态的线程可能会收到错误警告或伪唤醒,如果不在循环中检查等待条件,程序可能会在没有满足条件的时候退出。

  1. synchronized (monitor) {
  2.     //  判断条件谓词是否得到满足
  3.     while(!locked) {
  4.         //  等待唤醒
  5.         monitor.wait();
  6.     }
  7.     //  处理其他的业务逻辑
  8. }
  9. 1
  10. 2
  11. 3
  12. 4
  13. 5
  14. 6
  15. 7
  16. 8
  17. 9


4.7 为什么线程通信方法wait(),notify(),notifyAll()要被定义到Object类中
Java中任何对象都可以被当作锁对象,wait(),notify(),notifyAll()方法用于等待获取唤醒对象去获取锁,Java中没有提供任何对象使用的锁,但是任何对象都继承于Object类,所以定义在Object类中最合适。

有人会说,既然是线程放弃对象锁,那也可以把wait()放到Thread类中,新定义线程继承Thread类,也无需重新定义wait()
然而,这样做有一个很大的问题,因为一个线程可以持有多把锁,你放弃一个线程时,到底要放弃哪把锁,当然了这种设计不能实现,只是管理起来比较麻烦。
综上:wait(),notify(),notifyAll()应该要被定义到Object类中。

4.8 为什么线程通信方法wait(),notify(),notifyAll()要在同步代码块或同步方法中被调用?
wait(),notify(),notifyAll()方法都有一个特点,就是对象去调用它们的时候必须持有锁对象。
如对象调用wait()方法后持有的锁对象就释放出去,等待下一个线程来获取。
如对象调用notifyAll()要唤醒等待中的线程,也要讲自身用于的锁对象释放,让就绪状态中的线程竞争获取锁。
由于这些方法都需要线程持有锁对象,这样只能通过同步来实现,所以它们只能在同步块或同步方法中被调用。

4.9 Threadyiele方法有什么作用?
让出CPU的使用权,使当前线程从运行状态进入就绪状态,等待CPU的下次调度。

4.10 为什么Threadsleepyield是静态的?
Thread类的sleep()yield()方法将在当前正在运行的线程上工作,所以其它处于等待状态的线程调用它们是没有意义的,所以设置为静态最合适。

4.11 线程sleepyield方法有什么区别
线程调用sleep()方法进入堵塞状态,醒来后因为(没有释放锁)后直接进入了就绪状态,运行yield后也没有释放锁,于是进入了就绪状态。
sleep()方法使用时需要处理InterruptException异常,而yield没有。
sleep()执行后进入堵塞状态(计时等待),醒来后进入就绪状态(可能是堵塞队列),而yield是直接进入就绪状态。
4.12 如何停止一个正在运行的线程?
使用stop方法终止,但是这个方法已经过期,不被推荐使用。
使用interrupt方法终止线程
run方法执行结束,正常退出
4.13 如何在两个线程间共享数据?
两个线程之间共享变量即可实现共享数据。
一般来说,共享变量要求变量本身是线程安全的,然后在线程中对变量使用。

4.14 同步代码块和同步方法怎么选?
同步块是更好的选择,因为它不会锁着整个对象,当然你也可以然它锁住整个对象。同步方法会锁住整个对象,哪怕这个类中有不关联的同步块,这通常会导致停止继续执行,并等待获取这个对象锁。
同步块扩展性比较好,只需要锁住代码块里面相应的对象即可,可以避免死锁的产生。
原则:同步范围也小越好。

4.15 什么是线程安全?Servlet是线程安全吗?
线程安全是指某个方法在多线程的环境下被调用时,能够正确处理多线程之间的共享变量,能程序能够正确完成。
Servlet不是线程安全的,它是单实例多线程的,当多个线程同时访问一个方法时,不能保证共享变量是安全的。
Struts2是多实例多线程的,线程安全,每个请求过来都会new一个新的action分配这个请求,请求完成后销毁。
springMVCcontrollerServlet一样,属性单实例多线程的,不能保证共享变量是安全的。
Struts2好处是不用考虑线程安全问题,springMVCServlet需要考虑。
如果想既可以提升性能又可以不能管理多个对象的话建议使用ThreadLocal来处理多线程。

4.16 线程的构造方法,静态块是被哪个线程类调用的?
线程的构造方法,静态块是被哪个线程类调用的?
该线程在哪个类中被new出来,就是在哪个被哪个类调用,而run方法是线程类自身调用的。
例子:mian函数中new Thread2Thread2new Thread1

thread1线程的构造方法,静态块是thread2线程调用的,run方法是thread1调用的。
thread2线程的构造方法,静态块是main线程调用的,run方法是thread2调用的。

4.17 Java中是如何保证多线程安全的?
使用安全类,比如 java.util.concurrent 下的类,使用原子类AtomicInteger
使用自动锁,synchronized
Lock lock = new ReentrantLock(),使用手动锁lock .lock()lock.unlock()方法
4.18 线程同步和线程互斥的区别
线程同步:当一个线程对共享数据进行操作的时候,在没有完成相关操作时,不允许其它的线程来打断它,否则就会破坏数据的完整性,必然会引起错误信息,这就是线程同步。
线程互斥:
而线程互斥是站在共享资源的角度上看问题,例如某个共享资源规定,在某个时刻只能一个线程来访问我,其它线程只能等待,知道占有的资源者释放该资源,线程互斥可以看作是一种特殊的线程同步。
实现线程同步的方法:

同步代码块:sychronized(对象){}
同步方法:sychronized修饰的方法
使用重入锁实现线程同步:reentrantlock类的锁又互斥功能,Lock lock = new ReentrantLock(); Lock对象的ockunlock为其加锁
4.19 你对线程优先级有什么理解?
每个线程都具有优先级的,一般来说,高优先级的在线程调度时会具有优先被调用权。我们可以自定义线程的优先级,但这并不能保证高优先级又在低优先级前被调用,只是说概率有点大。
线程优先级是1-101代表最低,10代表最高。
Java的线程优先级调度会委托操作系统来完成,所以与具体的操作系统优先级也有关,所以如非特别需要,一般不去修改优先级。

4.20 谈谈你对乐观锁和悲观锁的理解?
乐观锁:每个去拿数据的时候都认为别人不会修改,所以不会都不会上锁,但是在更新的时候会判断一下在此期间有没有去更新这个数据。所以乐观锁使用了多读的场合,这样可以提高吞吐量,像数据库提供的类似write_condition机制,都是用的乐观锁,还有那个原子变量类,在java.util.concurrent.atomic包下
悲观锁:总是假设最坏的情况,每次去拿数据的时候都会认为有人会修改,所以每次在拿数据的时候都会上锁。这样别的对象想拿到数据,那就必须堵塞,直到拿到锁。传统的关系型数据库用到了很多这种锁机制,比如读锁,写锁,在操作之前都会先上锁,再比如Java的同步代码块synchronized/方法用的也是悲观锁。

Socket

什么是socket?
        Socket的英文原义是插座。在网络编程中,网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个socket

        Socket套接字是通信的基石,是支持TCP/IP协议的网络通信的基本操作单元。它是网络通信过程中端点的抽象表示,包含进行网络通信必须的五种信息:连接使用的协议,本地主机的IP地址,本地进程的协议端口,远地主机的IP地址,远地进程的协议端口。

        Socket本质是编程接口(API),对TCP/IP的封装,TCP/IP也要提供可供程序员做网络开发所用的接口,这就是Socket编程接口;HTTP是轿车,提供了封装或者显示数据的具体形式;Socket是发动机,提供了网络通信的能力。

    Socket的原理
        Socket实质上提供了进程通信的端点。进程通信之前,双方首先必须各自创建一个端点,否则是没有办法建立联系并相互通信的。正如打电话之前,双方必须各自拥有一台电话机一样。

        套接字之间的连接过程可以分为三个步骤:服务器监听,客户端请求,连接确认。

        1、服务器监听:是服务器端套接字并不定位具体的客户端套接字,而是处于等待连接的状态,实时监控网络状态。

        2、客户端请求:是指由客户端的套接字提出连接请求,要连接的目标是服务器端的套接字。为此,客户端的套接字必须首先描述它要连接的服务器的套接字,指出服务器端套接字的地址和端口号,然后就向服务器端套接字提出连接请求。

        3、连接确认:是指当服务器端套接字监听到或者说接收到客户端套接字的连接请求,它就响应客户端套接字的请求,建立一个新的线程,把服务器端套接字的描述发给客户端,一旦客户端确认了此描述,连接就建立好了。而服务器端套接字继续处于监听状态,继续接收其他客户端套接字的连接请求。
 

TCPUDP的区别
        UDP

        1、每个数据报中都给出了完整的地址信息,因此无需要建立发送方和接收方的连接。

        2UDP传输数据时是有大小限制的,每个被传输的数据报必须限定在64KB之内。

        3UDP是一个不可靠的协议,发送方所发送的数据报并不一定以相同的次序到达接收方

        TCP

        1、面向连接的协议,在socket之间进行数据传输之前必然要建立连接,所以在TCP中需要连接时间。

        2TCP传输数据没有大小限制,一旦连接建立起来,双方的socket就可以按统一的格式传输大的数据。

        3TCP是一个可靠的协议,它确保接收方完全正确地获取发送方所发送的全部数据。
 

集合框架

什么是集合

集合框架:用于存储数据的容器。Contains{接口+接口的实现+数据结构的算法}

接口:表示集合的抽象数据类型。

实现:集合接口的具体实现,是重用性很高的数据结构。

算法:在一个实现了某个集合框架中的接口的对象身上完成某种有用的计算的方法,例如查找、排序等。

容器都有哪些?他们有什么特点?哪些是线程安全的?哪些是线程不安全的?

Collection

                1. List

                      1) ArrayList(底层是数组,查找快)(线程不安全)

                      2) LinkedList(底层是双向链表, 在指定位置增删元素快)(线程不安全)

                      3) Vector(用法跟ArrayList差不多, 效率比ArrayList)(线程安全)

                2. Set

                      1) HashSet

                      2) TreeSet

Map

        1. HashMap

        2. HshTable

        3. ConcrruentHashMap(线程安全, 不涉及同步加锁)

集合和数组的区别

  • 数组是固定长度的;集合可变长度的。
  • 数组可以存储基本数据类型,也可以存储引用数据类型;集合只能存储引用数据类型。
  • 数组存储的元素必须是同一个数据类型;集合存储的对象可以是不同数据类型。

使用集合框架的好处

  1. 容量自增长;
  2. 提供了高性能的数据结构和算法,使编码更轻松,提高了程序速度和质量;
  3. 允许不同 API 之间的互操作,API之间可以来回传递集合;
  4. 可以方便地扩展或改写集合,提高代码复用性和可操作性。
  5. 通过使用JDK自带的集合类,可以降低代码维护和学习新API成本。

常用的集合类有哪些?
Map接口和Collection接口是所有集合框架的父接口:

Collection接口的子接口包括:Set接口和List接口
Map接口的实现类主要有:HashMapTreeMapHashtableConcurrentHashMap以及Properties
Set接口的实现类主要有:HashSetTreeSetLinkedHashSet
List接口的实现类主要有:ArrayListLinkedList、以及Vector

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

List Set 的区别

List 特点:一个有序容器,元素可以重复,可以插入多个null元素,元素都有索引。常用的实现类有 ArrayListLinkedList Vector

Set 特点:一个无序容器,不可以存储重复元素,必须保证元素唯一性 , 只允许存入一个null元素,。Set 接口常用实现类是 HashSetLinkedHashSet 以及 TreeSet

另外 List 支持for循环,也就是通过下标来遍历,也可以用迭代器,但是set只能用迭代,因为他无序,无法用下标来取得想要的值。

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

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

ArrayList LinkedList 的区别是什么?
ArrayList 底层是数组 , LinkedList 底层是双向链表。
ArrayList 查询快, LinkedList 增删快
LinkedList ArrayList 更占内存,因为 LinkedList 的节点除了存储数据,还存储了两个地址引用,一个指向前一个元素,一个指向后一个元素。
 

ArrayList Vector 的区别是什么?

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

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

HashSet底层封装的是什么?

封装的是HashMap

关于HashMap底层的一些问题

  1. HashMap的底层存储结构是怎样的啊?
  2. 线程安全吗?为什么不安全?
  3. 1.71.8版本的HashMap有什么区别?1.7的有什么隐患,什么原因导致的?
  4. hashcode是唯一的吗?插入元素的时候怎么比较的?
  5. HashTableConcurrentHashMap有什么区别?
  6. 为什么HashMap链表树化的标准是8个?
  7. HashMap的基本知识点
    HashMap可以说是Java项目里最常用的集合类了,作为一种典型的K-V存储的数据结构,它的底层是由数组 - 链表组成,当添加新元素时,它会根据元素的hash值找到对应的"",也就是HashMap源码中Node<K, V> 里的元素,并插入到对应位置的链表中,链表元素个数过长时会转化为红黑树(JDK1.8后的版本)
     

 为什么要转成红黑树呢?
我们都知道,链表取元素是从头结点一直遍历到对应的结点,这个过程的复杂度是O(N) ,而红黑树基于二叉树的结构,查找元素的复杂度为O(logN) ,所以,当元素个数过多时,用红黑树存储可以提高搜索的效率。
既然红黑树的效率高,那怎么不一开始就用红黑树存储呢?JDK的源码里已经对这个问题做了解释:
 

  1.  Because TreeNodes are about twice the size of regular nodes, we
  2.  use them only when bins contain enough nodes to warrant use
  3.  (see TREEIFY_THRESHOLD). And when they become too small (due to
  4.  removal or resizing) they are converted back to plain bins. 

看注释里的前面四行就不难理解,单个 TreeNode 需要占用的空间大约是普通 Node 的两倍,所以只有当包含足够多的 Nodes 时才会转成 TreeNodes,这个足够多的标准就是由 TREEIFY_THRESHOLD 的值(默认值8)决定的。而当桶中节点数由于移除或者 resize (扩容) 变少后,红黑树会转变为普通的链表

  1. UNTREEIFY_THRESHOLD(默认值6)。
  2. /**
  3. * The bin count threshold for using a tree rather than list for a
  4. * bin.  Bins are converted to trees when adding an element to a
  5. * bin with at least this many nodes. The value must be greater
  6. * than 2 and should be at least 8 to mesh with assumptions in
  7. * tree removal about conversion back to plain bins upon
  8. * shrinkage.
  9. */
  10. static final int TREEIFY_THRESHOLD = 8;
  11. /**
  12. * The bin count threshold for untreeifying a (split) bin during a
  13. * resize operation. Should be less than TREEIFY_THRESHOLD, and at
  14. * most 6 to mesh with shrinkage detection under removal.
  15. */
  16. static final int UNTREEIFY_THRESHOLD = 6;

看到这里就不难明白了,红黑树虽然查询效率比链表高,但是结点占用的空间大,只有达到一定的数目才有树化的意义,这是基于时间和空间的平衡考虑。
为什么树化标准是8
至于为什么树化标准的数量是8个,在源码中,上面那段笔记后面还有一段较长的注释,我们可以从那一段注释中找到答案,原文是这样:
 

  1. * usages with well-distributed user hashCodes, tree bins are
  2. * rarely used.  Ideally, under random hashCodes, the frequency of
  3. * nodes in bins follows a Poisson distribution
  4. * (http://en.wikipedia.org/wiki/Poisson_distribution) with a
  5. * parameter of about 0.5 on average for the default resizing
  6. * threshold of 0.75, although with a large variance because of
  7. * resizing granularity. Ignoring variance, the expected
  8. * occurrences of list size k are (exp(-0.5) * pow(0.5, k) /
  9. * factorial(k)). The first values are:
  10. *
  11. * 0:    0.60653066
  12. * 1:    0.30326533
  13. * 2:    0.07581633
  14. * 3:    0.01263606
  15. * 4:    0.00157952
  16. * 5:    0.00015795
  17. * 6:    0.00001316
  18. * 7:    0.00000094
  19. * 8:    0.00000006
  20. * more: less than 1 in ten million

大概意思就是:如果 hashCode的分布离散良好的话,那么红黑树是很少会被用到的,因为各个值都均匀分布,很少出现链表很长的情况。在理想情况下,链表长度符合泊松分布,各个长度的命中概率依次递减,注释中给我们展示了1-8长度的具体命中概率,当长度为8的时候,概率概率仅为0.00000006(亿分之六),这么小的概率,HashMap的红黑树转换几乎不会发生,因为我们日常使用不会存储那么多的数据,你会存上千万个数据到HashMap中吗?
当然,这是理想的算法,但不妨某些用户使用HashMap过程导致hashCode分布离散很差的场景,这个时候再转换为红黑树就是一种很好的退让策略。

首先说明一下,在HashMap中,决定某个对象落在哪一个,是由该对象的hashCode决定的,JDK无法阻止用户实现自己的哈希算法,如果用户重写了hashCode,并且算法实现比较差的话,就很可能会使HashMap的链表变得很长,就比如这样:

  1. public class HashMapTest {
  2.     public static void main(String[] args) {
  3.         Map<User, Integer> map = new HashMap<>();
  4.         for (int i = 0; i < 1000; i++) {
  5.             map.put(new User("闯哥最帅" + i), i);
  6.         }
  7.     }
  8.     static class User{
  9.         private String name;
  10.         public User(String name) {
  11.             this.name = name;
  12.         }
  13.         @Override
  14.         public int hashCode() {
  15.             return 1;
  16.         }
  17.     }
  18. }

我们设计了一个hashCode永远为1的类User,这样一来存储到HashMap的所有User对象都会存放到同一个里,查询效率无疑会非常的低下,而这也是HashMap设计链表转红黑树的原因之一,可以有效防止用户自己实现了不好的哈希算法时导致链表过长的情况。
hash方法
说到哈希算法,我们再来扩充一个知识点,这也是我觉得HashMap中非常牛逼的设计之一。
HashMap的源码中,存储对象hashCode的计算是由hash() 方法决定的,hash() HashMap 中的核心函数,在存储数据时,将key传入中进行运算,得出key的哈希值,通过这个哈希值运算才能获取key应该放置在的哪个位置,下面是方法的源码:
 

  1. static final int hash(Object key) {
  2.     int h;
  3.     return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
  4. }

从代码中可以看出,传入key之后,hash() 会获取keyhashCode进行无符号右移 16 位,然后进行按位异或,并把运算后的值返回,这个值就是key的哈希值。这样运算是为了减少碰撞冲突,因为大部分元素的hashCode在低位是相同的,不做处理的话很容易造成冲突。
除了做16位位移的处理,在添加元素的方法中,HashMap还把该hash值与table.length - 1,也就是数组的大小做与运算,得到的结果就是对应的数组的下标,从而找到该元素所属的链表。源码里这样的:
 

  1. // n的值是table.length
  2. if ((p = tab[i = (n - 1) & hash]) == null)
  3.     tab[i] = newNode(hash, key, value, null);

当查找不到对应的索引时,就会新建一个新的结点作为链表的头结点。那么这里为什么要用 i = (n - 1) & hash 作为索引运算呢?

这其实是一种优化手段,由于数组的大小永远是一个2次幂,在扩容之后,一个元素的新索引要么是在原位置,要么就是在原位置加上扩容前的容量。这个方法的巧妙之处全在于&运算,之前提到过&运算只会关注n – 1n =数组长度)的有效位,当扩容之后,n的有效位相比之前会多增加一位(n会变成之前的二倍,所以确保数组长度永远是2次幂很重要),然后只需要判断hash在新增的有效位的位置是0还是1就可以算出新的索引位置,如果是0,那么索引没有发生变化,如果是1,索引就为原索引加上扩容前的容量。

 通过位运算,在每次扩容时都不用重新计算hash,省去了不少时间,而且新增有效位是0还是1是带有随机性的,之前两个碰撞的Entry又有可能在扩容时再次均匀地散布开,达到较好的分布离散效果.
为什么退化为链表的阈值是6?
上面说到,当链表长度达到阈值8的时候会转为红黑树,但是红黑树退化为链表的阈值却是6,为什么不是小于8就退化呢?比如说7的时候就退化,偏偏要小于或等于6
主要是一个过渡,避免链表和红黑树之间频繁的转换。如果阈值是7的话,删除一个元素红黑树就必须退化为链表,增加一个元素就必须树化,来回不断的转换结构无疑会降低性能,所以阈值才不设置的那么临界。

反射        

  反射是框架的灵魂

  (使用的前提条件: 必须先得到代表的字节码的Class, Class类用于表示.class文件(字节码))

一、反射的概述
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象.


以上的总结就是什么是反射
反射就是把java类中的各种成分映射成一个个的Java对象
例如:一个类有:成员变量、方法、构造方法、包等等信息,利用反射技术可以对一个类进行解剖,把个个组成部分映射成一个个对象。
     (其实:一个类中这些成员方法、构造方法、在加入类中都有一个类来描述)
如图是类的正常加载过程:反射的原理在与class对象。
熟悉一下加载的时候:Class对象的由来是将class文件读入内存,并为之创建一个Class对象。

 其中这个Class对象很特殊。我们先了解一下这个C lass:

Class 类的实例表示正在运行的 Java 应用程序中的类和接口。也就是jvm中有N多的实例每个类都有该Class对象。(包括基本数据类型)
Class 没有公共构造方法。Class 对象是在加载类时由 Java 虚拟机以及通过调用类加载器中的defineClass 方法自动构造的。也就是这不需要我们自己去处理创建,JVM已经帮我们创建好了。没有公共的构造方法.

反射的使用
先写一个Student类。

1、获取Class对象的三种方式
1.1 Object ——> getClass();
1.2
任何数据类型(包括基本数据类型)都有一个静态class属性
1.3 通过Class类的静态方法:forNameString  className(常用)

其中1.1是因为Object类中的getClass方法、因为所有类都继承Object类。从而调用Object类来获取

数据库

常见的关系型数据库

oracle              Oracle(甲骨文) 收费, 支持各种操作系统,NBA金州勇士(甲骨文球场)

mysql              Oracle(甲骨文) 开源, 支持各种操作系统

mariadb

sql server       微软  .NET  收费, 中型数据库, 只能用windows系统

DB2                 IBM 收费, 银行业务用的比较多

Sqlite              迷你型数据库, 嵌入式设备(安卓, 苹果, pad)

概念简称DB, 按照数据结构组织, 存储和管理数据库的仓库

SQL中的日期和时间类型

  1. data                日历日期(年月日)
  2. time                时分秒
  3. timestamp           年月日时分秒
  4. Datatime            年月日时分秒

varcharchar的区别

char是固定长度的类型     varchar是可变长度的类型   char的效率要比varchar

关联查询内连接,外连接,自连接

(1) 外连接之笛卡尔积查询  (2)什么时候会发生笛卡尔积查询?

(1)就是两个集合数据条数相乘的结果  (2)关联查询没添加关联关系

 分组: group by(单独使用, 聚合函数, HAVING,

HAVING关键字和WHERE关键字的作用相同,都是用于设置条件表达式,对查询结果进行过滤。

两者的区别,HAVING关键字后,可以跟聚合函数,而WHERE关键字不能,通常情况下,HAVING关键字,都是和GROUP BY一起使用,用于对分组后的结果进行过滤

)

 数据库聚合函数: (MAX(最大值), min(最小值), sum(), avg(平均数), count(统计个数))

分页查询mysql: limit 0(跳过的数据)  5(显示的数据)

                oracle: ROWNUM <= 40RN >= 21控制分页查询的每页的范围。

索引建立索引对于mysql高效运行时很重要, 索引可以提高mysql的检索速度, 创建索引时, 需要确保该索引是应用在SQL查询语句的条件(一般为where条件)    创建索引的字段我们建议是唯一, 不为空, 经常被查询的字段       普通索引唯一索引(unique)

索引索然会大大提高了查询速度,同时却会降低更新表的速度,如对表进行Insert,Update,Delete.
为更新表时,Mysql不仅要保存数据,还要保存一下索引文件.
优点:可以大大加快数据的检索速度,这也是创建索引的最主要的原因.
缺点:1)增加了数据库的存储空间 2)减慢了数据录入的速度

约束:

主键约束(Primay Key Coustraint) 唯一性,非空性(唯一且非空)
唯一约束(Unique Constraint) 唯一性(可以为空,但是只能有一个)
检查约束(Check Counstraint) 对该列格式和范围的限制,自定义约束,自己决定限制条件
外键约束(Foreign Key Counstraint) 需要建立两表间的关系
默认约束(Default Coustraint) 该数据的默认值
非空约束(Not Null Counstraint) 该字段不能为空
有的说数据的5大约束,有的是6大约束

事务数据库的事务是指一组sql语句组成的数据库逻辑处理单元,在这组的sql操作中,要么全部执行成功,要么全部执行失败。

事务的ACID: Mysql中事务的四大特性主要包含:

原子性(Atomicity原子性是指事务的原子性操作,对数据的修改要么全部执行成功,要么全部失败,实现事务的原子性

一致性(Consistent一致性是指执行事务前后的状态要一致,可以理解为数据一致性。

隔离性(Isalotion隔离性侧重指事务之间相互隔离,不受影响,这个与事务设置的隔离级别有密切的关系。

持久性(Durable)持久性则是指在一个事务提交后,这个事务的状态会被持久化到数据库中,也就是事务提交,对数据的新增、更新将会持久化到书库中。

在我的理解中,原子性、隔离性、持久性都是为了保障一致性而存在的,一致性也是最终的目的。

简称为ACID

事务的隔离级别

Mysql中事务的隔离级别分为四大等级

读未提交(READ UNCOMMITTED读未提交会读到另一个事务的未提交的数据,产生脏读问题

读已提交 READ COMMITTED读已提交则解决了脏读的,出现了不可重复读,即在一个事务任意时刻读到的数据可能不一样,可能会受到其它事务对数据修改提交后的影响,一般是对于update的操作。

可重复读 REPEATABLE READ可重复读解决了之前不可重复读和脏读的问题,但是由带来了幻读的问题,幻读一般是针对inser操作。例如:第一个事务查询一个Userid=100发现不存在该数据行,这时第二个事务又进来了,新增了一条id=100的数据行并且提交了事务。 这时第一个事务新增一条id=100的数据行会报主键冲突,第一个事务再select一下,发现id=100数据行已经存在,这就是幻读。

串行化 SERIALIZABLE。可串行化是一个调度,即多个事务之间的执行方式;而多个事务之间的执行有个先后顺序,如果事务之间没有共同的操作对象(读或写操作),则事务之间的执行顺序前后置换是没有关系的;但是如果事物间存在共同的操作对象,则事务间先后执行的顺序则需要区分;对于存在共同操作对象的多个并发执行的事务,如果其执行结果等价于某个串行化调度,则这个调度才是可串行化的调度。满足可串行化的调度则具有了可串行化(Serializability)属性。所以,可串行化(Serializability)属性保证的是多个事务并发时的执行顺序要对数据的一致性没有影响。
 

悲观锁与乐观锁:

悲观锁(Pessimistic Locking):       

       悲观锁,正如其名,它指的是对数据被外界(包括本系统当前的其他事务,以及来自 外部系统的事务处理)修改持保守态度,因此,在整个数据处理过程中,将数据处于锁定状态。

       悲观锁的实现,往往依靠数据库提供的锁机制(也只有数据库层提供的锁机制才能 真正保证数据访问的排他性,否则,即使在本系统中实现了加锁机制,也无法保证外部系 统不会修改数据)。

       通俗理解:一段数据在被A以悲观锁的形式访问(select * from  card_record where status ="03' for update,那么在这个访问没有commit或者事务结束之前,这些 status ="03'的数据是无法被修改的。这条 sql 语句锁定了 card_record 表中所有符合检索条件( status ="03' )的记录。本次事务提交之前(事务提交时会释放事务过程中的锁),外界无法修改这些记录。 需要注意的是for update要放到mysql的事务中,即begincommit中,否则不起作用。

       悲观锁有两种:读锁和写锁。上面的操作属于写锁的操作,A上了锁,那么别人就不可以再上锁了,有锁的人可以对数据进行读写操作。而读锁相当于共享锁,开启事务后,A上了共享锁,那么其他人也可以上共享锁。大家都可以读这些数据,但是都不能修改。

       悲观的理解,别人在我读写数据的时候会进行更改,那么我就上锁,在我读写数据的时候不允许别人更改我正在操作的数据。

乐观锁(Optimistic Locking):        

         由于悲观锁在频繁依靠数据库的锁机制实现,以保证操作最大程度的独占性。但随之而来的就是数据库性能的大量开销。而乐观锁机制避免了长事务中的数据库加锁开销,大大提升了大并发量下的系统整体性能表现。

          乐观锁,大多是基于数据版本(Version )记录机制实现。何谓数据版本?即为数据增加一个版本标识,在基于数据库表的版本解决方案中,一般是通过为数据库表增加一个 “version” 字段来 实现。 读取出数据时,将此版本号一同读出,之后更新时,对此版本号加一。此时,将提交数据的版本数据与数据库表对应记录的当前版本信息进行比对,如果提交的数据 版本号大于数据库表当前版本号,则予以更新,否则认为是过期数据。

         通俗的讲,AB都在对一个版本为1的数据进行修改,在事务提交之前,这段数据的版本一直是1,那么下次提交的人会把这段数据的版本变为2。如果AB之前提交了,那么这段数据的版本变为2,之后B提交了数据,由于B拿到数据的时候版本是1,那么加一后,预估版本也是2,但是提交完发现,现在的版本已经是2了,所以B的修改和提交是无效的。

          需要注意的是,乐观锁机制往往基于系统中的数据存储逻辑,因此也具备一定的局限性,如在上例中,由于乐观锁机制是在我们的系统中实现,来自外部系统的用户对数据的操作不受我们系统的控制,因此可能会造成脏数据被更新到数据库中。在 系统设计阶段,我们应该充分考虑到这些情况出现的可能性,并进行相应调整(如将乐观锁策略在数据库存储过程中实现,对外只开放基于此存储过程的数据更新途 径,而不是将数据库表直接对外公开)。

          乐观的理解,在我读写数据的时候,没人对这段数据进行更改,所以可以随意更改数据,只是commit的时候可能会出现版本问题。

         悲观锁,前提是,一定会有并发抢占资源,强行独占资源,在整个数据处理过程中,将数据处于锁定状态。

         乐观锁,前提是,不会发生并发抢占资源,只有在提交操作的时候检查是否违反数据完整性。只能防止脏读后数据的提交,不能解决脏读。
 

视图:
试图是从一个或几个基本表(或视图)导出来的表。它与基本表不同,是一个虚表。数据库中只存放视图的定义,而不存放视图对应的数据,这些数据仍存放在原来的基本表中。所以一旦基本表中的数据发生变化,从视图中查询出来的数据也就随之改变了。从这个意义上说,视图就像是一个窗口,透过它可以看到数据库中自己感兴趣的数据及其变化。

视图一经定义,就可以和基本表一样被查询、删除,也可以在一个视图上再定义新的视图,但对视图的更新(增、删、改)操作则有一定的限制。

视图的作用:
1、视图机制使用户可以将注意力集中放在所关心的数据上。如果这些数据不是直接来自基本表,则可以通过定义视图使数据库看起来结构简单清晰,并且可以简化用户的数据查询操作。例如,那些定义了若干张表连接的视图就将表与表之间的连接操作对用户隐蔽起来,也就是说,用户所做的只是对一个虚表的简单查询,而这个虚表是怎么得来的,用户无需了解。

2、视图使用户能以多种角度看待同一数据。视图机制能使不同的用户以不同的方式看待同一数据,当许多不同种类的用户共享同一个数据库时,这种灵活性是很重要的。

3、视图对重构数据库提供了一定程度的逻辑独立性。具体就不解释了,想了解的去看书或者查资料。

4、视图能够对机密数据提供安全保护。有了视图机制,就可以在设计数据库应用系统时对不同的用户定义不同的视图,使机密数据不出现在不应该看到这些数据的用户视图上。这样视图机制就自动提供了对机密数据的安全保护功能。
CAS算法: 不写了暂时用不到

Spring Framework

官方网址(快速入门): Spring | Spring Quickstart Guide

特征:

  • 核心技术Ioc、依赖注入、数据绑定、类型转换、AOP等。
  • 数据访问:事务、DAO 支持、ORMMarshalling XML等。
  • 集成:远程处理、JMSJCAJMX、电子邮件、任务、调度、缓存。

Spring是一个分层的轻量级java开发框架,为企业级应用开发提供了全面的编程和配置模型。它为Java开发者提供了近乎于全面的基础架构支持,使得我们只需关注应用的业务逻辑而无需去理会那些与业务逻辑无关的繁琐、重复的编程工作。
Spring的众多功能特性大都依赖于他的两个核心功能模块:依赖注入( dependency Injection )简称DI面向切面编程(AOP

1.1Spring的优缺点
优点:
1、使用SpringIOC容器,将对象之间的依赖关系交由Spring管理,降低了组件之间的耦合性,实现了软件各层的解耦。
2、使用容器提供众多服务,如事务管理、消息队列等。
3AOP技术,利用它可以实现一些拦截,如权限拦截。
4、对主流的框架提供了很好的支持,如Mybatis
5、低浸入式设计
6SpringDI机制降低了业务对象替换的复杂性。
缺点:
Spring依赖反射机制,反射机制占用内存影响性能。

1.2Spring中有哪些功能模块
1Spring Core
框架的最基础部分,提供IoC容器,对bean进行管理
2Spring Context
基于Bean,提供上下文信息,扩展出JNDIEJB、电子邮件、国际化、校验和调度等功能。
3Spring Dao
提供了JDBC的抽象层,它可消除冗长的JDBC编码和解析数据库厂商特有的代码错误,还提供了声明式事务管理方法。
4Spring ORM
提供了常用的对象-关系映射APIs的集成,其中包括HibernateMybatis等。
5Spring AOP
提供了符合AOP Alliance规范的面向切面的编程实现。
6Spring Web
提供了基础的Web开发的上下文信息
7Spring Web MVC
提供了web应用的Model-View-Controller功能实现。

1.3Spring中用到的设计模式
1、工厂模式
Spring通过BeanFactory或者ApplicationContext创建Bean对象用到了工厂模式。

.什么是工厂模式
工厂模式(Factory Pattern)是 Java 中最常用的设计模式之一。

这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。利用工厂模式,我们可以在创建对象时不对客户端暴露创建逻辑,而是通过使用一个共同的接口来创建新的对象。使类实例化的操作使用对象的操作分开。

.工厂模式具体可以分为3:
1.简单工厂模式,2工厂模式,3抽象工厂模式

.为什么要用工厂模式
1. 将对象的使用的对象的创建分离开来
2. 如果创建一个类的步骤很复杂,很多地方都用得到这个类,我们可以使用工厂模式做统一创建。3. 数据库访问时的可维护、可扩展的问题。


.工厂模式的优点
增加了代码的重用性
增加了代码的可维护性


.工厂模式的缺点
使用了工厂模式,就会引入工厂类,会增加系统的复杂度
2、单例模式
Springbean的作用域默认为单例模式(singleton)。
3、代理模式
Spring AOP(面向切面编程)基于动态代理,如果要代理的对象实现了某个接口,SpringAOP会使用JDK Proxy创建代理对象;如果对象没有实现接口,Spring AOP会使用CGLib生成一个被代理对象的子类作为代理。
4、模板方法模式
Spring中的jdbcTemplate等以Template结尾的对数据库操作的类就使用了模板模式。
5、适配器模式
Spring AOP的实现基于代理模式,但是SpringAOP的增强(Advice)使用了适配器模式。SpringMVCHandlerAdapter也是适配器模式。
6、观察者模式
定义对象键一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都会得到通知被制动更新,Spring事件驱动模型就是观察者模式一个很经典的应用。

1.4Spring中的事件有哪几种?
1、上下文更新事件(ContextRefreshedEvent):在调用ConfigurableApplicationContext 接口中的refresh()方法时被触发。
2、上下文开始事件(ContextStartedEvent):当容器调用ConfigurableApplicationContextStart()方法开始/重新开始容器时触发该事件。
3、上下文停止事件(ContextStoppedEvent):当容器调用ConfigurableApplicationContextStop()方法停止容器时触发该事件。
4、上下文关闭事件(ContextClosedEvent):ApplicationContext被关闭时触发该事件。容器被关闭时,其管理的所有单例Bean都被销毁。
5、请求处理事件(RequestHandledEvent):Web应用中,当一个http请求(request)结束触发该事件。如果一个bean实现了ApplicationListener接口,当一个ApplicationEvent 被发布以后,bean会自动被通知。

2、什么是IoC
IOC(Inverse of Control)即控制反转——某一接口具体实现类的选择控制权从调用类中移除,转交由第三方Spring容器借由bean配置来进行控制。SPring IoC负责创建对象、管理对象、装配对象、配置对象,并且管理对象的整个生命周期。
(注:依赖注入(DI)其实指的就是IoC,其解释为:让调用类对某一接口实现类的依赖关系由第三方(Spring 容器)注入,以移除调用类对某一接口实现类的依赖。)

2.1IoC的作用
管理对象的创建和依赖关系的维护。对象的创建并不是一件简单的事,在对象关系比较复杂时,如果依赖关系需要程序员来维护的话成本是很高的。
解耦,由容器去维护具体的对象。
托管了类的生产过程,比如我们需要在类的生产过程中做一些处理,最直接的例子就是代理,如有容器程序可以把这部分处理交给容器,应用程序则无需去关心类是如何完成代理的。

2.2Spring IoC的实现机制
Spring 中的IoC的实现原理就是工厂模式加反射机制。

2.3SpringIoC支持哪些功能?
1、依赖注入
2、依赖检查
3、自动装配
4、支持集合
5、指定初始化方法和销毁方法

2.4Ioc的类型有哪几种?(依赖注入的类型有哪几种)
1、构造函数注入
通过调用类的构造函数,将接口实现类通过构造函数变量传入。
2、属性注入
有选择的通过Setter方法完成调用类所需依赖的注入,更加灵活方便。
3、接口注入
将调用类所有依赖注入的方法抽取到一个接口中,调用类通过实现改接口提供相应的注入方法(必须先声明一个接口)。
(注:Spring可以通过容器来完成依赖关系的注入)
 

3Spring AOP
OOP(Object-Oriented Programming)面向对象编程,允许开发者定义纵向的关系,但并不适用于定义横向的关系,导致了大量代码的重复,而不利于各个模块的重用。

AOP(Aspect-Oriented Programming),一般称为面向切面编程,作为面向对象的一种补充,用于将那些与业务无关,但却对多个对象产生影响的公共行为和逻辑,抽取并封装为一个可重用的模块,这个模块被命名为切面Aspect),减少系统中的重复代码,降低了模块间的耦合度,同时提高了系统的可维护性。可用于权限认证、日志、事务处理等。

特点

1、降低模块之间的耦合度

2、使系统容易扩展

3、更好的代码复用。

一、对AOP的初印象
首先先给出一段比较专业的术语(来自百度):

在软件业,AOPAspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方
式和运行期动态代理实现程序功能的统一维护的一种技术。AOPOOP的延续,是软件开发中的一个
热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑
的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高
了开发的效率。

要理解切面编程,就需要先理解什么是切面。用刀把一个西瓜分成两瓣,切开的切口就是切面;炒菜,锅与炉子共同来完成炒菜,锅与炉子就是切面。web层级设计中,web->网关层->服务层->数据层,每一层之间也是一个切面。编程中,对象与对象之间,方法与方法之间,模块与模块之间都是一个个切面。

二、AOP中的相关概念
看过了上面的例子,我想大家脑中对AOP已经有了一个大致的雏形,但是又对上面提到的切面之类的术语有一些模糊的地方,接下来就来讲解一下AOP中的相关概念,了解了AOP中的概念,才能真正的掌握AOP的精髓。
这里还是先给出一个比较专业的概念定义:
 

  •  Aspect(切面): Aspect 声明类似于 Java 中的类声明,在 Aspect 中会包含着一些 Pointcut 以及相应的 Advice
  • Joint point(连接点):表示在程序中明确定义的点,典型的包括方法调用,对类成员的访问以及异常处理程序块的执行等等,它自身还可以嵌套其它 joint point
  • Pointcut(切点):表示一组 joint point,这些 joint point 或是通过逻辑关系组合起来,或是通过通配、正则表达式等方式集中起来,它定义了相应的 Advice 将要发生的地方。
  • Advice(增强):Advice 定义了在 Pointcut 里面定义的程序点具体要做的操作,它通过 beforeafter around 来区别是在每个 joint point 之前、之后还是代替执行的代码。
  • Target(目标对象):织入 Advice 的目标对象.
  • Weaving(织入):将 Aspect 和其他对象连接起来, 并创建 Adviced object 的过程
  • 然后举一个csdn最火最容易理解的例子:
  • 看完了上面的理论部分知识, 我相信还是会有不少朋友感觉到 AOP 的概念还是很模糊, AOP 中的各种概念理解的还不是很透彻. 其实这很正常, 因为 AOP 中的概念是在是太多了, 我当时也是花了老大劲才梳理清楚的.
  • 下面我以一个简单的例子来比喻一下 AOP Aspect, Joint point, Pointcut Advice之间的关系.
  • 让我们来假设一下, 从前有一个叫爪哇的小县城, 在一个月黑风高的晚上, 这个县城中发生了命案. 作案的凶手十分狡猾, 现场没有留下什么有价值的线索. 不过万幸的是, 刚从隔壁回来的老王恰好在这时候无意中发现了凶手行凶的过程, 但是由于天色已晚, 加上凶手蒙着面, 老王并没有看清凶手的面目, 只知道凶手是个男性, 身高约七尺五寸. 爪哇县的县令根据老王的描述, 对守门的士兵下命令说: 凡是发现有身高七尺五寸的男性, 都要抓过来审问. 士兵当然不敢违背县令的命令, 只好把进出城的所有符合条件的人都抓了起来.
  • 来让我们看一下上面的一个小故事和 AOP 到底有什么对应关系.
  • 首先我们知道, Spring AOP Joint point 指代的是所有方法的执行点, point cut 是一个描述信息, 它修饰的是 Joint point, 通过 point cut, 我们就可以确定哪些 Joint point 可以被织入 Advice. 对应到我们在上面举的例子, 我们可以做一个简单的类比, Joint point 就相当于 爪哇的小县城里的百姓,pointcut 就相当于 老王所做的指控, 即凶手是个男性, 身高约七尺五寸, Advice 则是施加在符合老王所描述的嫌疑人的动作: 抓过来审问.
  • 为什么可以这样类比呢?
  • Joint point 爪哇的小县城里的百姓: 因为根据定义, Joint point 是所有可能被织入 Advice 的候选的点, Spring AOP, 则可以认为所有方法执行点都是 Joint point. 而在我们上面的例子中, 命案发生在小县城中, 按理说在此县城中的所有人都有可能是嫌疑人.
  • Pointcut :男性, 身高约七尺五寸: 我们知道, 所有的方法(joint point) 都可以织入 Advice, 但是我们并不希望在所有方法上都织入 Advice, Pointcut 的作用就是提供一组规则来匹配joinpoint, 给满足规则的 joinpoint 添加 Advice. 同理, 对于县令来说, 他再昏庸, 也知道不能把县城中的所有百姓都抓起来审问, 而是根据凶手是个男性, 身高约七尺五寸, 把符合条件的人抓起来. 在这里 凶手是个男性, 身高约七尺五寸 就是一个修饰谓语, 它限定了凶手的范围, 满足此修饰规则的百姓都是嫌疑人, 都需要抓起来审问.
  • Advice :抓过来审问, Advice 是一个动作, 即一段 Java 代码, 这段 Java 代码是作用于 point cut 所限定的那些 Joint point 上的. 同理, 对比到我们的例子中, 抓过来审问 这个动作就是对作用于那些满足 男性, 身高约七尺五寸 的爪哇的小县城里的百姓.
  • Aspect:Aspect point cut Advice 的组合, 因此在这里我们就可以类比: “根据老王的线索, 凡是发现有身高七尺五寸的男性, 都要抓过来审问这一整个动作可以被认为是一个 Aspect.

 AOP中的Joinpoint可以有多种类型:构造方法调用,字段的设置和获取,方法的调用,方法的执行,异常的处理执行,类的初始化。也就是说在AOP的概念中我们可以在上面的这些Joinpoint上织入我们自定义的Advice,但是在Spring中却没有实现上面所有的joinpoint,确切的说,Spring只支持方法执行类型的Joinpoint

Advice 的类型

  • before advice, join point 前被执行的 advice. 虽然 before advice 是在 join point 前被执行, 但是它并不能够阻止 join point 的执行, 除非发生了异常(即我们在 before advice 代码中, 不能人为地决定是否继续执行 join point 中的代码)
  • after return advice, 在一个 join point 正常返回后执行的 advice
  • after throwing advice, 当一个 join point 抛出异常后执行的 advice
  • after(final) advice, 无论一个 join point 是正常退出还是发生了异常, 都会被执行的 advice.
  • around advice, join point 前和 joint point 退出后都执行的 advice. 这个是最常用的 advice.
  • introductionintroduction可以为原有的对象增加新的属性和方法。

Aop代理之什么是代理?

指为一个目标对象提供一个代理对象, 并由代理对象控制对目标对象的引用. 使用代理对象, 是为了在不修改目标对象的基础上, 增强目标对象的业务逻辑

 静态代理

静态代理的特点是, 为每一个业务增强都提供一个代理类, 由代理类来创建代理对象. 下面我们通过静态代理来实现对转账业务进行身份验证.

(1) 转账业务

  1. public interface IAccountService {
  2.     //主业务逻辑: 转账
  3.     void transfer();
  4. }
  5. public class AccountServiceImpl implements IAccountService {
  6.     @Override
  7.     public void transfer() {
  8.         System.out.println("调用dao,完成转账主业务.");
  9.     }
  10. }

(2) 代理类

  1. public class AccountProxy implements IAccountService {
  2.     //目标对象
  3.     private IAccountService target;
  4.     public AccountProxy(IAccountService target) {
  5.         this.target = target;
  6.     }
  7.     /**
  8.      * 代理方法,实现对目标方法的功能增强
  9.      */
  10.     @Override
  11.     public void transfer() {
  12.         before();
  13.         target.transfer();
  14.     }
  15.     /**
  16.      * 前置增强
  17.      */
  18.     private void before() {
  19.         System.out.println("对转账人身份进行验证.");
  20.     }
  21. }

(3) 测试

  1. public class Client {
  2.     public static void main(String[] args) {
  3.         //创建目标对象
  4.         IAccountService target = new AccountServiceImpl();
  5.         //创建代理对象
  6.         AccountProxy proxy = new AccountProxy(target);
  7.         proxy.transfer();
  8.     }
  9. }
  10. 结果:
  11. 对转账人身份进行验证.
  12. 调用dao,完成转账主业务.

动态代理
静态代理会为每一个业务增强都提供一个代理类, 由代理类来创建代理对象, 而动态代理并不存在代理类, 代理对象直接由代理生成工具动态生成.

JDK动态代理

JDK动态代理是使用 java.lang.reflect 包下的代理类来实现. JDK动态代理动态代理必须要有接口.

(1) 转账业务

  1. public interface IAccountService {
  2.     //主业务逻辑: 转账
  3.     void transfer();
  4. }
  5. public class AccountServiceImpl implements IAccountService {
  6.     @Override
  7.     public void transfer() {
  8.         System.out.println("调用dao,完成转账主业务.");
  9.     }
  10. }

(2) 增强

因为这里没有配置切入点, 称为切面会有点奇怪, 所以称为增强.

  1. public class AccountAdvice implements InvocationHandler {
  2.     //目标对象
  3.     private IAccountService target;
  4.     public AccountAdvice(IAccountService target) {
  5.         this.target = target;
  6.     }
  7.     /**
  8.      * 代理方法, 每次调用目标方法时都会进到这里
  9.      */
  10.     @Override
  11.     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  12.         before();
  13.         return method.invoke(target, args);
  14.     }
  15.     /**
  16.      * 前置增强
  17.      */
  18.     private void before() {
  19.         System.out.println("对转账人身份进行验证.");
  20.     }
  21. }

 (3) 测试

  1. public class Client {
  2.     public static void main(String[] args) {
  3.         //创建目标对象
  4.         IAccountService target = new AccountServiceImpl();
  5.         //创建代理对象
  6.         IAccountService proxy = (IAccountService) Proxy.newProxyInstance(target.getClass().getClassLoader(),
  7.                 target.getClass().getInterfaces(),
  8.                 new AccountAdvice(target)
  9.         );
  10.         proxy.transfer();
  11.     }
  12. }
  13. 结果:
  14. 对转账人身份进行验证.
  15. 调用dao,完成转账主业务.

 CGLIB动态代理

JDK动态代理必须要有接口, 但如果要代理一个没有接口的类该怎么办呢? 这时我们可以使用CGLIB动态代理. CGLIB动态代理的原理是生成目标类的子类, 这个子类对象就是代理对象, 代理对象是被增强过的.

注意: 不管有没有接口都可以使用CGLIB动态代理, 而不是只有在无接口的情况下才能使用.

 (1) 转账业务

  1. public class AccountService {
  2.     public void transfer() {
  3.         System.out.println("调用dao,完成转账主业务.");
  4.     }
  5. }

(2) 增强

因为这里没有配置切入点, 称为切面会有点奇怪, 所以称为增强.

  1. public class AccountAdvice implements MethodInterceptor {
  2.     /**
  3.      * 代理方法, 每次调用目标方法时都会进到这里
  4.      */
  5.     @Override
  6.     public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
  7.         before();
  8.         return methodProxy.invokeSuper(obj, args);
  9.         //        return method.invoke(obj, args);  这种也行
  10.     }
  11.     /**
  12.      * 前置增强
  13.      */
  14.     private void before() {
  15.         System.out.println("对转账人身份进行验证.");
  16.     }
  17. }

(3) 测试

  1. public class Client {
  2.     public static void main(String[] args) {
  3.         //创建目标对象
  4.         AccountService target = new AccountService();
  5.         //
  6.         //创建代理对象
  7.         AccountService proxy = (AccountService) Enhancer.create(target.getClass(),
  8.                 new AccountAdvice());
  9.         proxy.transfer();
  10.     }
  11. }
  12. 结果:
  13. 对转账人身份进行验证.
  14. 调用dao,完成转账主业务.

 模拟Spring AOP场景

了解了动态代理后, 我们就可以自己来实现Spring AOP功能了, 所以下面我们来模拟下Spring AOP场景.

(1) 转账业务

  1. public interface IAccountService {
  2.     //主业务逻辑: 转账
  3.     void transfer();
  4. }
  5. public class AccountServiceImpl implements IAccountService {
  6.     @Override
  7.     public void transfer() {
  8.         System.out.println("调用dao,完成转账主业务.");
  9.     }
  10. }

(2) 切面抽象类

定义一个切面抽象类, 该类使用了模板方法的设计模式, 为开始, 结束, 异常, 前置增强, 后置增强提供了默认实现, 当我们定义切面类时只需要按需重写它们就行. isIntercept() 方法用来判断切入点是否正确, 切面类需要重写这个方法.

  1. public abstract class BaseAspect implements MethodInterceptor {
  2.     private static final Logger logger = LoggerFactory.getLogger(BaseAspect.class);
  3.     @Override
  4.     public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
  5.         Object result = null;
  6.         begin();
  7.         try {
  8.             if (isIntercept(method, args)) {
  9.                 before();
  10.                 result = methodProxy.invokeSuper(obj, args);
  11.                 after();
  12.             } else {
  13.                 result = methodProxy.invokeSuper(obj,args);
  14.             }
  15.         } catch (Exception e) {
  16.             logger.error("proxy failure", e);
  17.             error(e);
  18.             throw e;
  19.         } finally {
  20.             end();
  21.         }
  22.         return result;
  23.     }
  24.     /**
  25.      * 开始增强
  26.      */
  27.     public void begin() {
  28.     }
  29.     /**
  30.      * 切入点判断
  31.      */
  32.     public boolean isIntercept(Method method, Object[] args) throws Throwable {
  33.         return true;
  34.     }
  35.     /**
  36.      * 前置增强
  37.      */
  38.     public void before() throws Throwable {
  39.     }
  40.     /**
  41.      * 后置增强
  42.      */
  43.     public void after() throws Throwable {
  44.     }
  45.     /**
  46.      * 异常增强
  47.      */
  48.     public void error(Throwable e) {
  49.     }
  50.     /**
  51.      * 最终增强
  52.      */
  53.     public void end() {
  54.     }
  55. }

(3) 切面类

创建一个切面类, 类中配置切入点和增强.

  1. public class AccountAspect extends BaseAspect {
  2.     /**
  3.      * 切入点
  4.      */
  5.     public boolean isIntercept(Method method, Object[] args) throws Throwable {
  6.         return method.getName().equals("transfer");
  7.     }
  8.     /**
  9.      * 前置增强
  10.      */
  11.     public void before() throws Throwable {
  12.         System.out.println("对转账人身份进行验证.");
  13.     }
  14. }

(4) 代理工厂类

定义一个工厂类来创建代理, 其实不创建这个类也行, 但为了模仿Spring还是创建了. @SuppressWarnings是为了抑制警告, 就是编译器上面的黄线.

  1. public class ProxyFactory {
  2.     @SuppressWarnings("unchecked")
  3.     public static <T> T createProxy(final Class<?> targetClass, final MethodInterceptor methodInterceptor) {
  4.         return (T) Enhancer.create(targetClass,methodInterceptor);
  5.     }
  6. }

(5) 测试

  1. public class Client {
  2.     public static void main(String[] args) {
  3.         //创建目标对象
  4.         IAccountService target = new AccountServiceImpl();
  5.         //切面
  6.         BaseAspect accountAspect = new AccountAspect();
  7.         //创建代理对象
  8.         IAccountService proxy = (IAccountService) ProxyFactory.createProxy(target.getClass(), accountAspect);
  9.         proxy.transfer();
  10.     }
  11. }
  12. 结果:
  13. 对转账人身份进行验证.
  14. 调用dao,完成转账主业务.

4Spring的通知有哪些类型?
AOP 术语中,切面的工作被称为通知,实际上是程序执行时要通过Spring AOP框架触发的代码段。Spring切面可以应用5种类型的通知:

: 说中文不严谨, 说英文
1、前置通知(@Before):在目标方法被调用之前调用此功能
2、后置通知(@After):在目标方法完成之后执行
3、返回通知(@AfterReturning):在目标方法成功执行之后调用此功能
4、异常通知(@AfterThrowing):在目标方法抛出异常后调用此通知
5、环绕通知(@Around):在目标方法调用之前和调用之后执行

5Springbean的生命周期?
1Spring启动,查找并加载需要被Spring管理的bean,进行Bean的实例化
2Bean实例化后对将Bean的引入和值注入到Bean的属性中
3、如果Bean实现了BeanNameAware接口的话,SpringBeanId传递给setBeanName()方法
4、如果Bean实现了BeanFactoryAware接口的话,Spring将调用setBeanFactory()方法,将BeanFactory容器实例传入
5、如果Bean实现了ApplicationContextAware接口的话,Spring将调用BeansetApplicationContext()方法,将bean所在应用上下文引用传入进来。
6、如果Bean实现了BeanPostProcessor接口,Spring就将调用他们的postProcessBeforeInitialization()方法。
7、如果Bean 实现了InitializingBean接口,Spring将调用他们的afterPropertiesSet()方法。类似的,如果bean使用init-method声明了初始化方法,该方法也会被调用
8、如果Bean 实现了BeanPostProcessor接口,Spring就将调用他们postProcessAfterInitialization()方法。
9、此时,Bean已经准备就绪,可以被应用程序使用了。他们将一直驻留在应用上下文中,直到应用上下文被销毁。
10、如果bean实现了DisposableBean接口,Spring将调用它的destory()接口方法,同样,如果bean使用了destory-method 声明销毁方法,该方法也会被调用。
 

 解释Spring支持的几种bean的作用域。

Spring容器中的bean可以分为5个范围:

1singleton默认,每个容器中只有一个bean的实例,单例的模式由BeanFactory自身来维护。

2prototype为每一个bean请求提供一个实例。

3request为每一个网络请求创建一个实例,在请求完成以后,bean会失效并被垃圾回收器回收。

4sessionrequest范围类似,确保每个session中有一个bean的实例,在session过期后,bean会随之失效。

5global-session全局作用域,global-sessionPortlet应用相关。当你的应用部署在Portlet容器中工作时,它包含很多portlet。如果你想要声明让所有的portlet共用全局的存储变量的话,那么这全局变量需要存储在global-session中。全局作用域与Servlet中的session作用域效果相同。

8Spring框架中的单例Beans是线程安全的么?

        Spring框架并没有对单例bean进行任何多线程的封装处理。关于单例bean的线程安全和并发问题需要开发者自行去搞定。但实际上,大部分的Spring bean并没有可变的状态(比如Serview类和DAO),所以在某种程度上说Spring的单例bean是线程安全的。如果你的bean有多种状态的话(比如 View Model 对象),就需要自行保证线程安全。最浅显的解决办法就是将多态bean的作用域由“singleton”变更为“prototype”

9Spring如何处理线程并发问题?

在一般情况下,只有无状态的Bean才可以在多线程环境下共享,在Spring中,绝大部分Bean都可以声明为singleton作用域,因为Spring对一些Bean中非线程安全状态采用ThreadLocal进行处理,解决线程安全问题。

ThreadLocal和线程同步机制都是为了解决多线程中相同变量的访问冲突问题。同步机制采用了时间换空间的方式,仅提供一份变量,不同的线程在访问前需要获取锁,没获得锁的线程则需要排队。而ThreadLocal采用了空间换时间的方式。

ThreadLocal会为每一个线程提供一个独立的变量副本,从而隔离了多个线程对数据的访问冲突。因为每一个线程都拥有自己的变量副本,从而也就没有必要对该变量进行同步了。ThreadLocal提供了线程安全的共享对象,在编写多线程代码时,可以把不安全的变量封装进ThreadLocal

 10-1Spring基于xml注入bean的几种方式:

1Set方法注入;

2)构造器注入:通过index设置参数的位置;通过type设置参数类型;

3)静态工厂注入;

4)实例工厂;

beanFactoryapplicationContext的区别?
ApplicationContextBeanFactory派生而来,提供了更多面向实际应用的功能。beanFactorySpring里面最低层的接口,提供了最简单的容器的功能,只提供了实例化对象和拿对象的功能。而ApplicationContext应用上下文,继承BeanFactory接口,它是Spring的一各更高级的容器,提供了更多的有用的功能;1.国际化(MessageSource)、2.访问资源(ResourceLoader)、3.载入多个(有继承关系的上下文),使得每一个上下文都专注于一个特定的层次。4.消息发送、响应机制(ApplicationEventPublisher5.AOP拦截器
beanFactoryapplicationContext的主要区别体现在装载bean的方面:
beanfactory在容器启动的时候不会去实例化bean,而是在从容器中取bean的时候才实例化一个。
applicationContext在启动的时候就把所有的bean全部实例化了。

SpringBoot


1.谈谈你对Spring Boot的理解?
SpringBoot主要用来简化使用Spring的难度和繁重的XML配置,它是Spring组件的一站式解决方案,采取了习惯优于配置的方法。通过.properties或者.yml文件替代了Spring繁杂的XML配置文件,同时支持@ImportResource注解加载XML配置。Spring Boot还提供了嵌入式HTTP服务器、命令行接口工具、多种插件等等,使得应用程序的测试和开发简单起来。

2. 为什么需要Spring Boot
Spring Boot 优点非常多,如:独立运行、简化配置、自动配置和无需部署war文件等等

3. 说出Spring Boot 的优点
简化开发,提高整体生产力
Spring Boot 使用 JavaConfig 有助于避免使用 XML,同时避免大量的Maven导入和各种版本冲突
Spring Boot 引导的应用程序可以很容易地与 Spring 生态系统集成,如Spring JDBCSpring ORMSpring DataSpring Security等等
Spring Boot 应用程序提供嵌入式HTTP服务器,如TomcatJetty,可以轻松地开发和测试web应用程序。
Spring Boot 提供命令行接口工具,用于开发和测试应用程序
Spring Boot 提供了多种插件,可以使用内置Maven工具开发和测试 应用程序
Spring Boot 没有单独的 Web 服务器需要,这意味着不再需要启动 Tomcat或其他任何东西

4. Spring Boot 的核心配置文件有哪几个?它们的区别是什么?
Spring Boot 的核心配置文件是 application bootstrap 配置文件。
application 配置文件主要用于 Spring Boot 项目的自动化配置。
bootstrap 配置文件有三个应用场景。
使用Spring Cloud Config配置中心时,需要在 bootstrap 配置文件中添加连接到配置中心的配置属性,来加载外部配置中心的配置信息;
一些固定的不能被覆盖的属性;
一些加密或解密的场景;

5. Spring Boot 的配置文件有哪几种格式?它们有什么区别?
主要有.properties .yml格式,它们的区别主要是书写格式不同。另外,.yml 格式不支持 @PropertySource 注解导入配置。

6. 开启SpringBoot特性有哪几种方式?
继承spring-boot-starter-parent项目
导入spring-boot-dependencies项目依赖

7. 什么是Spring Boot Starter
Starters可以理解为启动器,它包含了一系列可以集成到应用里面的依赖包,可以一站式集成 Spring 和其他技术,而不需要到处找示例代码和依赖包。Spring Boot Starter的工作原理是:Spring Boot 在启动时扫描项目所依赖的JAR包,寻找包含spring.factories文件的JAR包,根据spring.factories配置加载AutoConfigure类,根据 @Conditional注解的条件,进行自动配置并将Bean注入Spring Context

8. Spring Boot 有哪几种读取配置的方式?
使用@Value注解加载单个属性值
使用@ConfigurationProperties注解可以加载一组属性的值,针对于要加载的属性过多的情况,比@Value注解更加简洁

9. Spring Boot 支持哪些日志框架?推荐和默认的日志框架是哪个?
Spring Boot 支持 Java Util Logging, Log4j2, Logback 作为日志框架,如果使用 Starters 启动器,Spring Boot 将使用 Logback 作为默认日志框架,推荐的日志框架是Log4j2

10. Spring Boot 可以兼容老 Spring 项目吗?
可以兼容,使用 @ImportResource 注解导入老 Spring 项目配置文件。

11. 保护 Spring Boot 应用有哪些方法?
在生产中使用HTTPS
使用Snyk检查依赖关系
升级到最新版本
启用CSRF保护
使用内容安全策略防止XSS攻击

12. 什么是 JavaConfig
JavaConfig Spring 社区的产品,它提供了配置 Spring IoC 容器的纯 Java 方法,有助于避免使用 XML 配置。

13. Spring Boot 的核心注解是哪个?它主要由哪几个注解组成的)介绍一下 @SpringBootApplication 注解
Spring Boot 的核心注解是@SpringBootApplication,它也是启动类使用的注解,主要包含了 3 个注解:
@SpringBootConfiguration:它组合了 @Configuration 注解,实现配置文件的功能。
@EnableAutoConfiguration:具有打开自动配置的功能,也可以关闭某个自动配置的选项。
@ComponentScan:用于Spring组件扫描。

14. Spring Boot 自动配置原理是什么?
@EnableAutoConfiguration注解、 @Configuration注解和 @ConditionalOnClass注解组成了Spring Boot自动配置的核心,首先它得是一个配置文件,其次根据类路径下是否有这个类去自动配置。具体是通过maven读取每个starter中的spring.factories文件,该文件配置了所有需要被创建在spring容器中的bean

15. 你如何理解 Spring Boot 配置加载顺序?
Spring Boot配置加载顺序优先级是:propertiese文件、YAML文件、系统环境变量、命令行参数。

16. Spring Boot支持哪些嵌入式Web容器?
Spring Boot支持的嵌入式servlet容器有: TomcatJettyUndertow

17. 什么是YAML?
YAML 是一种可读的数据序列化语言,它通常用于配置文件。

18. YAML 配置的优势在哪里 ?
配置有序
支持数组,数组中的元素可以是基本数据类型或者对象
简洁方便

19. Spring Boot 是否可以使用 XML 配置 ?
Spring Boot 推荐使用 Java 配置同时支持 XML 配置,通过 @ImportResource 注解加载 XML 配置。

20. application.propertiesbootstrap.properties有何区别 ?
bootstrap applicaton 优先加载,配置在应用程序上下文的引导阶段生效, 而且boostrap 里面的属性不能被覆盖;
application用于 spring boot 项目的自动化配置。

21. 什么是 Spring Profiles
Spring Profiles 允许用户根据配置文件(devprodtest等等)来注册 bean。当应用程序在开发环境中运行时,只有某些 bean 可以加载,而在生产环境中,某些其他 bean 也可以加载。比如要求 Swagger 文档仅适用于测试环境,并且禁用所有其他文档,可以使用配置文件来完成。

22. 如何在自定义端口上运行 Spring Boot 应用程序
可以在 application.properties 配置文件中指定端口,比如server.port = 8090

23. 如何实现 Spring Boot 应用程序的安全性?
为了实现 Spring Boot 的安全性,可以使用 spring-boot-starter-security 依赖,添加安全配置和重写WebSecurityConfigurerAdapter 配置类的方法。

24. 什么是 WebSocket
WebSocket 是一种计算机通信协议,通过单个 TCP 连接提供全双工通信信道。
WebSocket 是双向的 ,使用 WebSocket 客户端或服务器可以实现消息发送。
WebSocket 是全双工的 ,客户端和服务器通信是相互独立的。
WebScoket 使用单个 TCP 连接 ,与http 相比,WebSocket 消息数据交换要轻得多。

25. Spring Boot 中的监视器是什么?(什么是Spring Boot Actuator)?
Spring boot actuator spring 启动框架中的重要功能之一,Spring boot 监视器可以访问生产环境中正在运行的应用程序的当前状态。监视器模块公开了一组可直接作为 HTTP URL 访问的 REST 端点来检查状态。

26. 如何在 Spring Boot 中禁用 Actuator 端点安全性?
默认情况下,所有敏感的 HTTP 端点都是安全的,只有具有 ACTUATOR 角色的用户才能访问它们。
安全性是使用标准的 HttpServletRequest.isUserInRole 方法实施的,可以用来禁用安全性。
只有在执行机构端点在防火墙后访问时,才建议禁用安全性。

27. 什么是 CSRF 攻击?
CSRF 代表跨站请求伪造,这是一种攻击,迫使最终用户在当前通过身份验证的Web 应用程序上执行不需要的操作。CSRF 攻击专门针对状态改变请求,而不是数据窃取,因为攻击者无法查看对伪造请求的响应。

28. 如何使用 Spring Boot 实现异常处理?
Spring 通过使用 @ControllerAdvice 注解处理异常,实现一个ControllerAdvice 类来处理控制器类抛出的所有异常。

29. 如何监视所有 Spring Boot 微服务?
Spring Boot 提供监视器端点监控各个微服务,这些端点对于获取有关应用程序的信息(如它们是否已启动)以及它们的组件(如数据库等)是否正常运行很有帮助。但是用监视器的一个主要缺点是,必须单独打开应用程序的知识点以了解其状态或健康状况。

30. 运行 Spring Boot 有哪几种方式?
用命令打包或者放到容器中运行
Maven 插件运行
直接执行 main 方法运行


MyBatis 是一款优秀的持久层框架

中文官网:https://mybatis.org/mybatis-3/zh/getting-started.html

1.1、什么是MyBatis
它支持定制化 SQL、存储过程以及高级映射。
MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集。
MyBatis 可以使用简单的 XML 或注解来配置和映射原生类型、接口和 Java POJOPlain Old Java Objects,普通老式 Java 对象)为数据库中的记录。
 

1.2持久化

持久化就是将程序的数据在持久状态和瞬时状态转化的过程
内存:断电即失
数据库(JDBC),io文件持久化。
生活:冷藏、罐头。

为什么需要持久化?------------------------有一些对象,不能让他丢掉。 

1.3、持久层
Dao层、Service层、Controller层等

完成持久化工作的代码块 层界限十分明显 

1.4、为什么需要Mybatis
 
帮助程序员将数据存入到数据库中。

方便

传统的JDBC代码太复杂了。简化、框架、自动化。

不用Mybatis也可以。更容易上手。技术没有高低之分

优点:

    简单易学
    灵活
    sql和代码的分离,提高了可维护性。
    提供映射标签,支持对象与数据库的orm字段关系映射
    提供对象关系映射标签,支持对象关系组建维护
    提供xml标签,支持编写动态sql

可能会遇到的问题:

  •     配置文件没有注册
  •     绑定接口错误
  •     方法名不对
  •     返回类型不对
  •     Maven导出资源问题

3CRUD
3.1namespace
namespace中的包名要和Dao/mapper接口的包名保持一致

3.2select 
id就是对应的namespace中的方法名;

resultTypeSql语句执行的返回值!

parameterType参数类型!

  1. package com.rui.dao;
  2. import com.rui.pojo.User;
  3. import java.util.List;
  4. public interface UserMapper {
  5.     //根据id查询用户
  6.     User getUserById(int id);
  7. }

编写对应的mapper中的sql语句 

  1.    <select id="getUserById" resultType="com.rui.pojo.User" parameterType="int">
  2.        /*定义sql*/
  3.        select * from mybatis.user where id = #{id};
  4.    </select>

测试

  1.     @Test
  2.     public void getUserById(){
  3.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  4.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  5.         User user = mapper.getUserById(1);
  6.         System.out.println(user);
  7.         sqlSession.close();
  8.     }

 3.3Insert

3.4Update

3.5Delete

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE configuration
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-config.dtd">
  5. <!--configuration核心配置文件-->
  6. <configuration>
  7.     <!--environments配置环境组-->
  8.     <!--default默认环境-->
  9.     <environments default="development">
  10.         <!--environment单个环境-->
  11.         <environment id="development">
  12.             <!--transactionManager配置事务管理器-->
  13.             <transactionManager type="JDBC"/>
  14.             <!--配置连接池-->
  15.             <dataSource type="POOLED">
  16.                 <property name="driver" value="com.mysql.cj.jdbc.Driver"/>
  17.                 <property name="url" value="jdbc:mysql://localhost:3306/mybatis?serverTimezone=UTC&amp;useSSL=true&amp;useUnicode=true"/>
  18.                 <property name="username" value="root"/>
  19.                 <property name="password" value="root"/>
  20.             </dataSource>
  21.         </environment>
  22.     </environments>
  23.     <!--每一个Mapper.xml需要在Mybatis核心配置文件中注册-->
  24.     <mappers>
  25.         <mapper resource="com/newer/dao/UserMapper.xml"/>
  26.     </mappers>
  27. </configuration>
  1. package com.newer.utils;
  2. import org.apache.ibatis.io.Resources;
  3. import org.apache.ibatis.session.SqlSession;
  4. import org.apache.ibatis.session.SqlSessionFactory;
  5. import org.apache.ibatis.session.SqlSessionFactoryBuilder;
  6. import java.io.IOException;
  7. import java.io.InputStream;
  8. public class MyBatisUtils {
  9.     private static SqlSessionFactory sqlSessionFactory;
  10.     static {
  11.         try {
  12.             //使用mybatis第一步、获取sqlSessionFactory对象
  13.             String resource = "mybatis-config.xml";
  14.             InputStream inputStream = Resources.getResourceAsStream(resource);
  15.             sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
  16.         } catch (IOException e) {
  17.             e.printStackTrace();
  18.         }
  19.     }
  20.     //既然有了 SqlSessionFactory,顾名思义,我们就可以从中获得 SqlSession 的实例了。
  21.     // SqlSession 完全包含了面向数据库执行 SQL 命令所需的所有方法。
  22.     public static SqlSession getSqlSession(){
  23.       return sqlSessionFactory.openSession();
  24.     }
  25. }
  1. package com.newer.pojo;
  2. //实体类
  3. public class User {
  4.     private int id;
  5.     private String name;
  6.     private String pwd;
  7.     public User() {
  8.     }
  9.     public User(int id, String name, String pwd) {
  10.         this.id = id;
  11.         this.name = name;
  12.         this.pwd = pwd;
  13.     }
  14.     public int getId() {
  15.         return id;
  16.     }
  17.     public void setId(int id) {
  18.         this.id = id;
  19.     }
  20.     public String getName() {
  21.         return name;
  22.     }
  23.     public void setName(String name) {
  24.         this.name = name;
  25.     }
  26.     public String getPwd() {
  27.         return pwd;
  28.     }
  29.     public void setPwd(String pwd) {
  30.         this.pwd = pwd;
  31.     }
  32.     @Override
  33.     public String toString() {
  34.         return "user{" +
  35.                 "id=" + id +
  36.                 ", name='" + name + '\'' +
  37.                 ", pwd='" + pwd + '\'' +
  38.                 '}';
  39.     }
  40. }
  1. package com.newer.dao;
  2. import com.newer.pojo.User;
  3. import java.util.List;
  4. public interface UserMapper {
  5.     //查询全部用户
  6.     List<User> getUserList();
  7.     //根据ID查询用户
  8.     User getUserById(int id);
  9.     //insert一个用户
  10.     int addUser(User user);
  11.     //修改用户
  12.     int updateUser(User user);
  13.     //删除用户
  14.     int deleteUser(int id);
  15. }
  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <!--namespace绑定一个对应的Mapper接口-->
  6. <mapper namespace="com.newer.dao.UserMapper">
  7.    <select id="getUserList" resultType="com.newer.pojo.User">
  8.        select * from mybatis.user
  9.    </select>
  10.     <select id="getUserById" parameterType="int" resultType="com.newer.pojo.User">
  11.         select * from mybatis.user where id=#{id}
  12.     </select>
  13.     <insert id="addUser" parameterType="com.newer.pojo.User">
  14.         insert into mybatis.user (id,name,pwd) values (#{id},#{name },#{pwd})
  15.     </insert>
  16.     <update id="updateUser" parameterType="com.newer.pojo.User">
  17.         update mybatis.user set name=#{name},pwd=#{pwd} where id=#{id}
  18.     </update>
  19.     <delete id="deleteUser" parameterType="int">
  20.         delete  from mybatis.user where id=#{id}
  21.     </delete>
  22. </mapper>
  1. package com.newer.dao;
  2. import com.newer.pojo.User;
  3. import com.newer.utils.MyBatisUtils;
  4. import org.apache.ibatis.session.SqlSession;
  5. import org.junit.Test;
  6. import java.util.List;
  7. public class UserDaoTest {
  8.     @Test
  9.     public void test(){
  10.         //第一步:获得SqlSession对象
  11.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  12.         //方式一:getMapper
  13.         UserMapper userDao = sqlSession.getMapper(UserMapper.class);
  14.         List<User> userList = userDao.getUserList();
  15.         //方式二
  16.         //List<User> userList = sqlSession.selectList("com.newer.dao.UserMapper.getUserList");
  17.         for (User user : userList) {
  18.             System.out.println(user);
  19.         }
  20.         //关闭sqlSession
  21.         sqlSession.close();
  22.     }
  23.     @Test
  24.     public void getUserById(){
  25.         //第一步:获得SqlSession对象
  26.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  27.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  28.         User user = mapper.getUserById(1);
  29.         System.out.println(user);
  30.         //关闭sqlSession
  31.         sqlSession.close();
  32.     }
  33.     //增删改需要提交事务
  34.     @Test
  35.     public void addUser(){
  36.         //第一步:获得SqlSession对象
  37.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  38.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  39.         int res = mapper.addUser(new User(4, "哈哈", "12364"));
  40.         if(res>0){
  41.             System.out.println("插入成功!");
  42.         }
  43.         //提交事务
  44.         sqlSession.commit();
  45.         //关闭sqlSession
  46.         sqlSession.close();
  47.     }
  48.     @Test
  49.     public void updateUser(){
  50.         //第一步:获得SqlSession对象
  51.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  52.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  53.         int res = mapper.updateUser(new User(4,"呵呵","88888"));
  54.         if(res>0){
  55.             System.out.println("修改成功!");
  56.         }
  57.         //提交事务
  58.         sqlSession.commit();
  59.         //关闭sqlSession
  60.         sqlSession.close();
  61.     }
  62.     @Test
  63.     public void deleteUser(){
  64.         //第一步:获得SqlSession对象
  65.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  66.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  67.         int res = mapper.deleteUser(4);
  68.         if(res>0){
  69.             System.out.println("删除成功!");
  70.         }
  71.         //提交事务
  72.         sqlSession.commit();//这里就是提交事务!!!!!!!!!!!!!!!!!!
  73.         //关闭sqlSession
  74.         sqlSession.close();
  75.     }
  76. }

注意点:增删改需要提交事务!!!!!!!!!!!!!!!!!!

3.6、分析错误 

  • 标签不要匹配错误
  • resource绑定mapper,需要使用路径!
  • 程序配置文件必须符合规范
  • NullPointerException,没有注册到资源
  • 输出的xml文件中存在中文乱码问题
  • maven资源没有导出问题

3.7、万能Map 

我们的实体类,或者数据库中的表,字段或者参数过多,我们应当考虑使用Map

int addUser2(Map<String,Object> map);

  1. <!--对象中的属性,可以直接取出来 parameterType=传递map中的key-->
  2. <insert id="addUser2" parameterType="map">
  3.     insert into mybatis.user (id, name, pwd) values (#{userId},#{userName},#{password});
  4. </insert>
  1. @Test
  2. public void addUser2(){
  3. SqlSession sqlSession = MyBatisUtils.getSqlSession();
  4. UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  5. HashMap<String, Object> map = new HashMap<>();
  6. map.put(“userId”,4);
  7. map.put(“userName”,“闯哥”);
  8. map.put(“password”,“yuzhouzuishuai”);
  9. mapper.addUser2(map);
  10. //提交事务
  11. sqlSession.commit();
  12. sqlSession.close();
  13. }
  • Map传递参数,直接在sql中取出key即可!【parameterType=“map”
  • 对象传递参数,直接在sql中取对象的属性即可!【parameterType=“Object”
  • 只有一个基本类型参数的情况下,可以直接在sql中取到!
  • 多个参数用Map,或者注解!

3.8、思考模糊查询 

Java代码执行的时候传递通配符%%

List<User> userList=mapper.getUserLike("%%");

sql拼接中使用通配符!

select * from mybatis.user where name like "%"#{value}"%"                          ---------可能存在SQL注入问题
 

什么是SQL 注入?
SQL 注入是一种非常常见的数据库攻击手段,SQL 注入漏洞也是网络世界中最普遍的漏洞 之一。大家也许都听过某某学长通过攻击学校数据库修改自己成绩的事情,这些学长们一 般用的就是 SQL 注入方法。

SQL 注入其实就是恶意用户通过在表单中填写包含 SQL 关键字的数据来使数据库执行非常 规代码的过程。简单来说,就是数据「越俎代庖」(yuè zǔ dài páo)做了代码才能干的 事情。这个问题的来源是,SQL 数据库的操作是通过 SQL 语句来执行的,而无论是执行代 码还是数据项都必须写在 SQL 语句之中,这就导致如果我们在数据项中加入了某些 SQL 句关键字(比如说 SELECTDROP 等等),这些关键字就很可能在数据库写入或读取数据 时得到执行。
如何解决:

1】在JDBC中,使用Statement的子类PreparedStatement,这也是我们最先想到的方法,事先将sql语句传入PreparedStatement中,等会要传入的参数使用?代替,那么该sql语句会进行预编译,之后将前台获取的参数通过set方式传入编译后的sql语句中,那么此时被注入的sql语句无法得到编译,从而避免了sql注入的问题。而且使用PreparedStatement在一定程度上有助于数据库执行性能的提升。

2Mybatis中尽量使用#{}而不是${}

#{}就像是JDBC中的?,mybatis通过使用#{}的方式,代表该语句在执行之前会经过预编译的过程,后来传入的参数通过占位符的方式填入到已经编译完的语句中。但使用${}的参数会直接参与编译,不能避免sql注入。如果需求中非要使用到${}的话,只能手动过滤了。

 3】手动过滤参数

声明一个危险参数数组danger,将前台传入的参数用“ ”分隔后,产生的参数数组与danger有交集时,终止该sql的执行,并反馈前台,提示禁止输入非法字符。

4Mybatisbind标签使用,可解决模糊查询sql 注入,参数名不一致等  见如下代码块

  1.  <if test="titledescription != null and titledescription != ''">
  2.     <bind name="titledescription" value="'%'+titledescription+'%'"/>
  3.     and titleDescription like  #{titledescription}
  4.     或者
  5.     like concat('%', #{titledescription }, '%')
  6.  </if>

4、配置解析

4.1、核心配置文件

mybatis-config.xml
MyBatis
的配置文件包含了会深深影响MyBatis行为的设置和属性信息

configuration(配置)
properties(属性)
settings(设置)
typeAliases(类型别名)
typeHandlers(类型处理器)
objectFactory(对象工厂)
plugins(插件)
environments(环境配置)
environment(环境变量)
transactionManager(事务管理器)
dataSource(数据源)
databaseIdProvider(数据库厂商标识)
mappers(映射器)

 4.2、环境配置(environments

MyBatis 可以配置成适应多种环境

不过要记住:尽管可以配置多个环境,但每个 SqlSessionFactory 实例只能选择一种环境。

学会使用配置多套运行环境!

MyBatis默认的事务管理器就是JDBC,连接池:POOLED

4.3、属性(properties 

可以通过properties属性来实现引用配置文件

这些属性都是可外部配置且可动态替换的,既可以在典型的 Java 属性文件中配置,亦可通过 properties 元素的子元素来传递。

db.properties  

  1. driver=com.mysql.jdbc.Driver
  2. url=jdbc:mysql://localhost:3306/mybatis?
  3. useSSL=true&useUnicode=true&characterEncoding=utf8
  4. username=root
  5. password=Cc105481

 在核心配置文件中引入

  1. <properties resource="db.properties">
  2.     <property name="username" value="root"/>
  3.     <property name="password" value="Cc105481"/>
  4. </properties>

 可以直接引入外部文件 可以在其中增加一些属性配置 如果两个文件有同一字段,优先使用外部配置文件的!

 4.4、类型别名(typeAliases

类型别名是为 Java 类型设置一个短的名字。 存在的意义仅在于用来减少类完全限定名的冗余。

  1.   <!--可以给实体类起别名-->
  2.     <typeAliases>
  3.         <typeAlias type="com.rui.pojo.User" alias="User"/>
  4.     </typeAliases>

  可以指定一个包名,MyBatis 会在包名下面搜索需要的 Java Bean,比如: 扫描实体类的包,他的默认别名就为这个类的类名,首字母小写!

  1. <!--可以给实体类起别名-->
  2.     <typeAliases>
  3.         <package name="com.rui.pojo"/>
  4.     </typeAliases>
  • 在实体类比较少的时候,使用第一种方式。
  • 如果实体类十分多,建议使用第二种方式。

 第一种可以DIY别名,第二种则不行,如果非要改,需要在实体类(pojo)上增加@Alias注解 

  1. @Alias(“author”)
  2. public class Author {
  3. }

 4.5MyBatis设置

这是 MyBatis 中极为重要的调整设置,它们会改变 MyBatis 的运行时行为----------------Settings

  4.6、映射器(mappers

  1.  <!--每一个Mapper.xml需要在Mybatis核心配置文件中注册-->
  2.     <mappers>
  3.        <!-- <mapper resource="com/newer/dao/UserMapper.xml"/>-->
  4.        <!-- <mapper class="com.newer.dao.UserMapper"></mapper>-->
  5.         <package name="com.newer.dao"/>
  6.     </mappers>

除了第一种没有限制条件,其他两种有限制条件

  • 接口和它的Mapper配置文件必须同名
  •  接口和它的Mapper配置文件必须在同一个包下

 4.7MyBatis生命周期和作用域

生命周期,和作用域是至关重要的,因为错误的使用会导致非常严重的并发问题   !!!!!!!!

SqlSessionFactoryBuilder

  • 一旦创建了SqlSessionFactory,就不再需要它了
  • 局部变量

mybatis运行流程SqlSessionFactory

可以想象为:数据库连接池
SqlSessionFactory 一旦被创建就应该在应用的运行期间一直存在,没有任何理由丢弃它或重新创建另一个实例。
因此SqlSessionFactory的最佳作用域是应用作用域。
最简单的就是使用 单例模式 或者静态单例模式

SqlSession

  • 连接到连接池的一个请求!
  • SqlSession 的实例不是线程安全的,因此是不能被共享的,所以它的最佳的作用域是请求或方法作用域。
  • 用完之后需要赶紧关闭,否则会占用资源
  • SqlSessionFactory

这里的每一个Mapper,就代表一个具体的业务! 

  5resultMap

结果集映射

  • id name pwd
  • id name password
  1. <resultMap id="UserMap" type="User">
  2.     <!--column数据库中的字段,property实体类中的属性-->
  3.     <result column="    id" property="id"/>
  4.     <result column="name" property="name"/>
  5.     <result column="pwd" property="password"/>
  6. </resultMap>
  7. <select id="getUserById" resultMap="UserMap" parameterType="int">
  8.    /*定义sql*/
  9.    select * from mybatis.user where id = #{id};

resultMap 元素是 MyBatis 中最重要最强大的元素
ResultMap 的设计思想是,对于简单的语句根本不需要配置显式的结果映射,而对于复杂一点的语句只需要描述它们的关系就行了。
ResultMap 最优秀的地方在于,虽然你已经对它相当了解了,但是根本就不需要显式地用到他们。
 

6、日志

6.1、日志工厂

如果一个数据库操作,出现了异常,我们需要排错。日志就是最好的助手!

  • 曾经:soutdebug
  • 现在:日志工厂

正在上传…重新上传取消

  • SLF4J
  • LOG4J【掌握】
  • LOG4J2
  • JDK_LOGGING
  • COMMONS_LOGGING
  • STDOUT_LOGGING【掌握】
  • NO_LOGGING

STDOUT_LOGGING标准日志输出 

  1.   <settings>
  2.         <setting name="logImpl" value="STDOUT_LOGGING"/>
  3.     </settings>

 mybatis核心配置文件中,配置我们的日志!

6.2Log4j 

 Log4jApache的一个开源项目,通过使用Log4j,我们可以控制日志信息输送的目的地是控制台、文件、GUI组件
我们也可以控制每一条日志的输出格式

通过定义每一条日志信息的级别,我们能够更加细致地控制日志的生成过程。
通过一个配置文件来灵活地进行配置,而不需要修改应用的代码。

有一些公司也用sl4j 大同小异

导入log4j的包 

  1. <dependencies>
  2.     <!-- https://mvnrepository.com/artifact/log4j/log4j -->
  3.     <dependency>
  4.         <groupId>log4j</groupId>
  5.         <artifactId>log4j</artifactId>
  6.         <version>1.2.17</version>
  7.     </dependency>
  8. </dependencies>

 log4j.properties

  1. #将等级为DEBUG的日志信息输出到consolefile这两个目的地,consolefile的定义在下面的代码
  2. log4j.rootLogger=DEBUG,console,file
  3. #控制台输出的相关设置
  4. log4j.appender.console = org.apache.log4j.ConsoleAppender
  5. log4j.appender.console.Target = System.out
  6. log4j.appender.console.Threshold=DEBUG
  7. log4j.appender.console.layout = org.apache.log4j.PatternLayout
  8. log4j.appender.console.layout.ConversionPattern=[%c]-%m%n
  9. #文件输出的相关设置
  10. log4j.appender.file = org.apache.log4j.RollingFileAppender
  11. log4j.appender.file.File=./log/kuang.log
  12. log4j.appender.file.MaxFileSize=10mb
  13. log4j.appender.file.Threshold=DEBUG
  14. log4j.appender.file.layout=org.apache.log4j.PatternLayout
  15. log4j.appender.file.layout.ConversionPattern=[%p][%d{yy-MM-dd}][%c]%m%n
  16. #日志输出级别
  17. log4j.logger.org.mybatis=DEBUG
  18. log4j.logger.java.sql=DEBUG
  19. log4j.logger.java.sql.Statement=DEBUG
  20. log4j.logger.java.sql.ResultSet=DEBUG
  21. log4j.logger.java.sql.PreparedStatement=DEBUG

配置log4j为日志实现 

  1. <settings>
  2.     <setting name="logImpl" value="LOG4J"/>
  3. </settings>

log4j的使用!直接测试运行刚才的查询

  1. DEBUG [main] (LogFactory.java:105) - Logging initialized using ‘class org.apache.ibatis.logging.log4j.Log4jImpl’ adapter.
  2. DEBUG [main] (LogFactory.java:105) - Logging initialized using ‘class org.apache.ibatis.logging.log4j.Log4jImpl’ adapter.
  3. DEBUG [main] (PooledDataSource.java:353) - PooledDataSource forcefully closed/removed all connections.
  4. DEBUG [main] (PooledDataSource.java:353) - PooledDataSource forcefully closed/removed all connections.
  5. DEBUG [main] (PooledDataSource.java:353) - PooledDataSource forcefully closed/removed all connections.
  6. DEBUG [main] (PooledDataSource.java:353) - PooledDataSource forcefully closed/removed all connections.
  7. DEBUG [main] (JdbcTransaction.java:136) - Opening JDBC Connection
  8. Loading class com.mysql.jdbc.Driver'. This is deprecated. The new driver class iscom.mysql.cj.jdbc.Driver’. The driver is automatically registered via the SPI and manual loading of the driver class is generally unnecessary.
  9. DEBUG [main] (PooledDataSource.java:424) - Created connection 2049051802.
  10. DEBUG [main] (JdbcTransaction.java:100) - Setting autocommit to false on JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@7a220c9a]
  11. DEBUG [main] (BaseJdbcLogger.java:143) - ==> Preparing: /定义sql/ select * from mybatis.user where id = ?;
  12. DEBUG [main] (BaseJdbcLogger.java:143) - > Parameters: 1(Integer)
  13. DEBUG [main] (BaseJdbcLogger.java:143) - < Total: 1
  14. User{id=1, name=‘狂神’, password=‘123456’}
  15. DEBUG [main] (JdbcTransaction.java:122) - Resetting autocommit to true on JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@7a220c9a]
  16. DEBUG [main] (JdbcTransaction.java:90) - Closing JDBC Connection [com.mysql.cj.jdbc.ConnectionImpl@7a220c9a]
  17. DEBUG [main] (PooledDataSource.java:381) - Returned connection 2049051802 to pool.
  18. Disconnected from the target VM, address: ‘127.0.0.1:58296’, transport: ‘socket’
  19. Process finished with exit code 0

 简单使用

  1. //在要使用Log4j 的类中,导入org.apache.log4j.Logger;
  2. //日志对象,加载参数为当前类的class
  3.  static Logger logger = Logger.getLogger(UserDaoTest.class);
  4. //日志级别
  5. logger.info("info:进入了testLog4j方法");
  6. logger.debug("debug:进入了testLog4j");
  7. logger.error("error:进入了testLog4j");

 7、分页

为什么要分页?------------减少数据的处理量

 7.1、使用Limit分页

select * from user limit startIndex,pageSize

使用Mybatis实现分页,核心SQL

  1. //分页
  2. List<User> getUserByLimit(Map<String,Integer> map);

 Mapper.xml

  1. <!--分页-->
  2. <select id="getUserByLimit" parameterType="map" resultMap="UserMap">
  3.     select * from mybatis.user limit #{startIndex},#{pageSize}
  4. </select>

测试 

  1. @Test
  2.     public void getUserByLimit(){
  3.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  4.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  5.         HashMap<String, Integer> map = new HashMap<>();
  6.         map.put("startIndex",0);
  7.         map.put("pageSize",2);
  8.    List<User> userList = mapper.getUserByLimit(map);
  9.    for (User user : userList) {
  10.        System.out.println(user);
  11.    }
  12.    sqlSession.close();
  13. }

 7.2分页插件

8、使用注解开发

8.1、面向接口编程

面向接口编程的根本原因:解耦,可拓展,提高复用,分层开发中、上层不用管具体的实现,大家都遵守共同的标准,使得开发变得容易,规范性好

 8.2、使用注解开发

注解在接口上实现

  1. @Select(value = "select * from user")
  2. List<User> getUsers();

需要在核心配置文件中绑定接口!

  1. <!--绑定接口-->
  2. <mappers>
  3.     <mapper class="rui.dao.UserMapper"/>
  4. </mappers>

 测试

  1. public class UserMapperTest {
  2. @Test
  3. public void test(){
  4. SqlSession sqlSession = MyBatisUtils.getSqlSession();
  5. //底层主要应用反射
  6. UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  7. List users = mapper.getUsers();
  8. for (User user : users) {
  9. System.out.println(user);
  10. }
  11. sqlSession.close();
  12. }
  13. }
  • 本质:反射机制实现
  • 底层:动态代理!

 8.3CRUD

在工具类创建的时候实现自动提交事务!

  1. public static SqlSession getSqlSession(){
  2. return sqlSessionFactory.openSession(true);
  3. }

 编写接口,增加注解

  1. public interface UserMapper {
  2. @Select(value = “select * from user”)
  3. List getUsers();
  4. //方法存在多个参数,所有的参数前面必须加上@Param注解
  5. @Select("select * from user where id = #{id} or name = #{name}")
  6. User getUserByID(@Param("id")int id,@Param("name")String name);
  7. @Insert("insert into user(id,name,pwd) values (#{id},#{name},#{password})")
  8. int addUser(User user);
  9. @Update("update user set name = #{name},pwd = #{password} where id = #{id}")
  10. int updateUser(User user);
  11. @Delete("delete from user where id = #{uid}")
  12. int deleteUser(@Param("uid") int id);
  13. }

 测试类

【注意:我们必须要将接口注册绑定到我们的核心配置文件中!

关于@Param()注解

  • 基本类型的参数或者String类型,需要加上
  • 引用类型不需要加
  • 如果只有一个基本类型的话,可以忽略,但是建议大家都加上
  • 我们在SQL中引用的就是我们这里的@Param()中设定的属性名

 9Lombok

Project Lombok is a java library that automatically plugs into your editor and build tools, spicing up your java.

Project Lombok是一个java库,它可以自动插入你的编辑器和构建工具中,为你的java添彩。

Never write another getter or equals method again, with one annotation your class has a

再也不用写另一个getterequals方法了,只需一个注释,你的类就有了一个

fully featured builder, Automate your logging variables, and much more.

功能齐全的构建器,自动记录变量,还有更多。

 使用步骤:

  1. //IDEA中安装Lombok插件
  2. //在项目中导入lombokjar
  3.  <!-- https://mvnrepository.com/artifact/org.projectlombok/lombok -->
  4.         <dependency>
  5.             <groupId>org.projectlombok</groupId>
  6.             <artifactId>lombok</artifactId>
  7.             <version>1.18.10</version>
  8.         </dependency>
  9. //在实体类上加注解即可

@Getter and @Setter
@ToString
@EqualsAndHashCode
@AllArgsConstructor, @RequiredArgsConstructor and @NoArgsConstructor
@Log, @Log4j, @Log4j2, @Slf4j
@Data

 10、多对一处理

多对一:

  • 多个学生,对应一个老师
  • 对于学生这边而言,关联...多个学生,关联一个老师【多对一】
  • 对于老师而言,集合,一个老师又很多学生【一对多】
  1. CREATE TABLE teacher(
  2. id int(10) Not null,
  3. name VARCHAR(30) DEFAULT NULL,
  4. PRIMARY KEY (id)
  5. )ENGINE=INNODB DEFAULT CHARSET=utf8
  6. INSERT INTO teacher(id,name) VALUES (1,‘全’);
  7. CREATE TABLE student(
  8. id int(10) Not null,
  9. name VARCHAR(30) DEFAULT NULL,
  10. tid INT(10) DEFAULT NULL,
  11. PRIMARY KEY (id),
  12. KEY fktid(tid),
  13. CONSTRAINT fktid FOREIGN KEY (tid) REFERENCES teacher (id)
  14. )ENGINE=INNODB DEFAULT CHARSET=utf8
  15. INSERT INTO student(id,name,tid) VALUES (1,‘张三’,1);
  16. INSERT INTO student(id,name,tid) VALUES (2,‘李四’,1);
  17. INSERT INTO student(id,name,tid) VALUES (3,‘王五’,1);
  18. INSERT INTO student(id,name,tid) VALUES (4,‘赵六’,1);
  19. INSERT INTO student(id,name,tid) VALUES (5,‘帅全’,1);

测试环境:

  1. 导入lombok 新建实体类TeacherStudent
  2. 新建Mapper接口
  3. 建立Mapper.XML文件
  4. 在核心配置文件中绑定注册我们的MApper接口或者文件!【方式很多,随意选】
  5. 测试查询是否成功!

 按照查询嵌套处理

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.newer.dao.StudentMapper">
  6.     <!--1.查询所有学生信息
  7.         2.根据查询出来的学生的tid,寻找对应的老师
  8.     -->
  9.     <select id="getStudent" resultMap="StudentTeacher">
  10.         select * from student
  11.     </select>
  12.     <resultMap id="StudentTeacher" type="Student">
  13.         <result property="id" column="id"></result>
  14.         <result property="name" column="name"></result>
  15.         <!--复杂的属性,需要单独处理
  16.         对象:association
  17.         集合:collection
  18.         -->
  19.         <association property="teacher" column="tid" javaType="Teacher" select="getTeacher"></association>
  20.     </resultMap>
  21.     
  22.     <select id="getTeacher" resultType="Teacher">
  23.         select * from teacher where id=#{id}
  24.     </select>
  25. </mapper>

 按照结果嵌套处理

  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.newer.dao.StudentMapper">
  6.     <!--按照结果嵌套处理-->
  7.     <select id="getStudent2" resultMap="StudentTeacher2">
  8.         select s.id sid,s.name sname,t.name tname,t.id tid
  9.         from student s,teacher t
  10.         where s.tid=t.id
  11.     </select>
  12.     <resultMap id="StudentTeacher2" type="Student">
  13.         <result property="id" column="sid"></result>
  14.         <result property="name" column="sname"></result>
  15.         <association property="teacher" javaType="Teacher">
  16.             <result property="id" column="tid"></result>
  17.             <result property="name" column="tname"></result>
  18.         </association>
  19.     </resultMap>
  20.  </mapper>

 回顾Mysql多对一查询方式: 

  • 子查询
  • 多表联查

 11、一对多处理

比如:一个老师拥有多个学生!

对于老师而言,就是一对多的关系!

  1. @Data
  2. public class Teacher {
  3. private int id;
  4. private String name;
  5. //一个老师拥有多个学生
  6. private List<Student> students;
  7. }
  1. @Data
  2. public class Student {
  3. private int id;
  4. private String name;
  5. private int tid;
  6. }
  1. //按照结果嵌套处理
  2. select s.id sid,s.name sname,t.name tname,t.id tid from student s,teacher t where s.tid=t.id and t.id = #{tid}
  3. //按照查询嵌套处理
  4. select * from mybatis.teacher where id = #{tid} select * from mybatis.student where tid = #{tid}
  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.newer.dao.TeacherMapper">
  6.     <select id="getTeacher" resultMap="TeacherStudent">
  7.         select s.id sid ,s.name sname,t.name tname,t.id tid
  8.         from student s,teacher t
  9.         where s.tid=t.id and t.id=#{tid}
  10.     </select>
  11.     <resultMap id="TeacherStudent" type="Teacher">
  12.         <result property="id" column="tid"></result>
  13.         <result property="name" column="tname"></result>
  14.         <collection property="students" ofType="Student">
  15.             <result property="id" column="sid"></result>
  16.             <result property="name" column="sname"></result>
  17.             <result property="tid" column="tid"></result>
  18.          </collection>
  19.     </resultMap>
  20.     <!--=====================================================-->
  21.     <select id="getTeacher2" resultMap="TeacherStudent2">
  22.         select * from teacher where id=#{tid}
  23.     </select>
  24.     <resultMap id="TeacherStudent2" type="Teacher">
  25.         <collection property="students" column="id" javaType="ArrayList" ofType="Student" select="getStudentByTeacherId"></collection>
  26.     </resultMap>
  27.     <select id="getStudentByTeacherId" resultType="Student">
  28.         select * from student where tid=#{tid}
  29.     </select>
  30. </mapper>

关联-association【多对一】

集合-collection 【一对多】

javaType & ofType

  • JavaType用来指定实体类中属性的类型
  • ofType用来指定映射到List或者集合中的pojo类型,泛型中的约束类型!

 注意点:

  • 保证SQL的可读性,尽量保证通俗易懂
  • 注意一对多和多对一中,属性名和字段的问题!
  • 如果问题不好排查错误,可以使用日志,建议使用Log4j

面试高频:

  • Mysql引擎
  • InnoDB底层原理
  • 索引
  • 索引优化!

 12、动态SQL

动态SQL:动态SQL就是指根据不同的条件生成不同的SQL语句

动态 SQL 元素和 JSTL 或基于类似 XML 的文本处理器相似。在 MyBatis 之前的版本中,有很多元素需要花时间了解。MyBatis 3 大大精简了元素种类,现在只需学习原来一半的元素便可。MyBatis 采用功能强大的基于 OGNL 的表达式来淘汰其它大部分元素。

 动态SQL:动态SQL就是指根据不同的条件生成不同的SQL语句

动态 SQL 元素和 JSTL 或基于类似 XML 的文本处理器相似。在 MyBatis 之前的版本中,有很多元素需要花时间了解。MyBatis 3 大大精简了元素种类,现在只需学习原来一半的元素便可。MyBatis 采用功能强大的基于 OGNL 的表达式来淘汰其它大部分元素。

搭建环境 

  1. CREATE TABLE blog(
  2. id VARCHAR(50) NOT NULL COMMENT ‘博客id’,
  3. title VARCHAR(100) not null comment ‘博客标题’,
  4. author VARCHAR(30) not null comment ‘博客作者’,
  5. creat_time datetime not null comment ‘创建时间’,
  6. views int(30) not null comment ‘浏览量
  7. )ENGINE=InnoDB DEFAULT CHARSET=utf8

 创建一个基础工程

  1.  导包
  2. 编写配置文件
  3. 编写实体类
  4. 编写实体类对应的Mapper接口和Mapper.xml
  1. package com.newer.pojo;
  2. import lombok.Data;
  3. import java.util.Date;
  4. @Data
  5. public class Blog {
  6.     private String id;
  7.     private String title;
  8.     private String author;
  9.     private Date creatTime;
  10.     private int views;
  11. }
  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE configuration
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-config.dtd">
  5. <!--configuration核心配置文件-->
  6. <configuration>
  7.     <settings>
  8.         <setting name="logImpl" value="STDOUT_LOGGING"/>
  9.         <setting name="mapUnderscoreToCamelCase" value="true"/>
  10.     </settings>
  11.     <typeAliases>
  12.         <package name="com.newer.pojo"/>
  13.     </typeAliases>
  14.     <!--environments配置环境组-->
  15.     <!--default默认环境-->
  16.     <environments default="development">
  17.         <!--environment单个环境-->
  18.         <environment id="development">
  19.             <!--transactionManager配置事务管理器-->
  20.             <transactionManager type="JDBC"/>
  21.             <!--配置连接池-->
  22.             <dataSource type="POOLED">
  23.                 <property name="driver" value="com.mysql.cj.jdbc.Driver"/>
  24.                 <property name="url" value="jdbc:mysql://localhost:3306/mybatis?serverTimezone=UTC&amp;useSSL=true&amp;useUnicode=true"/>
  25.                 <property name="username" value="root"/>
  26.                 <property name="password" value="root"/>
  27.             </dataSource>
  28.         </environment>
  29.     </environments>
  30.     <!--每一个Mapper.xml需要在Mybatis核心配置文件中注册-->
  31.     <mappers>
  32.         <mapper class="com.newer.dao.BlogMapper"/>
  33.     </mappers>
  34. </configuration>
  1. package com.newer.utils;
  2. import org.apache.ibatis.io.Resources;
  3. import org.apache.ibatis.session.SqlSession;
  4. import org.apache.ibatis.session.SqlSessionFactory;
  5. import org.apache.ibatis.session.SqlSessionFactoryBuilder;
  6. import java.io.IOException;
  7. import java.io.InputStream;
  8. public class MyBatisUtils {
  9.     private static SqlSessionFactory sqlSessionFactory;
  10.     static {
  11.         try {
  12.             //使用mybatis第一步、获取sqlSessionFactory对象
  13.             String resource = "mybatis-config.xml";
  14.             InputStream inputStream = Resources.getResourceAsStream(resource);
  15.             sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
  16.         } catch (IOException e) {
  17.             e.printStackTrace();
  18.         }
  19.     }
  20.     //既然有了 SqlSessionFactory,顾名思义,我们就可以从中获得 SqlSession 的实例了。
  21.     // SqlSession 完全包含了面向数据库执行 SQL 命令所需的所有方法。
  22.     public static SqlSession getSqlSession(){
  23.       return sqlSessionFactory.openSession(true);
  24.     }
  25. }
  1. package com.newer.utils;
  2. import org.junit.Test;
  3. import java.util.UUID;
  4. @SuppressWarnings("all")//抑制警告
  5. public class IDutils {
  6.     public static String getId(){
  7.         return UUID.randomUUID().toString().replace("-","");
  8.     }
  9.     @Test
  10.     public void test(){
  11.         System.out.println(IDutils.getId());
  12.     }
  13. }
  1. package com.newer.dao;
  2. import com.newer.pojo.Blog;
  3. import java.util.List;
  4. import java.util.Map;
  5. public interface BlogMapper {
  6.     //插入数据
  7.     int addBlog(Blog blog);
  8.     //查询博客
  9.     List<Blog> queryBlogIF(Map map);
  10. }
  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.newer.dao.BlogMapper">
  6.     <insert id="addBlog" parameterType="Blog" >
  7.         insert into mybatis.blog(id,title,author,creat_time,views)
  8.         values (#{id},#{title},#{author},#{creatTime},#{views})
  9.     </insert>
  10.     <select id="queryBlogIF" parameterType="map" resultType="Blog">
  11.         select * from mybatis.blog where 1=1
  12.         <if test="title !=null">
  13.             and title=#{title}
  14.         </if>
  15.         <if test="author !=null">
  16.             and author=#{author}
  17.         </if>
  18.     </select>
  19. </mapper>

IF

select * from mybatis.bolg where 1=1 and title = #{title} and author = #{author}

  1. import com.newer.dao.BlogMapper;
  2. import com.newer.pojo.Blog;
  3. import com.newer.utils.IDutils;
  4. import com.newer.utils.MyBatisUtils;
  5. import org.apache.ibatis.session.SqlSession;
  6. import org.junit.Test;
  7. import java.util.Date;
  8. import java.util.HashMap;
  9. import java.util.List;
  10. public class MyTest {
  11.     @Test
  12.     public void addInitBlog(){
  13.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  14.         BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
  15.         Blog blog = new Blog();
  16.         blog.setId(IDutils.getId());
  17.         blog.setTitle("Mybatis如此简单");
  18.         blog.setAuthor("闯哥说");
  19.         blog.setCreatTime(new Date());
  20.         blog.setViews(9999);
  21.         mapper.addBlog(blog);
  22.         blog.setId(IDutils.getId());
  23.         blog.setTitle("Java如此简单");
  24.         mapper.addBlog(blog);
  25.         blog.setId(IDutils.getId());
  26.         blog.setTitle("Spring如此简单");
  27.         mapper.addBlog(blog);
  28.         blog.setId(IDutils.getId());
  29.         blog.setTitle("微服务如此简单");
  30.         mapper.addBlog(blog);
  31.         sqlSession.close();
  32.     }
  33.     @Test
  34.     public void queryBlogIF(){
  35.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  36.         BlogMapper mapper = sqlSession.getMapper(BlogMapper.class);
  37.         HashMap map = new HashMap();
  38.         map.put("author","G");
  39.         List<Blog> blogs = mapper.queryBlogIF(map);
  40.         for (Blog blog : blogs) {
  41.             System.out.println(blog);
  42.         }
  43.         sqlSession.close();
  44.     }
  45. }

所谓的动态SQL,本质还是SQL语句,只是我们可以在SQL层面,去执行一些逻辑代码

if

Where,set,choose,when

SQL片段-------有的时候,我们可能会将一些公共的部分抽取出来,方便复用!

使用SQL标签抽取公共的部分 

  1. <sql id="if-title-author">
  2.     <if test="title != null">
  3.         title = #{title}
  4.     </if>
  5.     <if test="author != null">
  6.         and author = #{author}
  7.     </if>
  8. </sql>

 在需要使用的地方使用Include标签引用即可

  1. <select id="queryBlogIF" parameterType="map" resultType="com.rui.pojo.Blog">
  2.     select * from mybatis.bolg
  3.     <where>
  4.         <include refid="if-title-author"></include>
  5.     </where>
  6. </select>

注意事项:

  • 最好基于单表来定义SQL片段!
  • 不要存在where或者set标签,片段里尽量只有if就好了

 foreach

 select * from user where 1=1 and (id=1 or id=2 or id=3)

   select * from mybatis.bolg   
   <where>
   <foreach collection="ids" item="id" open="(" close=")" separator="or">
       id = #{id}
   </foreach>
   </where>

 动态SQL就是在拼接SQL语句,我们只要保证SQL的正确性,按照SQL的格式,去排列组合就可以了建议:先在Mysql中写出完整的SQL,在对应的去修改称为我们的动态SQL

13、缓存(了解)

13.1、简介

查询  连接数据库,耗资源!
一次查询的结果,给他暂存在一个可以直接取到的地方!—>内存 缓存

我们再次查询相同数据的时候,直接走缓存,就不用走数据库了

缓存[Cache]: 

存在内存中的临时数据。
将用户经常查询的数据放在缓存(内存)中,用户去查询数据就不用从磁盘上(关系型数据库数据文件)查询,
从缓存中查询,从而提高查询效率,解决了高并发系统的性能问题。
为什么使用缓存?

减少和数据库的交互次数,减少系统开销,提高系统效率。
什么样的数据能使用缓存?

经常查询并且不经常改变的数据

MyBatis包含一个非常强大的查询缓存特性,它可以非常方便地定制和配置缓存。缓存可以极大的提升查询效率。

MyBatis系统中默认定义了两级缓存:一级缓存和二级缓存

默认情况下,只有一级缓存开启。(SqlSession级别的缓存,也称为本地缓存)
二级缓存需要手动开启和配置,他是基于namespace级别的缓存。
为了提扩展性,MyBatis定义了缓存接口Cache。我们可以通过实现Cache接口来自定义二级缓存

13.3、一级缓存 

一级缓存也叫本地缓存:SqlSession

  • 与数据库同义词会话期间查询到的数据会放在本地缓存中。
  • 以后如果需要获取相同的数据,直接从缓存中拿,没有必要再去查询数据;

测试步骤:

  1. 开启日志!
  2. 测试在一个Session中查询两次相同的记录
  3. 查看日志输出

缓存失效的情况:

  • 查询不同的东西
  • 增删改操作,可能会改变原来的数据,所以必定会刷新缓存!
  • 查询不同的Mapper.xml 手动清理缓存!
  • sqlsession.clearCache(); //手动清理缓存
  1. import com.newer.dao.UserMapper;
  2. import com.newer.pojo.User;
  3. import com.newer.utils.MyBatisUtils;
  4. import org.apache.ibatis.session.SqlSession;
  5. import org.junit.Test;
  6. public class MyTest {
  7.     @Test
  8.     public void test(){
  9.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  10.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  11.         User user = mapper.queryUser(1);
  12.         System.out.println(user);
  13.         //调用修改方法
  14.         //mapper.updateUser(new User(2,"aaa","bbbb"));
  15.         //手动清理缓存
  16.         sqlSession.clearCache();
  17.         System.out.println("============================");
  18.         User user2 = mapper.queryUser(1);
  19.         System.out.println(user2);
  20.         System.out.println(user==user2);
  21.         sqlSession.close();
  22.     }
  23. }

 一级缓存默认是开启的,只在一次SqlSession中有效,也就是拿到连接到关闭连接这个区间段!

一级缓存就是一个Map

 13.4、二级缓存

二级缓存也叫全局缓存,一级缓存作用域太低了,所以诞生了二级缓存

基于namespace级别的缓存,一个名称空间,对应一个二级缓存;

工作机制

一个会话查询一条数据,这个数据就会被放在当前会话的一级缓存中;
如果当前会话关闭了,这个会话对应的一级缓存就没了;但是我们想要的是,会话关闭了,一级缓存中的数据会被保存到二级缓存中;
新的会话查询信息,就可以从二级缓存中获取内容;
不同的mapper查出的数据会放在自己对应的缓存(map)中;

开启全局缓存 

  1. <!--显式的开启全局缓存-->
  2. <setting name="cacheEnabled" value="true"/>

 在要使用二级缓存的Mapper中开启

  1. <!--在当前Mapper.xml中使用二级缓存-->
  2. <cache/>

 测试

问题:我们需要将实体类序列化!否则就会报错 java.io.NotSerializableException: com.rui.pojo.User 

  1. <--也可以自定义参数-->
  2. <cache eviction="FIFO"
  3.        flushInterval="60000"
  4.        size="512"
  5.        readOnly="true"/>
  1. <?xml version="1.0" encoding="UTF-8" ?>
  2. <!DOCTYPE mapper
  3.         PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
  4.         "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.newer.dao.UserMapper">
  6.     <cache eviction="FIFO"
  7.            flushInterval="60000"
  8.            size="512"
  9.            readOnly="true"/>
  10.     <select id="queryUser" resultType="User" useCache="true">
  11.         select * from mybatis.user where id=#{id}
  12.     </select>
  13.     <update id="updateUser" parameterType="User">
  14.         update mybatis.user set name =#{name},pwd=#{pwd} where id=#{id}
  15.     </update>
  16. </mapper>
  1. import com.newer.dao.UserMapper;
  2. import com.newer.pojo.User;
  3. import com.newer.utils.MyBatisUtils;
  4. import org.apache.ibatis.session.SqlSession;
  5. import org.junit.Test;
  6. public class MyTest {
  7.     @Test
  8.     public void test(){
  9.         SqlSession sqlSession = MyBatisUtils.getSqlSession();
  10.         SqlSession sqlSession2= MyBatisUtils.getSqlSession();
  11.         UserMapper mapper = sqlSession.getMapper(UserMapper.class);
  12.         User user = mapper.queryUser(1);
  13.         System.out.println(user);
  14.         sqlSession.close();
  15.         //调用修改方法
  16.         //mapper.updateUser(new User(2,"aaa","bbbb"));
  17.         //手动清理缓存
  18.         //sqlSession.clearCache();
  19.         System.out.println("============================");
  20.         UserMapper mapper2 = sqlSession2.getMapper(UserMapper.class);
  21.         User user2 = mapper2.queryUser(1);
  22.         System.out.println(user2);
  23.         System.out.println(user==user2);
  24.         sqlSession2.close();
  25.     }
  26. }
  • 只要开启了二级缓存,在同一个Mapper下就有效
  • 所有的数据都会先放在一级缓存中;
  • 只有当会话提交,或者关闭的时候,才会提交到二级缓存中!

 

SpringMVC框架

SpringMVC框架介绍:

Spring MVC属于SpringFrameWork的后续产品,已经融合在Spring Web Flow里面。Spring 框架提供了构建 Web 应用程序的全功能 MVC 模块。使用 Spring 可插入的 MVC 架构,从而在使用Spring进行WEB开发时,可以选择使用SpringSpring MVC框架或集成其他MVC开发框架,如Struts1(现在一般不用)Struts 2(一般老项目使用)等等。

---------------------------------上述内容来自百度百科--------------------------------------------------------

  • springmvcspring框架的一个模块,springmvcspring无需通过中间整合层进行整合。
  • springmvc是一个基于mvcweb框架。
  • springmvc 表现层:方便前后端数据的传输
  • Spring MVC 拥有控制器,作用跟Struts类似,接收外部请求,解析参数传给服务层
  • MVC是指,C控制层,M模块层,V显示层这样的设计理念,而SSM框架里面SPRING MVC本身就是MVC框架,作用是帮助(某种意义上也可以 理解为约束)我们要按照MVC这样的设计来开发WEB项目,而另外两个框架spring主要是用作IOC,AOP等其他的一些设计原则,至于mybatis是用来方便操作数据库的,所以他们都在MV里面

 Springmvc架构原理记录
发起请求到前端控制器(DispatcherServlet)
前端控制器请求HandlerMapping查找 Handler,可以根据xml配置、注解进行查找
处理器映射器HandlerMapping向前端控制器返回Handler
前端控制器调用处理器适配器去执行Handler
处理器适配器去执行Handler
Handler
执行完成给适配器返回ModelAndView
处理器适配器向前端控制器返回ModelAndViewModelAndViewspringmvc框架的一个底层对象,包括 Modelview
前端控制器请求视图解析器去进行视图解析,根据逻辑视图名解析成真正的视图(jsp)
视图解析器向前端控制器返回View
前端控制器进行视图渲染,视图渲染将模型数据(ModelAndView对象中)填充到request
前端控制器向用户响应结果

组件:
1、前端控制器DispatcherServlet(不需要程序员开发)
作用接收请求,响应结果,相当于转发器,中央处理器。
有了DispatcherServlet减少了其它组件之间的耦合度。

2、处理器映射器HandlerMapping(不需要程序员开发)
作用:根据请求的url查找Handler

3、处理器适配器HandlerAdapter
作用:按照特定规则(HandlerAdapter要求的规则)去执行Handler

4、处理器Handler(需要程序员开发)
注意:编写Handler时按照HandlerAdapter的要求去做,这样适配器才可以去正确执行Handler

5、视图解析器View resolver(不需要程序员开发)
作用:进行视图解析,根据逻辑视图名解析成真正的视图(view

6、视图View(需要程序员开发jsp)
View是一个接口,实现类支持不同的View类型(jspfreemarkerpdf…

SpringMVC常用注解
@Controller
   负责注册一个bean spring 上下文中

@RequestMapping
   注解为控制器指定可以处理哪些 URL 请求

@RequestBody
   该注解用于读取Request请求的body部分数据,使用系统默认配置的HttpMessageConverter进行解析,然后把相应的数据绑定到要返回的对象上 ,再把HttpMessageConverter返回的对象数据绑定到 controller中方法的参数上

@ResponseBody
   该注解用于将Controller的方法返回的对象,通过适当的HttpMessageConverter转换为指定格式后,写入到Response对象的body数据区

@ModelAttribute    
  在方法定义上使用 @ModelAttribute 注解:Spring MVC 在调用目标处理方法前,会先逐个调用在方法级上标注了@ModelAttribute 的方法
  
  在方法的入参前使用 @ModelAttribute 注解:可以从隐含对象中获取隐含的模型数据中获取对象,再将请求参数绑定到对象中,再传入入参将方法入参对象添加到模型中

@RequestParam 
  在处理方法入参处使用 @RequestParam 可以把请求参 数传递给请求方法

@PathVariable
  绑定 URL 占位符到入参
  
@ExceptionHandler
  注解到方法上,出现异常时会执行该方法
  
@ControllerAdvice
  使一个Contoller成为全局的异常处理类,类中用@ExceptionHandler方法注解的方法可以处理所有Controller发生的异常

支持自动匹配参数与自动装

SpringMVC 响应数据处理

ModelAndView 应用

我们有一业务,现在需要将响应数据封装到ModelAndView对象,然后响应到客户端,如何实现呢?

第一步:定义ModelViewController以及方法,代码如下:

  1. package com.cy.pj.module.controller;
  2. import org.springframework.stereotype.Controller;
  3. import org.springframework.ui.Model;
  4. import org.springframework.web.bind.annotation.RequestMapping;
  5. @Controller
  6. public class ModelViewController {
  7.     @RequestMapping("/doModelAndView")
  8.     public String doModelAndView(Model model) {
  9.         model.addAttribute("username", "jason");
  10.         model.addAttribute("state", true);
  11.         return "view";
  12.     }
  13. }

第二步:templates相关目录下定义view.html,并在页面中添加呈现数据的代码,例如:

  1. <div>
  2.     <ul>
  3.         <li>username:[[${username}]]</li>
  4.         <li>state:[[${state}]]</li>
  5.     </ul>
  6. </div>

第三步:启动项目进行访问测试,并检测输出结果,例如:

第四步:运行过程中的结果分析 

JSON数据响应

我们有一业务不需要页面,只需要将响应数据转换为json,然后响应到客户端,如何实现呢?
第一步:定义ResponseResult对象用于封装响应数据,例如:

  1. package com.cy.pj.module.pojo;
  2. public class ResponseResult {
  3.     private Integer code;
  4.     private String message;
  5.     public Integer getCode() {
  6.         return code;
  7.     }
  8.     public void setCode(Integer code) {
  9.         this.code = code;
  10.     }
  11.     public String getMessage() {
  12.         return message;
  13.     }
  14.     public void setMessage(String message) {
  15.         this.message = message;
  16.     }
  17. }

 第二步:定义JsonObjectController以及方法,代码如下:

  1. package com.cy.pj.module.controller;
  2. @RestController
  3. public class JsonObjectController {
  4.     @RequestMapping("/doConvertResponseToJson")
  5.     public ResponseResult doConvertResponseToJson(){
  6.         ResponseResult rs=new ResponseResult();
  7.         rs.setCode(200);
  8.         rs.setMessage("OK");
  9.         return rs;
  10.     }
  11.     
  12.     @RequestMapping("/doConvertMapToJson")
  13.     public Map<String,Object> doConvertMapToJson(){
  14.         Map<String,Object> map=new HashMap<>();
  15.         map.put("username","刘德华");
  16.         map.put("state",true);
  17.         return map;
  18.     }
  19.     
  20.     @RequestMapping("/doPrintJSON")
  21.     public void doPrint(HttpServletResponse response)throws Exception{
  22.         Map<String,Object> map=new HashMap<>();
  23.         map.put("username","刘德华");
  24.         map.put("state",true);
  25.         //map中的数据转换为json格式字符串
  26.         ObjectMapper om=new ObjectMapper();
  27.         String jsonStr=om.writeValueAsString(map);
  28.         System.out.println("jsonStr="+jsonStr);
  29.         //将字符串响应到客户端
  30.         //设置响应数据的编码
  31.         response.setCharacterEncoding("utf-8");
  32.         //告诉客户端,要向它响应的数据类型为text/html,编码为utf-8.请以这种编码进行数据呈现
  33.         response.setContentType("text/html;charset=utf-8");
  34.         PrintWriter pw=response.getWriter();
  35.         pw.println(jsonStr);
  36.     }
  37.     
  38. }

第三步:启动服务器分别进行访问测试,代码如下:

第四步:启动或访问过程中的问题分析 

 

 SpingMVC 请求参数数据处理

我们在执行业务的过程中通常会将一些请求参数传递到服务端,服务端如何获取参数并注入给我们的方法参数的呢?

准备工作

定义一个controller对象,用户处理客户端请求,例如:

  1. package com.cy.pj.module.controller;
  2. import com.cy.pj.module.pojo.RequestParameter;
  3. import org.springframework.web.bind.annotation.*;
  4. import java.util.Map;
  5. @RestController
  6. public class ParamObjectController {}

直接量方式

ParamObjectController中添加方法,基于直接量方式接受客户端请求参数,例如:

  1. @GetMapping("/doParam01")
  2. public String doMethodParam(String name){
  3.  return "request params "+name;
  4. }

访问时,可以这样传参,例如:

http://localhost/doParam01?name=beijing

POJO对象方式

定义pojo对象,用于接受客户端请求参数,例如:

  1. package com.cy.pj.module.pojo;
  2. public class RequestParameter {
  3.     private String name;
  4.     //......
  5.  public String getName() {
  6.         return name;
  7.     }
  8.     public void setName(String name) {
  9.         this.name = name;
  10.     }
  11.     @Override
  12.  public String toString() {
  13.         return "RequestParam{" +
  14.                 "name='" + name + ''' +
  15.                 '}';
  16.     }
  17. }

定义Controller方法,方法中使用pojo对象接收方法参数,例如:

  1. @RequestMapping("/doParam02")
  2. public String doMethodParam(RequestParameter param){//pojo对象接收请求参数,pojo对象中需提供与参数名相匹配的set方法
  3.  return "request params "+param.toString();
  4. }

启动服务进行访问测试,可以这样传参,例如:

http://localhost/doParam02?name=beijing

Map对象方式

有时候我们不想使用pojo对象接收请求参数,我们可以使用map对象来接收,又该如何实现呢?

定义Controller方法,基于map对象接收请求参数,例如:

  1. @GetMapping("/doParam03")
  2. public String doMethodParam(@RequestParam Map<String,Object> param){
  3.  return "request params "+param.toString();
  4. }

其中,map接收请求参数,必须使用@RequestParam对参数进行描述.

启动服务进行访问测试,可以这样传参,例如:

http://localhost/doParam03?name=beijing

总结(Summary)

上述内容对springboot工程下spring mvc技术的应用做了一个入门实现,并结合实际项目中的业务应用,讲解了MVC中请求数据的获取和响应数据处理的一个基本过程.

SpringSecurity(安全管理框架)

        对于一个Web项目来说,最重要的不是功能酷不酷炫,而是这个项目安不安全。做过项目的人都知道,一个项目在上线前一定会经过安全漏扫,只有通过安全漏扫后这个项目才能正式上线。
Spring Security是一个功能强大且高度可定制的身份验证和访问控制框架,类似的安全框架还有Shiro
Spring Security主要做两个事情,认证、授权

一般我们常做的四个功能认证 、授权、注销、记住密码

1&2. 认证与授权

在上面的这个页面中有两个问题,第一未做登陆认证的处理,第二三个level页面现在都能被所有人访问,未做授权。而上面的这两个问题都可以通过SpringSecurity来解决。

实现SpringSecurity只需要引入spring-boot-starter-security依赖,进行少量的配置,就可以实现强大的安全管理功能。

在正式接触前我们需要记住几个类:
1.WebSecurityConfigurerAdapter :自定义Security策略
2.AuthenticationManagerBuilder:自定义认证策略
3.@EnableWebSecurity :开启WebSecurity模式

我们新建一个config包,创建一个配置类SecurityConfig,继承WebSecurityConfigurerAdapter接口
 

  1. @EnableWebSecurity//开启WebSecurity模式
  2. public class SecurityConfig extends WebSecurityConfigurerAdapter /*继承自定义security策略*/{
  3.     //授权
  4.     @Override
  5.     protected void configure(HttpSecurity http) throws Exception {
  6.         //首页所有人都能访问,level页面只有有权限的人才能访问
  7.         http.authorizeRequests()
  8.                 .antMatchers("/").permitAll()
  9.                 .antMatchers("/level1/**").hasRole("vip1")
  10.                 .antMatchers("/level2/**").hasRole("vip2")
  11.                 .antMatchers("/level3/**").hasRole("vip3");
  12.         //没有权限默认跳到登陆页,默认会重定向到/login
  13.         http.formLogin();
  14.     }
  15.     //认证
  16.     @Override
  17.     protected void configure(AuthenticationManagerBuilder auth/*自定义认证策略*/) throws Exception {
  18.         auth.inMemoryAuthentication()
  19.                 .passwordEncoder(new BCryptPasswordEncoder())
  20.                 .withUser("root").password(new BCryptPasswordEncoder().encode("123456")).roles("vip1","vip2","vip3")
  21.                 .and()
  22.                 .withUser("admin").password(new BCryptPasswordEncoder().encode("123456")).roles("vip1");
  23.     }
  24. }

通过上面的这段代码,实现了授权和认证的功能,其中认证方法在内存中存入的用户信息。正常情况下用户的数据是从数据库中获取,授权功能给不同的页面设置不同的角色。

我们再次打开首页http://localhost:8080/,所有人都有权限看到,当我们点击里面的level1 2 3里的链接时,自动跳转到了http://localhost:8080/login
 

你会发现这个登陆页面明明不是自己写的,怎么就出现了?其实这就是SpringSecurity所提供的,http.formLogin();这段代码做了对登陆页的处理,没有权限默认跳到登陆页,默认会重定向到/login

当我们用不同权限的账号登陆时,就能点击不同权限的链接,如果点击权限外的链接时,会报403错误

403错误,可以简单理解为没有权限访问此网站,该状态表示服务器理解了此次请求,但是拒绝执行此次任务。

导致403错误的主要原因

  1. 你的IP被列入黑名单。
  2. 你在一定时间内过多地访问此网站(一般是用采集程序),被防火墙拒绝访问了。
  3. 网站域名解析到了空间,但空间未绑定此域名。
  4. 你的网页脚本文件在当前目录下没有执行权限。
  5. 在不允许写/创建文件的目录中执行了创建/写文件操作。
  6. http方式访问需要ssl连接的网址。
  7. 浏览器不支持SSL 128时访问SSL 128的连接。
  8. 在身份验证的过程中输入了错误的密码。
  9. DNS解析错误,手动更改DNS服务器地址。
  10. 连接的用户过多,可以过后再试。
  11. 服务器繁忙,同一IP地址发送请求过多,遭到服务器智能屏蔽。

 3. 注销的操作

既然有认证和授权,那么一定免不了注销的功能,SpringSecurity实现注销很简单,你只需要在SecurityConfig类的授权代码里增加一条代码。

  1. //登出
  2. http.logout();

然后在前端index.html代码中增加一个注销的标签

<a href="/logout">注销</a>

点击首页的注销按钮后就会跳转到SpringSecurity的注销提示界面

 点击logout按钮后自动退出到登陆页面,如果你希望注销后还是回到首页的话,可以将注销代码修改成下面的方式:

http.logout().logoutSuccessUrl("/");

4. 记住密码功能

一般情况下登陆页面肯定会有一个记住密码的功能,这个功能其实就是在本地生成一个cookie,用原生的java实现虽然不难,但是也需要一定的代码。而使用SpringSecurity只需要一行:

http.rememberMe();

再次进入登陆页面后,就可以看到增加了一行记住密码的选项

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值