8月07日

接口

概述

接口:是java语言中一种引用类型,是方法的集合,如果说类的内部封装了成员变量、构造方法和成员方法,那么接口的内部主要就是封装了方法,包含抽象方法(JDK1.7以前)、默认方法、静态方法(JDK1.8)和私有方法(JDK1.9)。

接口的定义:它与定义类的方式相似,但是使用interface关键字。它也会被编译成.class文件,但一定要明确它并不是类,而是另外一种引用数据类型。

引用数据类型:数组、类、接口。

接口的使用:它不能创建对象,但是可以被实现(implements,类似于被继承)。一个实现接口的类(可以看做是接口的子类),需要实现接口中所有的抽象方法,创建该类对象,就可以调用方法,否则它必须是一个抽象类。

定义格式

 public interface 接口名{
    	//抽象方法;
    	//默认方法;
    	//静态方法;
    	//私有方法;
    }

抽象方法

抽象方法:使用abstract关键字去修饰,可以省略,没有方法,该方法子类实现使用。

public interface InterfaceName {
	//抽象方法:
	public abstract void method();
}

默认方法

默认方法:使用default关键字修饰,不可省略,供子类调用或者子类重写。

public interface InterfaceName {
	//默认方法:
	public default void method1(){
    	//执行语句
    }
}

静态方法

静态方法:使用static关键字修饰,供接口直接调用。

 public interface InterfaceName {
    	//静态方法:
      	public static void method2(){
        	//执行语句
      	}
    }

私有方法和私有静态方法

私有方法:使用private关键字修饰,供接口中的默认方法调用。

public interface InterfaceName {
	//私有方法:
  	private void add(){
    	//执行语句
  	}
}

私有静态方法:使用private static关键字修饰,供接口中的默认方法或者静态方法调用(JDK1.9特性)。

public interface InterfaceName {
	//私有静态方法方法:
  	private static void add(){
    	//执行语句
  	}
}

基本实现

类和接口的关系就是实现关系,我们类称为接口的实现类,也可以称为接口的子类,使用implements关键字。

非抽象类实现接口

1、必须重写接口中所有的抽象方法。

2、继承接口的默认方法,可以直接调用,也可以重写。

public class Demo implements InterfaceName{
	//重写接口中所有的抽象方法,(必须要搞)
	//重写接口中默认的方法(可选性)
}

抽象方法的使用

接口:

public interface Person {
	//定义一些抽象方法
	public abstract void eat();
	public abstract void sleep();
}

实现类:

 public class Student implements Person{
    	@Override
    	public void eat() {
    		System.out.println("吃的挺好!");
    	}
    	@Override
    	public void sleep() {
    		System.out.println("睡的还行");
    	}
    }

测试类:

 public class TestDemo {
    	public static void main(String[] args) {
    		//创建对象,直接调用
    		Student student=new Student();
    		//调用实现过的方法
    		student.eat();
    		student.sleep();
    	}
    }

默认方法的使用

可以继承,也可以重写,但是只能通过我们实现类的对象去调用。

1、继承默认方法

public interface Person {
	//默认方法:
	public default void add(){
		System.out.println("我是接口中的默认方法!");
	}
}
 public class Student implements Person{
    	//不重写我们接口中的默认方法
    }
 public class TestDemo {
    	public static void main(String[] args) {
    		//创建对象,直接调用
    		Student student=new Student();
    		student.add();
    	}
    }

2、重写默认方法

 public interface Person {
    	//默认方法:
    	public default void add(){
    		System.out.println("我是接口中的默认方法!");
    	}
    }
 public class Student implements Person{
    	@Override
    	public void add() {
    		System.out.println("我是实现类中的方法add");
    	}
    }
public class TestDemo {
	public static void main(String[] args) {
		//创建对象,直接调用
		Student student=new Student();
		student.add();
	}
}

静态方法的使用

静态和.class文件相关,只能是我们的接口的名字取 调用,不可以通过我们实现类的类名或者实现类的对象去调用。

 public interface Person {
    	public static void add2(){
    		System.out.println("我是接口中的静态方法");
    	}
    }
