4.1 修饰符

4.1.1 访问权限修饰符

修饰符 同一个类中 同一个包中子类无关类 不同包的子类 不同包的无关类
private
默认
protected
public

4.1.2 状态修饰符

4.1.2.1 final(最终态)

  • final关键字是最终的意思,可以修饰成员方法,成员变量,类。
  • 特点:
    • 修饰方法:表明方法是最终方法,不能被重写。
    • 修饰变量:表明变量是常量,不能再次被赋值。
    • 修饰类:表明该类是最终类,不能被继承。
  • 修饰局部变量:
    • 变量是基本类型:final修饰指的是基本类型的数据值不能发生改变
    • 变量是引用类型:final修饰指的是引用类型的地址值不能发生改变,但是地址里面的内容是可以发生改变的。
  • 修饰实例变量:

    • final修饰实例变量的时候,系统不管赋默认值,要求手动赋值。
    • final修饰的实例变量一般和static联合使用,称为常量。

      4.1.2.2 static(静态)

  • 可以修饰成员方法,成员变量

  • 特点:
    • 被类的所有对象共享:这也是判断是否使用静态关键字的条件。
    • 可以通过类名调用:也可以通过对象名调用,推荐使用类名调用。
  • 访问特点:
    • 非静态的成员方法:
      • 能访问静态的成员变量
      • 能访问非静态的成员变量
      • 能访问静态的成员方法
      • 能访问非静态的成员方法
    • 静态成员方法:只能访问静态的成员方法和变量

      4.2 方法

      4.2.1 成员变量和局部变量

      | | 成员变量 | 局部变量 | | —- | —- | —- | | 类中位置不同 | 类中方法外 | 方法内或者方法声明上 | | 内存中位置不同 | 堆内存 | 栈内存 | | 生命周期不同 | 随着对象的存在而存在,对象消失而消失 | 随着方法调用而存在,随着方法的调用完毕而消失 | | 初始化值不同 | 有默认的初始化值 | 没有默认的初始化值,必须先定义,赋值,才能使用 |

4.2.2 方法参数的传递

  • 对于基本数据类型的参数,形参的改变,不影响实际参数的值

    1. public class Demo {
    2. public static void change(int number){
    3. int num = 200 ;
    4. }
    5. public static void main(String[] args) {
    6. int num = 100 ;
    7. System.out.println(num); //100
    8. change(num);
    9. System.out.println(num); //100
    10. }
    11. }
  • 对于引用类型的参数,形式参数的改变,影响实际参数的值

    1. public class Demo {
    2. public static void change(int[] arr){
    3. arr[1] = 100 ;
    4. }
    5. public static void main(String[] args) {
    6. int[] arr = {1,2,3} ;
    7. System.out.println(arr[1]); //2
    8. change(arr);
    9. System.out.println(arr[1]); //100
    10. }
    11. }

    4.2.3 方法重载的特点

  • 重载仅对应方法的定义,与方法的调用无关,调用方式参照标准格式;

  • 重载仅针对同一个类中方法的名称与参数进行识别,与返回值无关;
  • 需要多个方法在同一个类中,具有相同的方法名,参数不同或者类型不同或者数量不同。

    4.2.4 构造方法概述

  • 构造方法是一种特殊的方法,作用是创建对象。

    1. 格式:
    2. public class 类名{
    3. 修饰符 类名(参数){
    4. }
    5. }
  • 注意事项:

    • 构造方法的创建:
      • 如果没有定义构造方法,系统将给出一个默认的无参构造方法
      • 如果定义了构造方法,系统将不再提供默认的构造方法
    • 构造方法的重载:如果自定义带参构造方法,还要使用无参构造方法,就必须在写一个无参构造方法
    • 建议无论是否使用,都手工书写无参构造方法。

      4.2.5 方法重写

  • @Override 可以帮助检查重写方法的声明的正确性

  • 注意事项:

    • 私有方法不能被重写(父类私有成员子类是不能继承的)
    • 子类方法访问权限不能更低(public>默认>私有)

      4.2.6 方法引用

      4.2.6.1 方法引用符

  • :: 该符号是引用运算符,而它所在的表达式被称为方法引用

  • Lambda表达式:usePrintable(s->System.out.println(s));分析:拿到参数s之后通过Lambda表达式,传递给 System.out.println方法去处理。
  • 方法引用:usePrintable(System.out::println);分析:直接使用System.out中的println方法来取代Lambda,代码更加的简洁。 ```java public class Demo { public static void main(String[] args) {

    1. usePrintable((s)-> System.out.println(s));
    2. usePrintable(System.out::println);

    } public static void usePrintable(Printable p){

    1. p.printString("666666666666");

    } }


public interface Printable { public void printString(String s); }

  1. - 推导与省略
  2. - 如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定重载形式,它们都将被自动推导。
  3. - 如果使用方法引用,也是同样可以根据上下文进行推导
  4. - 方法引用是Lambda的孪生兄弟
  5. <a name="UoApT"></a>
  6. #### 4.2.6.2 引用类方法
  7. - 其实就是引用类的静态方法
  8. - 格式:类名::静态方法
  9. ```java
  10. public class Demo {
  11. public static void main(String[] args) {
  12. useConvert((s)-> {
  13. return Integer.parseInt(s);
  14. });
  15. useConvert(Integer::parseInt);
  16. }
  17. public static void useConvert(Converter p){
  18. int i = p.convert("66666666");
  19. System.out.println(i);
  20. }
  21. }
  22. ***********************************************************************************
  23. public interface Converter {
  24. public int convert(String s);
  25. }

