一、类变量‼️
1. 基本介绍
类变量也叫静态变量/静态属性,是该类的所有对象共享的变量,任何一个该类的对象去访问它时,取到的都是相同的值,同样任何一个该类的对象去修改它时,修改的也是同一个变量
2. 定义
- 访问修饰符 static 数据类型 变量名
- static 访问修饰符 数据类型 变量名
3. 访问
- 类名.类变量名
- 对象名.类变量名
4. 使用
当我们需要让某个类的所有对象都共享一个变量时,就可以考虑使用类变量
5. 注意示例
- 类变量与实例变量的区别:类变量是该类的所有对象共享的,实例变量是每个对象独享的
- 加上static称为类变量或静态变量,否则称为实例变量/普通变量/非静态变量
- 实例对象不能通过类名.类变量名 来访问
- 类变量是在类加载时就初始化了
- 类变量的生命周期是随类的加载开始的,随类销往而销毁
二、类方法‼️
1. 基本介绍
类方法也叫静态方法
2. 定义
- 访问修饰符 static 数据返回类型 方法名(){}
- static 访问修饰符 数据返回类型 方法名(){}
3. 调用
- 类名.类方法名
- 对象名.类方法名
4. 使用
当方法中不涉及到任何和对象相关的成员,则可以将饭给发设计成静态方法,提高效率
5. 注意示例
- 类方法和普通方法都是随着类的加载而加载的,将结构信息存储在方法区:类方法中无this的参数;普通方法中隐含着this的参数
- 普通方法和对象有关,需要通过对象名调用
- 类方法中不允许使用和对象有关的关键字this和super;普通方法可以
- 类方法中只能访问静态变量和静态方法
- 普通成员方法既可以访问非静态成员,也可访问静态成员
三、main方法
1. 深入理解
- 解释main方法的形式:public static void main(String [] args):
- main方法是虚拟机调用
- java虚拟机需要调用类的main()方法,所以该方法的访问权限必须是public
- java虚拟机在执行main()方法时不必创建对象。所以该方法必须是static
- 该方法接收String类型的数组参数,该数组中保存执行java命令时传递给所运行的类的参数
- java执行的程序 参数1 参数2 参数3 参数4
2. 特别提示
- 在main()方法中,我们可以直接调用main方法所在类的静态方法或静态属性
- 但是,不能直接访问该类的非静态成员,必须创建该类的一个实例对象后,才能通过这个对象去访问类中的非静态成员
四、代码块‼️‼️
1. 基本介绍
代码化块又称为初始化块,属于类的成员,类似于方法,将逻辑语句封装在方法体中,通过{}包围起来
但和方法不同,没有方法名,没有返回,没有参数,只要方法体,而且不用通过对象或显式调用,而是加载类时,或创建对象时隐式调用
2. 基本语法
- [修饰符]{
代码;
};
注意:
1)修饰符可选,要写的话也只能写static
2)代码块分两种1⃣️使用static修饰的静态代码块2⃣️没有static修饰的普通代码块
3);号可以写,也可省略
3. 优点
- 相当于另外一种形式的构造器(对构造器的补充机制),可以做初始化操作
- 如果多个构造器中都有重复的语句,可以抽取到初始化块中,提高代码的重用性
4. 注意示例‼️‼️
- static代码块也叫静态代码块,作用就是对类进行初始化,而且它随着类的加载而执行,并且只会执行一次。如果是普通代码块,每创建一个对象,就执行一次
- 类什么时候被加载‼️
1⃣️创建对象实例时(new)
2⃣️创建子类对象实例,父类也会被加载
3⃣️使用类的静态成员时(静态属性,静态方法) - 普通的代码块,在创建对象实例时,会被隐式的调用。被创建一次,就会调用一次。如果只是使用类的静态成员时,普通代码块井不会执行。
- 创建一个对象时,在一个类调用顺序是:‼️
1⃣️调用静态代码块和静态属生初给化
(注意:静态代码块和静态属性初始化调用的优先级一样,如果有多个静态代码块和多个静态变量初始化,则按他们定义的顺序调用)[举例说明]
2⃣️调用普通代码块和普通属性的初始化
(注意:普通代码块和普通属性初始化调用的优先级一样,如果有多个普通代码块和多个普通属性初始化,则按定义顺序调用
3⃣️调用构造方法 - 构造器的最前面其实隐含了super() 和 调用普通代码块,静态相关的代码块,属性初始化,在类加载时,就执行完毕因此是优先于构造器和普通代码块执行的
- 创建一个子类对象时,调用顺序:‼️‼️
1⃣️父类的静态代码块和静态属性(优先级一样,按定义顺序执行)
2⃣️子类的静态代码块和静态属性(优先级一样,按定义顺序执行)
3⃣️父类的普通代码块和普通属性初始化(优先级一样,披定义顺序执行)
4⃣️父类的构造方法
5⃣️子类的普通代码块和普通属性初始化(优先级一样,披定义顺序执行)
6⃣️子类的构造方法
五、单例设计模式
1. 基本介绍
采取一定的方法保证在整个的软件系统中对某个类只能存在一个对象实例,并且该类只提供一个取得其对象实例的方法
2. 饿汉式、懒汉式 -步骤
1⃣️构造器私有化
2⃣️类的内部创建对象
3⃣️向外暴露一个静态的公共方法
3. 饿汉式 VS 懒汉式
- 创建对象的时机不同:饿汉式是在类加载就创建了对象实例,懒汉式是在使用时创建的
- 饿汉式不存在线程安全问题,懒汉式存在
- 饿汉式有浪费资源的可能
六、final关键字
1. 基本介绍
可以修饰类、属性、方法、局部变量
2. 使用时机
- 类:不希望类被继承
- 方法:不希望父类的某个方法被子类覆盖/重写
- 属性:不希望类的某个属性被修改
- 局部变量:不希望局部变量被修改
3. 示例
- final修饰的属性又叫常量,一般用XX_XX_XX命名
- final修饰的属性在定义时,必须赋初值,并且以后不能再修改
- 赋初值可以在:
1⃣️定义时
2⃣️在构造器中
3⃣️在代码块中 - 如果final修饰的属性是静态的,赋初值位置:⚠️
1⃣️定义时
2⃣️在静态代码块(不能在构造器中赋值) - final类不能被继承,但可以实例化对象
- 如果类不是final类,但含有final方法,则该方法虽然不能被重写,但是可以被继承
- final不能修饰构造方法
- final和static往往搭配使用,效率更高,不会导致类加载,底层编译器做了优化处理
七、抽象类‼️
1. 基本介绍
- 用abstract关键字来修饰一个类时,这个类就是抽象类
访问修饰符 abstract 类名{
} - 用abstract关键字修饰一个方法时,这个方法就是抽象方法
访问修饰符 abstract 返回类型 方法名(参数列表);//没有方法体 - 抽象类的价值更多作用是在于设计,设计好后,让子类继承并实现抽象类
2. 注意示例
- 抽象类不能被实例化
- 抽象类不一定包含abstract方法
- 一旦包含abstract方法,则这个类必须声明为abstract
- abstract只能修饰类和方法,不能修饰属性和其他的
- 抽象类可以有任意成员
- 抽象方法不能有主体
- 如果一个类继承了抽象类,则它必须实现抽象类的所有抽象方法,除非它自己也声明为abstract类
- 抽象方法不能使用private、final、static来修饰,因为这些关键字都和重写相违背
八、模版设计模式
1. 基本介绍
抽象类体现的就是一种模版设计模式,抽象类可以作为多个子类的通用模版,子类在抽象类的基础上进行扩展、改造,但子类总体上会保留抽象类的行为方式
2. 解决问题
- 当功能内部一部分是实现是确定的,一部分实现是不确定的。这时可以把不确定的部分暴露出来,让子类去实现
- 编写一个抽象父类,父类提供了多个子类的通用方法,并把一个或多个方法留给其子类实现,这就是一种模版模式
3. 代码示例
package Java.AbstractTemplate;
abstract public class AbstractTemplate {
public abstract void job();
public void calculateTime() {
long start = System.currentTimeMillis();
job();
long end = System.currentTimeMillis();
System.out.println("执行时间为 " + (end - start));
}
}
package Java.AbstractTemplate;
public class AA extends AbstractTemplate{
@Override
public void job() {
int num = 0;
for (int i = 0; i < 80000; i++) {
num += i;
}
}
}
package Java.AbstractTemplate;
public class BB extends AbstractTemplate {
@Override
public void job() {
int num = 0;
for (int i = 0; i < 8000; i++) {
num *= i;
}
}
}
package Java.AbstractTemplate;
public class Test {
public static void main(String[] args) {
AA aa = new AA();
aa.calculateTime();
BB bb = new BB();
bb.calculateTime();
}
}
九、接口‼️
1. 基本介绍
接口就是给出一些没有实现的方法,封装在一起,到某个类要使用的时候,再根据具体情况把这些方法写出来
interface 接口名{
//属性
//抽象方法
}
class 类名 implements 接口{
1自己属性
2自己方法
3必须实现的接口的抽象方法
}
- 【JDK7.0】接口是更加抽象的抽象的类,抽象类里的方法可以有方法体,接口里的所有方法都没有方法体
- 【JDK8.0后】接口类可以有静态方法、默认方法,也就是说接口中可以有方法的具体实现
- 接口体现了程序设计的多态和高内聚低耦合的设计思想
package Java.Interface;
public interface Interface {
public void start();
public void end();
}
package Java.Interface;
public class Camera implements Interface{
@Override
public void start() {
System.out.println("照相机开始工作");
}
@Override
public void end() {
System.out.println("照相机结束工作");
}
}
package Java.Interface;
public class Phone implements Interface{
@Override
public void start() {
System.out.println("手机开始工作");
}
@Override
public void end() {
System.out.println("手机结束工作");
}
}
package Java.Interface;
public class Computer {
public void work(Interface usbInterface){
usbInterface.start();
usbInterface.end();
}
}
package Java.Interface;
public class Test {
public static void main(String[] args) {
Phone phone = new Phone();
Camera camera = new Camera();
Computer computer = new Computer();
computer.work(phone);
computer.work(camera);
}
}
2. 注意示例
- 接口不能被实例化
- 接口中所有的方法是public方法,接口中抽象方法可以不用abstract修饰
- 一个普通类实现接口,就必须将该接口的所有方法实现
- 抽象类实现接口,可以不用实现接口的方法
- 一个类可以实现多个接口
- 接口中的属性只能是final的,并且是 public static final 修饰符(比如:int a = 1;—>实际上是public static final int a = 1;)
- 接口中属性的访问形式:接口名.属性名
- 接口不能继承其他类,但是可以继承多个接口
- 接口的修饰符只能是public和默认,这点和类的修饰符一样
3. 接口 VS 继承
- 继承的价值:解决代码的复用性和可维护性
- 接口的价值:设计好的各种规范,让其他类去实现这些方法,比继承更加灵活
- 接口在一定程度上实现代码的解耦
package Java.Interface;
public class Extends_VS_Interface {
public static void main(String[] args) {
LittleMonkey littleMonkey = new LittleMonkey("悟空");
littleMonkey.flying();
littleMonkey.swimming();
littleMonkey.climbing();
}
}
class Monkey {
private String name;
public Monkey(String name) {
this.name = name;
}
public void climbing() {
System.out.println(name + " 爬树");
}
public String getName() {
return name;
}
}
interface Fishable {
void swimming();
}
interface Birdable {
void flying();
}
class LittleMonkey extends Monkey implements Fishable, Birdable {
public LittleMonkey(String name) {
super(name);
}
@Override
public void swimming() {
System.out.println(getName() + " 学会了游泳");
}
@Override
public void flying() {
System.out.println(getName() + " 学会了飞行");
}
}
4. 多态特性
- 多态参数
- 多态数组
package Java.Interface;
import com.zhou.package1.Person;
public class InterfacePolyArr {
public static void main(String[] args) {
Usb[] usbs = new Usb[2];
usbs[0] = new Phone_();
usbs[1] = new Camera_();
for (int i = 0; i < 2; i++) {
usbs[i].work();
if (usbs[i] instanceof Phone_){
((Phone_) usbs[i]).call();
}
}
}
}
interface Usb{
void work();
}
class Phone_ implements Usb{
@Override
public void work() {
System.out.println("手机工作中");
}
public void call() {
System.out.println("手机正在打电话");
}
}
class Camera_ implements Usb {
@Override
public void work() {
System.out.println("照相机工作中");
}
}
- 接口存在多态传递现象
package Java.Interface;
public class Phone implements Interface{
@Override
public void start() {
System.out.println("手机开始工作");
}
@Override
public void end() {
System.out.println("手机结束工作");
}
}
十、内部类‼️⚠️⚠️‼️
基本介绍
一个类的内部又完整的嵌套了另一个类的结构
基本语法
class Outer{ //外部类
class Inner{ //内部类
}
}
class Other{ //外部其他类
}
分类
- 定义在外部类的局部位置上(比如方法内):
- 局部内部类(有类名)
- 匿名内部类(没有类名)
- 定义在外部类的成员位置
- 成员内部类(没有static修饰)
- 静态内部类(使用static修饰)
1. 局部内部类
定义在外部类局部位置上(有类名)
- 不能添加访问修饰符,可以使用final修饰,本身是一个局部变量
- 作用域:1⃣️定义它的方法中 2⃣️代码块中
- 局部内部类——访问——>外部内部类 [直接访问]
- 外部类——访问——>局部内部类 [创建对象,再访问]
- 外部其它类——不能❌访问——>局部内部类
- 如果外部类和局部内部类的成员重名时,默认遵循就近原则(访问:外部类名.this.成员)
2. 匿名内部类‼️⚠️
定义在外部类局部位置上(没有类名)
new 类/接口(参数列表){
类体
};
- 不能添加访问修饰符,本身是一个局部变量
- 作用域:1⃣️定义它的方法中 2⃣️代码块中 ‼️
- 匿名内部类——访问——>外部内部类 [直接访问]
- 外部类——访问——>匿名内部类 [创建对象,再访问]
- 外部其它类——不能❌访问——>匿名内部类 ⚠️
- 如果外部类和匿名内部类的成员重名时,默认遵循就近原则(访问:外部类名.this.成员)
package Java.InnerClass;
public class AnonymousClass {
public static void main(String[] args) {
Outer outer = new Outer();
outer.fun();
}
}
class Outer {
public void fun() {
//ia的编译类型是IA
//ia的运行类型是Outer$1
/**
* class Outer$1 implements IA{
* }
*/
IA ia = new IA() {
@Override
public void cry() {
System.out.println("匿名内部类");
}
};
System.out.println("ia的运行类型:" + ia.getClass());
ia.cry();
//other的编译类型是Other
//other的运行类型是Outer$2
/**
* class Outer$2 exends Other{
* }
*/
//注意:("Tom")参数列表会传递给构造器
Other other = new Other("Tom"){
@Override
public void test() {
System.out.println("匿名内部类重写了test方法");
}
};
System.out.println("other的运行类型:" + other.getClass());
other.test();
}
}
interface IA {
void cry();
}
class Other{
public Other(String name) {
System.out.println("接收到name= " + name);
}
public void test() {
}
}
package Java.InnerClass;
public class AnonymousClassExercise {
public static void main(String[] args) {
// new Other_("Tom"){
// @Override
// public void test() {
// System.out.println("匿名内部类重写了test方法");
// }
// }.test();
fun(new IA_() {
@Override
public void show() {
System.out.println("匿名内部类作为实参");
}
});
}
public static void fun(IA_ ia_) {
ia_.show();
}
}
interface IA_{
void show();
}
//class Other_{
// public Other_(String name) {
// System.out.println("接收到name=" + name);
// }
//
// public void test() {
// }
//}
3. 成员内部类
定义在外部类的成员位置,没有static修饰
- 可以添加任意访问修饰符,本身是一个成员
- 作用域:整个类体
- 成员内部类——访问——>外部内部类 [直接访问]
- 外部类——访问——>成员内部类 [创建对象,再访问]
- 外部其它类——访问——>成员内部类
Outer outer = new Outer();
//第一种:把new Inner_()当作outer_的成员
Outer.Inner inner1 = outer.new Inner();
inner1.show();
//第二种:在外部类中,编写一个方法,返回Inner_对象
Outer.Inner inner2 = outer.getInner_();
inner2.show();
- 如果外部类和成员内部类的成员重名时,默认遵循就近原则(访问:外部类名.this.成员)
package Java.InnerClass;
public class MemberClass {
public static void main(String[] args) {
Outer_ outer_ = new Outer_();
// outer_.fun();
//第一种:把new Inner_()当作outer_的成员
Outer_.Inner_ inner_1 = outer_.new Inner_();
inner_1.show();
//第二种:在外部类中,编写一个方法,返回Inner_对象
Outer_.Inner_ inner_2 = outer_.getInner_();
inner_2.show();
}
}
class Outer_ {
private String name = "Tom";
private static int age = 18;
public void fun() {
System.out.println("外部类的fun()");
}
public class Inner_{
public void show() {
System.out.println(name + " " + age);
fun();
}
}
public Inner_ getInner_() {
return new Inner_();
}
}
4.静态内部类
定义在外部类的成员位置,使用static修饰
- 可以添加任意访问修饰符,本身是一个成员
- 作用域:整个类体
- 静态内部类——访问——>外部内部类 [直接访问]
- 外部类——访问——>静态内部类 [创建对象,再访问]
- 外部其它类——访问——>静态内部类
//第一种:把new Inner_()当作outer_的成员
Outer.Inner inner1 = new Outer.Inner();
inner1.show();
//第二种:在外部类中,编写一个方法,返回Inner_对象
Outer.Inner inner2 = Outer.getInner_();
inner2.show();
- 如果外部类和静态内部类的成员重名时,默认遵循就近原则(访问:外部类名.成员)