public class Student implements Person{
	//无法重写我们接口中的静态方法
}
public class TestDemo {
	public static void main(String[] args) {
		//创建对象,直接调用
		Student student=new Student();
		//student.add2();不可以通过实现类的对象去调用
		//Student.add2();不可以通过实现类的类名去调用
		Person.add2();//只能通过接口名去调用
	}
}

私有方法的使用

私有方法:只能在接口中的默认方法可以调用。

 public interface Person {
    	//默认方法:
    	public default void add(){
    		f();
    		f2();
    	}
    	//定义了两个私有的方法
    	private void f(){
    		System.out.println("我是接口中的私有方法f");
    	}
    	private void f2(){
    		System.out.println("我是接口中的私有方法f2");
    	}
    }

私有静态方法:当我们在接口中定义的私有方法是静态的时候,可以在接口中的默认方法和静态方法中调用。

 public interface Person {
    	public static void add2(){
    		f();
    		f2();
    		System.out.println("我是接口中的静态方法");
    	}
    	private static void f(){
    		System.out.println("我是接口中的私有方法f");
    	}
    	private static void f2(){
    		System.out.println("我是接口中的私有方法f2");
    	}
    }

接口的多实现

在继承体系中,一个类智能继承一个父类,而对于接口来说,一个类可以实现多个接口,这个叫接口的多实现,同时我们一个类,在继承父类的前提下,可以实现多个接口。

格式

class 类名 【extends 父类名】 implements 接口1,接口2{
	//重写我们接口中的抽象方法
	//重写接口中默认方法【不重名的可以可选】
}

抽象方法

接口中,有多个抽象方法,实现类必须重写接口中的所有的抽象方法,如果抽象方法有重名,只需要重写一次。

public interface A {
	//定义抽象方法
	public abstract void method1();
	public abstract void method2();
}
 public interface B {
    	//定义抽象方法
    	public abstract void method1();
    	public abstract void method3();
    }
public class C implements A,B{
	@Override
	public void method1() {
		System.out.println("c中的method1");
	}
	@Override
	public void method3() {
		System.out.println("c中的method3");
	}
	@Override
	public void method2() {
		System.out.println("c中的method2");
	}
}

默认方法

在接口中,有多个默认方法,实现类都可以继承,如果默认方法重名,必须重写一次。

  public interface A {
    	//默认方法
    	public default void add1(){}
    	public default void add2(){}
    }
public interface B {
	//默认方法
	public default void add1(){}
	public default void add3(){}
}
 public class C implements A,B{
    	//因为我们A和B都有一个add1的方法,所以我们必须要重写一下
    	//add2和add3可以根据自己的选择去重写,也可以不重写
    	@Override
    	public void add1() {
          
    	}
    }

静态方法

多个接口中,存在同名的静态方法并不会冲突,原因就是各个接口只能通过接口名去调用静态方法。

优先级问题

当一个类,既继承了父类,又实现了若干个接口的时候,父类中的方法和接口中的默认方法重名,子类就近原则执行父类中的成员方法。

public interface A {
	//默认方法
	public default void add1(){
		System.out.println("我是接口a中的add1");
	}
}
public class D {
	public void add1(){
		System.out.println("我是类d中的add1");
	}
}
 public class C extends D implements A{
    	//并不会重写我D中和A中相同的方法名的方法
    }
public class TestDemo {
	public static void main(String[] args) {
		C c=new C();
		c.add1();//访问是我们的D中的add1方法
	}
}

接口的多继承

一个接口可以继承另外一个接口或者多个接口,这和类之间继承关系相似,接口继承也用extends关键字,子接口继承父接口的方法,如果父接口中默认的方法有重名,子接口只重写一次。

public interface A {
	//默认方法
	public default void add1(){
		System.out.println("我是接口a中的add1");
	}
}
public interface B {
	//默认方法
	public default void add1(){
		System.out.println("我是接口B的add1");
	}
}
public interface E extends A,B{
	@Override
	default void add1() {
	//因为我们A和B中有重名的方法,所以我们E接口需要重写
	}
}

注意点

1、子接口重写默认方法的时候,default关键字可以保留。