4.2.6.3 引用对象的实例方法

  • 引用对象的实例方法就是引用类中的成员方法
  • 格式:对象::成员方法 ```java public class Demo{ public static void main(String[] args) {
    1. usePrint(s-> System.out.println(s.toUpperCase()));
    2. PrintString ps = new PrintString();
    3. usePrint(ps::printUpper);
    } public static void usePrint(Printer p){
    1. p.printUpperCase("HelloWorld");
    } }

public class PrintString { public void printUpper(String s){ String result = s.toUpperCase() ; System.out.println(result); } }


public interface Printer { void printUpperCase(String s); }

  1. <a name="i9RRM"></a>
  2. #### 4.2.6.4 引用构造器
  3. - 就是引用构造方法
  4. - 格式:类名::new
  5. ```java
  6. public class Demo{
  7. public static void main(String[] args) {
  8. useStudentBuilder((name,age)->new Student(name,age));
  9. useStudentBuilder(Student::new);
  10. }
  11. public static void useStudentBuilder(StudentBuilder sb){
  12. Student s = sb.build("666",66);
  13. System.out.println(s.getName()+","+s.getAge());
  14. }
  15. }
  16. ************************************************************************************
  17. public class Student {
  18. private String name ;
  19. private int age ;
  20. public Student(String name, int age) {
  21. this.name = name;
  22. this.age = age;
  23. }
  24. public Student() { }
  25. public int getAge() { return age; }
  26. public void setAge(int age) {this.age = age;}
  27. public String getName() { return name;}
  28. public void setName(String name) {this.name = name;}
  29. }
  30. ************************************************************************************
  31. public interface StudentBuilder {
  32. Student build(String name,int age);
  33. }

