类
普通类和抽象类
类:是一组相关属性和行为的集合。(内部封装了成员变量、构造方法、成员方法)
对象:是一类事物的具体体现。对象是类的实例,必然具备该类事物的属性和行为。
类和对象的关系:类是对象的抽象,对象是类的实体。
构造方法<作用>:初始化成员变量。
普通类:没有被abstract修饰,没有抽象方法的类。
抽象类:被abstract修饰,可能含有抽象方法的类。
两者的区别:
① 含有抽象方法的类,一定是抽象类;
② 抽象类有构造器,但不能被实例化;(因为抽象方法没有方法体,不需要实现。)
③ 抽象类被abstract修饰,而普通类没有;
④ 抽象类的子类,必须重写父类中的所有抽象方法。除非子类也是抽象类。
内部类:将一个类A定义在类B或者方法C里面,类A即为内部类。内部类中不能定义静态成员。
- Java 内部类种类及使用解析
作用:①方便代码编写 ②使得多继承机制变得更加完善 ③省去实现类的定义
补充:内部类仍然是一个独立的类,在编译之后会内部类会被编译成独立的.class文件,但是前面冠以外部类的类名和$符号。比如,Person$Heart.class(内部类编译文件)
成员内部类:定义在类中 方法外的类。
格式如下
class 外部类 {
class 内部类 {
//
}
}
访问特点
- 内部类可以直接访问外部类的成员,包括私有成员和静态成员。
- 外部类名.内部类名 对象名= new 外部类型().new 内部类型();)
- 同名的成员变量和方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员,如果想访问外部类的成员,则需要以下形式进行访问:
- 外部类.this.成员变量/成员方法
匿名内部类:没有名字的内部类,是内部类的简化写法,是唯一一种没有构造器的类。匿名内部类写出来就会产生一个匿名内部类的对象。(匿名内部类对象类型相当于是new Xxx的子类类型)
实现前提:必须继承一个父类或者实现了一个父接口。
作用:方便创建子类对象,最终目的为了简化代码编写。
格式:
new 父类名或者接口名() {
//方法重写
@override
public void method( ) {
//执行语句
}
}
举例
// 抽象类
public abstract class FlyAble{
public abstract void fly();
}
//demo
public class demo {
public static void main( string[]args) {
/*
* 创建匿名内部类,直接传递给showFly(FlyAble f)
*/
showFly(
new FlyAble(){
public void fly() {
system.out.println("我飞了~~~" );
}
}
);
public static void showFly(FlyAble f) {
f.fly();
}
}
}
静态内部类:也是定义在一个类里面的类,用static修饰,不需要依赖于外部类,并且不能使用外部类的非static成员访问,因为外部类的非static成员必须依附于具体的对象。
外部类名.内部类名 对象名= 外部类对象名.new 内部类名();
接口
包含抽象方法(JDK7及以前),默认方法和静态方法(JDK8),私有方法(JDK9,供默认方法和静态方法调用)。
① 类对接口可以多实现,使用implements关键字;接口对接口可以多继承,使用extends关键字;
多实现中,实现类必须重写所有的抽象方法,如果抽象方法有重名,只需要重写一次;
② 继承了接口的默认方法,可以使用子类的对象直接调用,也可以重写默认方法;
多实现中,如果默认方法有重名,必须重写一次;
子接口重写默认方法时,default关键字可以保留。
子类重写默认方法时,default关键字不可以保留。
③ 静态与.class文件相关,只能使用接口名调用,不可以通过实现类的类名或者实现类的对象调用;
④ 私有方法的调用:
a.私有方法:只有默认方法可以调用;
b.私有静态方法:默认方法和静态方法都可以调用;
⑤ 接口中,存在同名的静态方法并不会冲突,原因是只能通过各自接口名访问静态方法;
⑥ 优先级的问题:当一个类,既继承一个父类,又实现若干个接口时,父类中的成员方法与接口中的默认方法重名,子类就近选择执行父类的成员方法;
定义静态常量,其值不可以改变,默认使用public static final修饰;
⑧ 接口中,没有构造方法,不能创建对象;
⑨ 接口中,没有静态代码块。
抽象类和接口的对比
a) 语法层面的区别
参数 | 抽象类 | 接口 |
默认的方法实现 | 它可以有默认的方法实现 | 所有的方法都是抽象的。 |
实现 | 子类使用extends关键字来继承抽象类。如果子类不是抽象类的话,它需要提供抽象类中所有声明的方法的实现。 | 子类使用关键字implements来实现接口。它需要提供接口中所有声明的方法的实现 |
构造器 | 抽象类有构造器,用于子类初始化 | 接口不能有构造器 |
普通类 | 除了不能实例化抽象类之外,它和普通Java类没有任何区别 | 接口是完全不同的类型 |
访问修饰符 | 抽象方法可以有public、protected和default这些修饰符 | 接口方法默认修饰符是public。不可以使用其它修饰符。 |
main方法 | 抽象方法可以有main方法并且我们可以运行它 | 接口没有main方法,因此我们不能运行它。 |
多继承 | 抽象方法可以继承一个类和实现多个接口 | 接口只可以继承一个或多个其它接口 |
速度 | 它比接口速度要快 | 接口是稍微有点慢的,因为它需要时间去寻找在类中实现的方法。 |
添加新方法 | 如果你往抽象类中添加新的方法,你可以给它提供默认的实现。因此你不需要改变你现在的代码。 | 如果你往接口中添加方法,那么你必须改变实现该接口的类。 |
b) 设计层面上的区别
抽象类是对一种事物的抽象,即对类抽象,而接口是对行为的抽象。抽象类是对整个类整体进行抽象,包括属性、行为,但是接口却是对类局部(行为)进行抽象。举个简单的例子,飞机和鸟是不同类的事物,但是它们都有一个共性,就是都会飞。那么在设计的时候,可以将飞机设计为一个类Airplane,将鸟设计为一个类Bird,但是不能将 飞行 这个特性也设计为类,因此它只是一个行为特性,并不是对一类事物的抽象描述。此时可以将 飞行 设计为一个接口Fly,包含方法fly( ),然后Airplane和Bird分别根据自己的需要实现Fly这个接口。然后至于有不同种类的飞机,比如战斗机、民用飞机等直接继承Airplane即可,对于鸟也是类似的,不同种类的鸟直接继承Bird类即可。从这里可以看出,继承是一个 "是不是"的关系,而 接口 实现则是 "有没有"的关系。如果一个类继承了某个抽象类,则子类必定是抽象类的种类,而接口实现则是有没有、具备不具备的关系,比如鸟是否能飞(或者是否具备飞行这个特点),能飞行则可以实现这个接口,不能飞行就不实现这个接口。
设计层面不同,抽象类作为很多子类的父类,它是一种模板式设计。而接口是一种行为规范,它是一种辐射式设计。什么是模板式设计?最简单例子,大家都用过ppt里面的模板,如果用模板A设计了ppt B和ppt C,ppt B和ppt C公共的部分就是模板A了,如果它们的公共部分需要改动,则只需要改动模板A就可以了,不需要重新对ppt B和ppt C进行改动。而辐射式设计,比如某个电梯都装了某种报警器,一旦要更新报警器,就必须全部更新。也就是说对于抽象类,如果需要添加新的方法,可以直接在抽象类中添加具体的实现,子类可以不进行变更;而对于接口则不行,如果接口进行了变更,则所有实现这个接口的类都必须进行相应的改动。