2、子类重写默认方法的时候,default关键字不可以保留。

其他的成员特点

1、接口中,无法定义成员变量,但是可以定义常量,常量值不可以改变。默认使用public static final修饰。

public interface A {
	public static final int a=0;
	//public int a1;
}

2、接口中没有构造方法,不能创建对象。

3、接口中也没有静态代码块。

多态

多态是我们三大特性之一,前面已经讲过我们封装、继承。比如我们动物,猫、狗。他们的叫声不一样,可见同一个行为,通过不同的事物,表现出不同的形态。多态,就是描述的这个状态。

前提

1、继承或者实现(二选一)。

2、方法重写【重写,不重写就没意义了】。

3、父类的引用指向子类对象【多态的格式】。

格式

父类类型  类名=new 子类类型();
类名.变量名;

父类的类型:指子类对象继承父类的引用,或者实现父接口类型。

 public class Fu {
    }
public class Zi extends Fu{
}
 public class Demo {
    	public static void main(String[] args) {
    		//多态的定义格式
    		Fu fu=new Zi();
    	}
    }

多态的表现形式

public class Fu {
	//普通的变量
	public int a=10;
	//普通的方法
	public void add(){
		System.out.println("父类中的add方法");
	}
	public static void addd2(){
		System.out.println("我是父类中的addd2");
	}
}
public class Zi extends Fu{
	public int a=20;
	public void add(){
		System.out.println("子类中的add方法");
	}
	public static void addd2(){
		System.out.println("我是子类中的addd2");
	}
}
 public class Demo {
    	public static void main(String[] args) {
    		//多态的定义格式
    		Fu fu=new Zi();
    		//如果这个调用是成员方法
    		//编译看左边,运行结果看右边
    		fu.add();
    		//如果它是个普通的变量
    		//编译看左边,运行看左边
    		System.out.println(fu.a);
    	}
    }

多态的好处

实际开发的时候,父类类型作为我们方法的参数,传递子类对象给方法,进行方法的声明。

 public abstract class Animal {
    	public abstract void eat();
    }
public class Cat extends Animal{
	@Override
	public void eat() {
		System.out.println("猫吃鱼");
	}
}
public class Dog extends Animal {
	@Override
	public void eat() {
		System.out.println("狗吃屎!");
	}
}
 public class Demo {
    	public static void main(String[] args) {
    		Cat cat=new Cat();
    		Dog dog=new Dog();
    		showDogAndCatEat(cat);
    		showDogAndCatEat(dog);
    	}
    	public static void showDogAndCatEat(Animal animal){
    		animal.eat();
    	}
    }

引用类型的转换

多态的转型分为向上转型和向下转型。

向上转型

多态本身就是子类类型指向父类类型的向上转型这么一个过程。

父类类名 类名=new 子类类型();
Animal animal=new Dog();

向下转型

父类类型想子类类型向下转化的过程,这个过程其实就是强制转换的一个过程。

子类类型 名字=(子类类型) 父类变量名;
Dog dog= (Dog) animal;

为什么会有转型

当使用多态的方式调用方法时,首先检查父类中是否有该方法,如果没有,编译错误,不能调用子类拥有而父类没有的方法,所以才有了多态向下转型。

public abstract class Animal {
	public abstract void eat();
}
public class Dog extends Animal {
	@Override
	public void eat() {
		System.out.println("狗吃屎!");
	}
	public void add(){
		System.out.println("狗中的add");
	}
}
 public class Demo {
    	public static void main(String[] args) {
    		Animal animal=new Dog();
    		Dog dog= (Dog) animal;
    		dog.add();
    	}
    }

转型异常:

转型过程中,我们容易出现类型转换异常。

public static void main(String[] args) {
	Animal animal=new Dog();
	Cat cat=(Cat) animal;
	cat.eat();
}

在这里插入图片描述
当我们使用多态定义的时候,我们用动物的父类声明,子类Dog接受,但是我们后面用向下转型转为了Cat,逻辑没有任何问题,编译也看不出,但是运行会报错—ClassCastException。

判断我们类型是否是定义的类型:

if(cat instanceof Cat) {
	cat.eat();
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值