4.2.6.5 引用类的实例方法

  • 就是引用类中的成员方法
  • 格式:类名::成员方法 ```java public class Demo{ public static void main(String[] args) {
    1. useMyString((s,x,y) ->s.substring(x,y));
    2. useMyString(String::substring);
    } public static void useMyString(MyString ms){
    1. String s = ms.mySubString("HelloWorld",2,5);
    2. System.out.println(s);
    } }

public interface MyString { String mySubString(String s,int x,int y); }

  1. <a name="XlkhF"></a>
  2. ## 4.3 封装
  3. <a name="TrzMD"></a>
  4. ### 4.3.1 this关键字
  5. - this修饰的变量用于指代成员变量
  6. - 方法的形参如果与成员变量同名,不带this修饰的变量指的是形参,而不是成员变量
  7. - 方法的形参没有与成员变量同名,不带this修饰变量指的是成员变量
  8. - 解决局部变量隐藏成员变量的时候使用this
  9. - this代表所在类的对象引用,方法被那个对象调用,this就代表那个对象。
  10. <a name="Xbxa6"></a>
  11. ## 4.4 继承
  12. <a name="Lkaha"></a>
  13. ### 4.4.1 this和super
  14. - this 代表本类对象的引用:
  15. - this.成员变量:访问本类成员变量
  16. - this(...):访问本类构造方法
  17. - this.成员方法(...):访问本类成员方法
  18. - super 代表父类存储空间的标识(可以理解为父类对象的引用)
  19. - super.成员变量:访问父类成员变量
  20. - super(...):访问父类构造方法
  21. - super.成员方法(...):访问父类成员方法
  22. <a name="KvmGm"></a>
  23. ### 4.4.2 构造方法访问特点
  24. - 子类中的所有构造方法都会默认访问父类中无参的构造方法
  25. - 因为子类会继承父类中的数据,可以还会使用父类的数据,所以,子类初始化之前,一定要先完成父类数据的初始化。
  26. - 每一个子类构造方法的第一条语句默认都是super()
  27. - 如果父类汇总没有无参构造方法,只有带参怎么办?(默认子类会报错)
  28. - 通过使用super关键字去显示调用父类的带参构造方法
  29. - 在父类中自己提供一个无参构造方法(推荐)
  30. - 注意事项:
  31. - java中类只支持单继承,不支持多继承
  32. - java中支持多层继承
  33. <a name="aTVZv"></a>
  34. ## 4.5 多态
  35. - 多态的前提和体现:
  36. - 有继承/实现关系
  37. - 有方法重写
  38. - 有父类引用指向子类对象
  39. - 访问特点:
  40. - 成员变量:编译看左边,执行看右边
  41. - 成员方法:编译看左边,执行看右边
  42. - 不一样的原因是成员方法可有写,而成员变量没有
  43. - 利弊:
  44. - 利:提高了程序的延展性。具体实现:定义方法的时候,使用父类型作为参数,将来咋使用的时候,使用具体的子类型参与操作。
  45. - 弊:不能使用子类的特有功能
  46. - 多态转型,转型后可以解决弊端:(Animal为父类,Cat为子类)
  47. - Animal a = new Cat() ; //向上转型
  48. - Cat c = (Cat)a ; //向下转型
  49. <a name="MsTxS"></a>
  50. ## 4.6 内部类
  51. - 概述:内部类就是在一个类中定义一个类
  52. - 访问特点:
  53. - 内部类可以直接访问外部类的成员,包括私有
  54. - 外部类要访问内部类的成员,必须创建对象
  55. - 根据定义的位置不同,可以分为两种形式:
  56. - 在类的成员位置:**成员内部类**
  57. - 外界创建成员内部类对象的格式: 外部类名.内部类名 对象名 = 外部类对象.内部类对象
  58. ```java
  59. package com.bai;
  60. public class Outer {
  61. public class Inner{
  62. private int age = 66 ;
  63. public int getAge() {
  64. return age;
  65. }
  66. }
  67. }
  68. package com.bai;
  69. public class Demo {
  70. public static void main(String[] args) {
  71. Outer.Inner inner = new Outer().new Inner() ;
  72. System.out.println(inner.getAge()); //66
  73. }
  74. }
  • 在类的局部位置:局部内部类
    • 局部内部类是在方法中定义的类,所以外界无法直接使用,需要在方法内部创建对象并使用。该类可以直接访问外部类的成员,也可以访问方法内的局部变量。
    • 匿名内部类:
      • 前提:存在一个类或者接口,这里的类可以是具体类也可以是抽象类
      • 格式:new 类名或者接口名(){ 重写方法 ; }
      • 本质是一个继承了该类或者实现了该接口的子类匿名对象 ```java package com.bai;

public interface Jumpping { void jump(); }

  1. ```java
  2. package com.bai;
  3. public class Outer {
  4. public void method(){
  5. Jumpping j = new Jumpping(){ //接口
  6. @Override
  7. public void jump(){
  8. System.out.println("跳高");
  9. }
  10. };
  11. j.jump();
  12. }
  13. }
  1. package com.bai;
  2. public class Demo {
  3. public static void main(String[] args) {
  4. Outer outer = new Outer() ;
  5. outer.method();
  6. }
  7. }
  8. 运行结果:
  9. 跳高
  10. Process finished with exit code 0

