4.1 修饰符
4.1.1 访问权限修饰符
| 修饰符 | 同一个类中 | 同一个包中子类无关类 | 不同包的子类 | 不同包的无关类 |
|---|---|---|---|---|
| private | √ | |||
| 默认 | √ | √ | ||
| protected | √ | √ | √ | |
| public | √ | √ | √ | √ |
4.1.2 状态修饰符
4.1.2.1 final(最终态)
- final关键字是最终的意思,可以修饰成员方法,成员变量,类。
- 特点:
- 修饰方法:表明方法是最终方法,不能被重写。
- 修饰变量:表明变量是常量,不能再次被赋值。
- 修饰类:表明该类是最终类,不能被继承。
- 修饰局部变量:
- 变量是基本类型:final修饰指的是基本类型的数据值不能发生改变
- 变量是引用类型:final修饰指的是引用类型的地址值不能发生改变,但是地址里面的内容是可以发生改变的。
修饰实例变量:
可以修饰成员方法,成员变量
- 特点:
- 被类的所有对象共享:这也是判断是否使用静态关键字的条件。
- 可以通过类名调用:也可以通过对象名调用,推荐使用类名调用。
- 访问特点:
4.2.2 方法参数的传递
对于基本数据类型的参数,形参的改变,不影响实际参数的值
public class Demo {public static void change(int number){int num = 200 ;}public static void main(String[] args) {int num = 100 ;System.out.println(num); //100change(num);System.out.println(num); //100}}
对于引用类型的参数,形式参数的改变,影响实际参数的值
public class Demo {public static void change(int[] arr){arr[1] = 100 ;}public static void main(String[] args) {int[] arr = {1,2,3} ;System.out.println(arr[1]); //2change(arr);System.out.println(arr[1]); //100}}
4.2.3 方法重载的特点
重载仅对应方法的定义,与方法的调用无关,调用方式参照标准格式;
- 重载仅针对同一个类中方法的名称与参数进行识别,与返回值无关;
需要多个方法在同一个类中,具有相同的方法名,参数不同或者类型不同或者数量不同。
4.2.4 构造方法概述
构造方法是一种特殊的方法,作用是创建对象。
格式:public class 类名{修饰符 类名(参数){}}
注意事项:
@Override 可以帮助检查重写方法的声明的正确性
注意事项:
:: 该符号是引用运算符,而它所在的表达式被称为方法引用
- 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) {
usePrintable((s)-> System.out.println(s));usePrintable(System.out::println);
} public static void usePrintable(Printable p){
p.printString("666666666666");
} }
public interface Printable { public void printString(String s); }
- 推导与省略- 如果使用Lambda,那么根据“可推导就是可省略”的原则,无需指定参数类型,也无需指定重载形式,它们都将被自动推导。- 如果使用方法引用,也是同样可以根据上下文进行推导- 方法引用是Lambda的孪生兄弟<a name="UoApT"></a>#### 4.2.6.2 引用类方法- 其实就是引用类的静态方法- 格式:类名::静态方法```javapublic class Demo {public static void main(String[] args) {useConvert((s)-> {return Integer.parseInt(s);});useConvert(Integer::parseInt);}public static void useConvert(Converter p){int i = p.convert("66666666");System.out.println(i);}}***********************************************************************************public interface Converter {public int convert(String s);}
4.2.6.3 引用对象的实例方法
- 引用对象的实例方法就是引用类中的成员方法
- 格式:对象::成员方法
```java
public class Demo{
public static void main(String[] args) {
} public static void usePrint(Printer p){usePrint(s-> System.out.println(s.toUpperCase()));PrintString ps = new PrintString();usePrint(ps::printUpper);
} }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); }
<a name="i9RRM"></a>#### 4.2.6.4 引用构造器- 就是引用构造方法- 格式:类名::new```javapublic class Demo{public static void main(String[] args) {useStudentBuilder((name,age)->new Student(name,age));useStudentBuilder(Student::new);}public static void useStudentBuilder(StudentBuilder sb){Student s = sb.build("666",66);System.out.println(s.getName()+","+s.getAge());}}************************************************************************************public class Student {private String name ;private int age ;public Student(String name, int age) {this.name = name;this.age = age;}public Student() { }public int getAge() { return age; }public void setAge(int age) {this.age = age;}public String getName() { return name;}public void setName(String name) {this.name = name;}}************************************************************************************public interface StudentBuilder {Student build(String name,int age);}
4.2.6.5 引用类的实例方法
- 就是引用类中的成员方法
- 格式:类名::成员方法
```java
public class Demo{
public static void main(String[] args) {
} public static void useMyString(MyString ms){useMyString((s,x,y) ->s.substring(x,y));useMyString(String::substring);
} }String s = ms.mySubString("HelloWorld",2,5);System.out.println(s);
public interface MyString { String mySubString(String s,int x,int y); }
<a name="XlkhF"></a>## 4.3 封装<a name="TrzMD"></a>### 4.3.1 this关键字- this修饰的变量用于指代成员变量- 方法的形参如果与成员变量同名,不带this修饰的变量指的是形参,而不是成员变量- 方法的形参没有与成员变量同名,不带this修饰变量指的是成员变量- 解决局部变量隐藏成员变量的时候使用this- this代表所在类的对象引用,方法被那个对象调用,this就代表那个对象。<a name="Xbxa6"></a>## 4.4 继承<a name="Lkaha"></a>### 4.4.1 this和super- this 代表本类对象的引用:- this.成员变量:访问本类成员变量- this(...):访问本类构造方法- this.成员方法(...):访问本类成员方法- super 代表父类存储空间的标识(可以理解为父类对象的引用)- super.成员变量:访问父类成员变量- super(...):访问父类构造方法- super.成员方法(...):访问父类成员方法<a name="KvmGm"></a>### 4.4.2 构造方法访问特点- 子类中的所有构造方法都会默认访问父类中无参的构造方法- 因为子类会继承父类中的数据,可以还会使用父类的数据,所以,子类初始化之前,一定要先完成父类数据的初始化。- 每一个子类构造方法的第一条语句默认都是super()- 如果父类汇总没有无参构造方法,只有带参怎么办?(默认子类会报错)- 通过使用super关键字去显示调用父类的带参构造方法- 在父类中自己提供一个无参构造方法(推荐)- 注意事项:- java中类只支持单继承,不支持多继承- java中支持多层继承<a name="aTVZv"></a>## 4.5 多态- 多态的前提和体现:- 有继承/实现关系- 有方法重写- 有父类引用指向子类对象- 访问特点:- 成员变量:编译看左边,执行看右边- 成员方法:编译看左边,执行看右边- 不一样的原因是成员方法可有写,而成员变量没有- 利弊:- 利:提高了程序的延展性。具体实现:定义方法的时候,使用父类型作为参数,将来咋使用的时候,使用具体的子类型参与操作。- 弊:不能使用子类的特有功能- 多态转型,转型后可以解决弊端:(Animal为父类,Cat为子类)- Animal a = new Cat() ; //向上转型- Cat c = (Cat)a ; //向下转型<a name="MsTxS"></a>## 4.6 内部类- 概述:内部类就是在一个类中定义一个类- 访问特点:- 内部类可以直接访问外部类的成员,包括私有- 外部类要访问内部类的成员,必须创建对象- 根据定义的位置不同,可以分为两种形式:- 在类的成员位置:**成员内部类**- 外界创建成员内部类对象的格式: 外部类名.内部类名 对象名 = 外部类对象.内部类对象```javapackage com.bai;public class Outer {public class Inner{private int age = 66 ;public int getAge() {return age;}}}package com.bai;public class Demo {public static void main(String[] args) {Outer.Inner inner = new Outer().new Inner() ;System.out.println(inner.getAge()); //66}}
- 在类的局部位置:局部内部类
- 局部内部类是在方法中定义的类,所以外界无法直接使用,需要在方法内部创建对象并使用。该类可以直接访问外部类的成员,也可以访问方法内的局部变量。
- 匿名内部类:
- 前提:存在一个类或者接口,这里的类可以是具体类也可以是抽象类
- 格式:new 类名或者接口名(){ 重写方法 ; }
- 本质是一个继承了该类或者实现了该接口的子类匿名对象 ```java package com.bai;
public interface Jumpping { void jump(); }
```javapackage com.bai;public class Outer {public void method(){Jumpping j = new Jumpping(){ //接口@Overridepublic void jump(){System.out.println("跳高");}};j.jump();}}
package com.bai;public class Demo {public static void main(String[] args) {Outer outer = new Outer() ;outer.method();}}运行结果:跳高Process finished with exit code 0
4.7 抽象类
- 概述:
- 在java中,一个没有方法体的方法应该定义为抽象方法,而类中如果有抽象方法,该类必须定义为抽象类。
- 抽象类是引用数据类型。
- 特点:
- 抽象类和抽象方法必须使用 abstract 关键字修饰
- public abstract class 类名(){}
- public abstract void eat();
- 抽象类中不一定有抽象方法,有抽象方法的类一定是抽象类
- 抽象类不嫩实例化
- 如果需要实例化就要参照多态的方式,通过子类对象实例化,这叫抽象类多态
- 抽象类子类
- 要么重写抽象类中所有抽象方法
- 要么是抽象类
- 抽象类和抽象方法必须使用 abstract 关键字修饰
成员特点:
接口用关键字 interface 修饰,例:public interface 接口名{}
- 类实现接口用 implements 表示,例:public class 类名 implements 接口名{}
- 接口不能实例化:
- 接口实例化参照多态方式,通过实现类对象实例化,这叫接口多态
- 多态形式:具体类多条,抽象类多态,接口多态
- 多态的前提:有继承或者实现关系;有方法重写;有父类/父接口引用指向(子/实现)类对象。
接口实现类:
成员变量:
- 只能是常量
- 默认修饰符:public static final
- 构造方法:
- 接口没有构造方法,因为接口主要就是对行为进行抽象的,是没有具体存在的
- 一个类如果没有父类,默认继承自Object类
成员方法:
类和类的关系:继承关系,只能单继承,但是可以多层继承。
- 类和接口的关系:实现关系,可以单实现,也可以多实现,还可以在继承一个类的同时实现多个接口。
-
4.8.4 抽象类和接口的区别
成员区别:
- 接口:常量,抽象方法
- 抽象类:变量,常量;有构造方法;也有非抽象方法。
- 关系区别:与类和接口的关系一样
设计理念区别:
常量:public static final
- 抽象方法:public abstract
- 默认方法(Java8)
- 定义格式:public default 返回值类型 方法名(参数列表){}
- 注意事项:默认方法不是抽象方法,所以不强制被重写,但是可以被重写;public可以省略,default不能省略。
- 静态方法(Java8)
- 定义格式:public static 返回值类型 方法名(参数列表)
- 注意事项:静态方法只能通过接口名调用,不能通过实现类名或者对象调用。public可以省略,static不能省略。
私有方法(Java9)
函数式接口:有且仅有一个抽象方法接口
- Java中的函数编程体现就是Lambda表达式,所以函数式接口就是可以适用于Lambda使用接口,只有确保接口中有且仅有一个抽象方法,Java中的Lambda才能顺利进行。
- @functionalInterface
- 检测一个接口是不是函数式接口
- 放在接口上方,如果是函数式接口,编译通过;如果不是,编译失败。
- 我们自己定义函数式接口时,这个东西是可选的,只要保证满足函数式接口定义的条件,也照样是函数式接口。建议加上这个注解。
```java / 函数式接口作为方法的返回值 / import java.util.ArrayList; import java.util.Collections; import java.util.Comparator;/* 函数式接口作为方法的参数 */public class Demo {public static void main(String[] args) {startThread(new Runnable() {@Overridepublic void run() {System.out.println(Thread.currentThread().getName()+","+"线程启动了1");}});startThread(()-> System.out.println(Thread.currentThread().getName()+","+"线程启动了2"));}public static void startThread(Runnable r){new Thread(r).start();}}
public class Demo {
public static void main(String[] args) {
ArrayList
**常用函数式接口**- **_Supplier_**- Supplier<T>:包含一个无参的方法- **_T get()_**:获得接口- 该方法不需要参数,他会按照实现逻辑(由Lambda表达式实现)返回一个数据- Supplier<T>接口也称为生产型接口,如果我们指定了接口的泛型是什么类型,那么接口中的get方法就会生产什么类型的数据供我们使用。```javaimport java.util.function.Supplier;public class Demo {public static void main(String[] args) {String s = getString(()->"666") ;System.out.println(s);Integer i = getInteger(()->666) ;System.out.println(i);}private static String getString(Supplier<String > sup){return sup.get() ;}private static Integer getInteger(Supplier<Integer> sup ){return sup.get() ;}}
/* 返回数组中最大值 */import java.util.Arrays;import java.util.function.Supplier;public class Demo {public static void main(String[] args) {int[] arr = {2,6,1,38,78,12} ;int i = getMax(()->{Arrays.sort(arr);return arr[arr.length-1] ;}) ;System.out.println(i);}private static int getMax(Supplier<Integer> sup){return sup.get() ;}}
- Consumer
- Consumer
:包含两个方法 - void accept(T t):对给定的参数执行此操作
- default Consumer
andThen(Consumber after) :返回一个组合的Consumer,依次执行此操作,然后执行after操作 - Consumer
接口也被称为消费型接口,他消费的数据的数据类型由泛型指定 ```java import java.util.function.Consumer;
- 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()));
operatorString("456",s-> System.out.println(s),s -> System.out.println(new StringBuilder(s).reverse().toString()));}private static void operatorString(String name, Consumer<String> con1,Consumer<String> con2){//con1.accept(name);//con2.accept(name);con1.andThen(con2).accept(name); //和上面两行执行结果一样}private static void operatorString(String name, Consumer<String> con1){con1.accept(name);}
}
```java/* 打印信息 */import java.util.function.Consumer;public class Demo {public static void main(String[] args) {String[] strArray = {"小黑,30","小黄,35","小绿,33"} ;getStudent(strArray,s-> System.out.print("姓名-->"+s.split(",")[0]),s -> System.out.println(" 年龄-->"+s.split(",")[1]));}public static void getStudent(String[] name,Consumer<String> c1,Consumer<String> c2){for(String str:name){c1.andThen(c2).accept(str);}}}运行结果:姓名-->小黑 年龄-->30姓名-->小黄 年龄-->35姓名-->小绿 年龄-->33Process 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;
- 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
```java/* 输出名字三个字,大于33岁的 */import java.util.ArrayList;import java.util.function.Predicate;public class Demo {public static void main(String[] args) {String[] strArray = {"小黑黑,30","小黄,34","小绿绿,35","小白,31","小蓝蓝,33"};ArrayList<String> arrayList = getMessage(strArray,s->s.split(",")[0].length()>2,s -> Integer.parseInt(s.split(",")[1])>33);System.out.println(arrayList);}private static ArrayList<String> getMessage(String[] arr,Predicate<String> p1,Predicate<String> p2){ArrayList<String> arrayList = new ArrayList<>() ;for(String str:arr){if(p1.and(p2).test(str))arrayList.add(str);}return arrayList ;}}运行结果:[小绿绿,35]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;
- 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
```java/* 按照指定要求操作数据 */import java.util.function.Function;public class Demo {public static void main(String[] args) {String ss = "小黑,30" ;//Integer.parseInt(s) 可以通过方法引用修改为 Integer::parseIntmodifyAge(ss,s->s.split(",")[1],s->Integer.parseInt(s),s->s+70);}private static void modifyAge(String s,Function<String,String> f1,Function<String,Integer> f2,Function<Integer,Integer> f3){int i = f1.andThen(f2).andThen(f3).apply(s) ;System.out.println(i); //100}}