4.7 抽象类

  • 概述:
    • 在java中,一个没有方法体的方法应该定义为抽象方法,而类中如果有抽象方法,该类必须定义为抽象类。
    • 抽象类是引用数据类型。
  • 特点:
    • 抽象类和抽象方法必须使用 abstract 关键字修饰
      • public abstract class 类名(){}
      • public abstract void eat();
    • 抽象类中不一定有抽象方法,有抽象方法的类一定是抽象类
    • 抽象类不嫩实例化
      • 如果需要实例化就要参照多态的方式,通过子类对象实例化,这叫抽象类多态
    • 抽象类子类
      • 要么重写抽象类中所有抽象方法
      • 要么是抽象类
  • 成员特点:

    • 成员变量:可以是变量,也可以是常量
    • 构造方法:有构造方法,但是不能实例化;构造方法的作用是用于子类访问父类数据的初始化
    • 成员方法:可以有抽象方法,限定子类必须完成某些动作;也可以有非抽象方法,提高代码复用性;

      4.8 接口

      4.8.1 特点

  • 接口用关键字 interface 修饰,例:public interface 接口名{}

  • 类实现接口用 implements 表示,例:public class 类名 implements 接口名{}
  • 接口不能实例化:
    • 接口实例化参照多态方式,通过实现类对象实例化,这叫接口多态
    • 多态形式:具体类多条,抽象类多态,接口多态
    • 多态的前提:有继承或者实现关系;有方法重写;有父类/父接口引用指向(子/实现)类对象。
  • 接口实现类:

    • 要么重写接口中所有抽象方法
    • 要么是抽象类

      4.8.2 成员特点

  • 成员变量:

    • 只能是常量
    • 默认修饰符:public static final
  • 构造方法:
    • 接口没有构造方法,因为接口主要就是对行为进行抽象的,是没有具体存在的
    • 一个类如果没有父类,默认继承自Object类
  • 成员方法:

    • 只能是抽象方法
    • 默认修饰符:public abstract

      4.8.3 类和接口的关系

  • 类和类的关系:继承关系,只能单继承,但是可以多层继承。

  • 类和接口的关系:实现关系,可以单实现,也可以多实现,还可以在继承一个类的同时实现多个接口。
  • 接口和接口的关系:继承关系,可以单继承,也可以多继承。

    4.8.4 抽象类和接口的区别

  • 成员区别:

    • 接口:常量,抽象方法
    • 抽象类:变量,常量;有构造方法;也有非抽象方法。
  • 关系区别:与类和接口的关系一样
  • 设计理念区别:

    • 抽象类:对类抽象,包括属性、行为
    • 接口:对行为抽象,主要是行为

      4.8.5 接口组成更新

      接口的组成:
  • 常量:public static final

  • 抽象方法:public abstract
  • 默认方法(Java8)
    • 定义格式:public default 返回值类型 方法名(参数列表){}
    • 注意事项:默认方法不是抽象方法,所以不强制被重写,但是可以被重写;public可以省略,default不能省略。
  • 静态方法(Java8)
    • 定义格式:public static 返回值类型 方法名(参数列表)
    • 注意事项:静态方法只能通过接口名调用,不能通过实现类名或者对象调用。public可以省略,static不能省略。
  • 私有方法(Java9)

    • 定义格式
      • 格式一:private 返回值类型 方法名(参数列表){}
      • 格式二:private static 返回值类型 方法名(参数列表){}
    • 注意事项:默认方法可以调用私有的静态方法和非静态方法;静态方法只能调用私有静态方法;

      4.8.6 函数式接口

  • 函数式接口:有且仅有一个抽象方法接口

  • Java中的函数编程体现就是Lambda表达式,所以函数式接口就是可以适用于Lambda使用接口,只有确保接口中有且仅有一个抽象方法,Java中的Lambda才能顺利进行。
  • @functionalInterface
    • 检测一个接口是不是函数式接口
    • 放在接口上方,如果是函数式接口,编译通过;如果不是,编译失败。
    • 我们自己定义函数式接口时,这个东西是可选的,只要保证满足函数式接口定义的条件,也照样是函数式接口。建议加上这个注解。
      1. /* 函数式接口作为方法的参数 */
      2. public class Demo {
      3. public static void main(String[] args) {
      4. startThread(new Runnable() {
      5. @Override
      6. public void run() {
      7. System.out.println(Thread.currentThread().getName()+","+"线程启动了1");
      8. }
      9. });
      10. startThread(()-> System.out.println(Thread.currentThread().getName()+","+"线程启动了2"));
      11. }
      12. public static void startThread(Runnable r){
      13. new Thread(r).start();
      14. }
      15. }
      ```java / 函数式接口作为方法的返回值 / import java.util.ArrayList; import java.util.Collections; import java.util.Comparator;

public class Demo { public static void main(String[] args) { ArrayList arr = new ArrayList<>() ; arr.add(“444”) ; arr.add(“1111”) ; arr.add(“33333”) ; arr.add(“22”) ; System.out.println(“排序前:”+”,”+arr); //Collections.sort(arr); //自然排序 Collections.sort(arr,getComparator());//按长度排序 System.out.println(“排序后:”+”,”+arr); } private static Comparator getComparator() { //方式一 /Comparator comp = new Comparator() { @Override public int compare(String s1, String s2) { return s1.length()-s2.length(); } }; return comp ;/ //方式二 /return new Comparator() { @Override public int compare(String s1, String s2) { return s1.length() - s2.length(); } };/ //方式三 return (s1,s2)->s1.length()-s2.length() ; } }

  1. **常用函数式接口**
  2. - **_Supplier_**
  3. - Supplier<T>:包含一个无参的方法
  4. - **_T get()_**:获得接口
  5. - 该方法不需要参数,他会按照实现逻辑(由Lambda表达式实现)返回一个数据
  6. - Supplier<T>接口也称为生产型接口,如果我们指定了接口的泛型是什么类型,那么接口中的get方法就会生产什么类型的数据供我们使用。
  7. ```java
  8. import java.util.function.Supplier;
  9. public class Demo {
  10. public static void main(String[] args) {
  11. String s = getString(()->"666") ;
  12. System.out.println(s);
  13. Integer i = getInteger(()->666) ;
  14. System.out.println(i);
  15. }
  16. private static String getString(Supplier<String > sup){
  17. return sup.get() ;
  18. }
  19. private static Integer getInteger(Supplier<Integer> sup ){
  20. return sup.get() ;
  21. }
  22. }
  1. /* 返回数组中最大值 */
  2. import java.util.Arrays;
  3. import java.util.function.Supplier;
  4. public class Demo {
  5. public static void main(String[] args) {
  6. int[] arr = {2,6,1,38,78,12} ;
  7. int i = getMax(()->{
  8. Arrays.sort(arr);
  9. return arr[arr.length-1] ;
  10. }) ;
  11. System.out.println(i);
  12. }
  13. private static int getMax(Supplier<Integer> sup){
  14. return sup.get() ;
  15. }
  16. }
  • Consumer
    • Consumer:包含两个方法
    • void accept(T t):对给定的参数执行此操作
    • default Consumer andThen(Consumber after):返回一个组合的Consumer,依次执行此操作,然后执行after操作
    • Consumer接口也被称为消费型接口,他消费的数据的数据类型由泛型指定 ```java import java.util.function.Consumer;

public class Demo { public static void main(String[] args) { operatorString(“123”,(s)-> System.out.println(s)); operatorString(“123”,s -> System.out.println(new StringBuilder(s).reverse().toString()));

  1. operatorString("456",s-> System.out.println(s),
  2. s -> System.out.println(new StringBuilder(s).reverse().toString()));
  3. }
  4. private static void operatorString(String name, Consumer<String> con1,Consumer<String> con2){
  5. //con1.accept(name);
  6. //con2.accept(name);
  7. con1.andThen(con2).accept(name); //和上面两行执行结果一样
  8. }
  9. private static void operatorString(String name, Consumer<String> con1){
  10. con1.accept(name);
  11. }

}

  1. ```java
  2. /* 打印信息 */
  3. import java.util.function.Consumer;
  4. public class Demo {
  5. public static void main(String[] args) {
  6. String[] strArray = {"小黑,30","小黄,35","小绿,33"} ;
  7. getStudent(strArray,s-> System.out.print("姓名-->"+s.split(",")[0]),
  8. s -> System.out.println(" 年龄-->"+s.split(",")[1]));
  9. }
  10. public static void getStudent(String[] name,Consumer<String> c1,Consumer<String> c2){
  11. for(String str:name){
  12. c1.andThen(c2).accept(str);
  13. }
  14. }
  15. }
  16. 运行结果:
  17. 姓名-->小黑 年龄-->30
  18. 姓名-->小黄 年龄-->35
  19. 姓名-->小绿 年龄-->33
  20. Process finished with exit code 0
  • Predicate
    • Predicate:常用四个方法
    • boolean test(T t) 对给定的参数进行判断(判断逻辑由Lambda表达式实现),返回一个布尔值
    • default Predicate negate() 返回一个逻辑的否定,对应逻辑非
    • default Predicate and(Predicate other) 返回一个组合判断,对应短路与
    • default Predicate or(Predicate other) 返回一个组合判断,对应短路或
    • Predicate接口通常用于判断参数是否满足指定的条件 ```java import java.util.function.Predicate;

public class Demo { public static void main(String[] args) { System.out.println(checkString(“helloWorld”, s -> s.length()>8, s -> s.length()<15)); } public static boolean checkString(String s, Predicate p1,Predicate p2){ / boolean b1 = p1.test(s) ; boolean b2 = p2.test(s) ; return b1&b2 ;/ return p1.and(p2).test(s) ; } private static boolean checkString(String s,Predicate p1){ return p1.test(s) ; } }

  1. ```java
  2. /* 输出名字三个字,大于33岁的 */
  3. import java.util.ArrayList;
  4. import java.util.function.Predicate;
  5. public class Demo {
  6. public static void main(String[] args) {
  7. String[] strArray = {"小黑黑,30","小黄,34","小绿绿,35","小白,31","小蓝蓝,33"};
  8. ArrayList<String> arrayList = getMessage(strArray,
  9. s->s.split(",")[0].length()>2,
  10. s -> Integer.parseInt(s.split(",")[1])>33);
  11. System.out.println(arrayList);
  12. }
  13. private static ArrayList<String> getMessage(String[] arr,Predicate<String> p1,Predicate<String> p2){
  14. ArrayList<String> arrayList = new ArrayList<>() ;
  15. for(String str:arr){
  16. if(p1.and(p2).test(str))
  17. arrayList.add(str);
  18. }
  19. return arrayList ;
  20. }
  21. }
  22. 运行结果:
  23. [小绿绿,35]
  24. Process finished with exit code 0
  • Function
    • Function:常用两个方法
    • R apply(T,t) 将此函数应用于给定的参数
    • default Function andThen(Function after) 返回一个组合函数,首先将该函数应用于输入,然后将after函数应用于结果
    • Function接口通常用于对参数进行处理,转换(处理逻辑由Lambda表达式实现),然后返回一个新的值 ```java import java.util.function.Function;

public class Demo { public static void main(String[] args) { convert(“66”,s->Integer.parseInt(s)); convert(100,s->String.valueOf(s+66)); convert(“33”,s->Integer.parseInt(s),s->String.valueOf(s+100)); } //定一个方法,把一个字符串转换int类型,在控制台输出 private static void convert(String s, Function fun){ int i = fun.apply(s); System.out.println(i); } //定义一个方法,把一个int类型的数据上加一个整数后,转为字符串在控制台输出 private static void convert(int i,Function fun){ String s = fun.apply(i) ; System.out.println(s); } //定义一个方法,把一个字符串转换int类型,把int类型的数据加上一个整数之后,转为字符串控制台输出 private static void convert(String s, Function fun1, Function fun2){ / int i = fun1.apply(s); String ss = fun2.apply(i) ; System.out.println(ss);/ String ss = fun1.andThen(fun2).apply(s) ; System.out.println(ss); } }

  1. ```java
  2. /* 按照指定要求操作数据 */
  3. import java.util.function.Function;
  4. public class Demo {
  5. public static void main(String[] args) {
  6. String ss = "小黑,30" ;
  7. //Integer.parseInt(s) 可以通过方法引用修改为 Integer::parseInt
  8. modifyAge(ss,s->s.split(",")[1],s->Integer.parseInt(s),s->s+70);
  9. }
  10. private static void modifyAge(String s,Function<String,String> f1,Function<String,Integer> f2,Function<Integer,Integer> f3){
  11. int i = f1.andThen(f2).andThen(f3).apply(s) ;
  12. System.out.println(i); //100
  13. }
  14. }