抽象类和接口
前言
更新,更新,更新!!!
来吧,展示,希望有所帮助!!!
(本文如有任何错误的言论,还请路过的大佬不吝赐教,感激不尽!!!)
一、抽象类
使用Java语言解决实际问题的时候,一般将父类定义为抽象类,需要使用这个父类进行继承与多态处理。回想一下继承和多态的原理,越是往上的类就越抽象。在多态机制中,并不需要将父类初始化对象,我们需要的只是子类对象,所以在Java语言中设置抽象类不可实例化对象,但是它的子类却可以。
语法:
public abstract class Test{
abstract void testAbstract(); //定义抽象方法
}
使用abstract关键字定义的类称为抽象类,而使用该关键字定义的方法则称之为抽象方法。抽象方法没有方法体,这个方法本身没有任何意义,除非他被重写,而承载这个抽象方法抽象类必须被继承。实际上抽象类除了被继承之外没有任何意义。
反过来讲,如果声明一个抽象方法,就必须将承载这个抽象方法的类定义为抽象类,不可能在非抽象类中获取抽象方法。也就是说,只要某一个类中有一个抽象方法,那么这个类就被标记为抽象类。
抽象类被继承后需要实现其中所有的抽象方法,也就是保证相同的方法名称、参数列表和相同返回值类型创建出非抽象方法,当然也可以是抽象方法。
代码示例:
/**
* 抽象类
*/
public abstract class Employee {
private String name;
private String address;
private int number = 0;
/**
* 有参构造
* @param name
* @param address
* @param number
*/
public Employee(String name, String address, int number) {
System.out.println("Constructing an Employee");
this.name = name;
this.address = address;
this.number = number;
}
/**
* 无参
* @return
*/
public double computePay() {
System.out.println("Inside Employee computePay");
return 0.0;
}
public void mailCheck() {
System.out.println("Mailing a check to " + this.name
+ " " + this.address);
}
public String toString() {
return name + " " + address + " " + number;
}
/**
* get和set方法
* @return
*/
public String getName() {
return name;
}
public String getAddress() {
return address;
}
public void setAddress(String newAddress) {
address = newAddress;
}
public int getNumber() {
return number;
}
}
这样看上起,这个抽象类好像没什么太多的变化。尽管它是一个抽象类,同样也有成员变量、成员方法和构造方法。但是我们不能实例化一个Employee对象,不过可以继承它。
我们实例化一个 Salary 类对象,该对象将从 Employee 类继承 7 个成员方法,且通过该方法可以设置或获取三个成员变量。
继承类:
/**
* Salary继承Employee类
*/
public class Salary extends Employee {
private double salary; //年薪
public Salary(String name, String address, int number, double salary) {
super(name, address, number);
setSalary(salary);
}
public void mailCheck() {
System.out.println("Within mailCheck of Salary class ");
System.out.println("Mailing check to " + getName()
+ " with salary " + salary);
}
public double getSalary() {
return salary;
}
public void setSalary(double newSalary) {
if (newSalary >= 0.0) {
salary = newSalary;
}
}
public double computePay() {
System.out.println("Computing salary pay for " + getName());
return salary / 52;
}
}
演示类:
/**
* 演示类
*/
public class AbstractDemo {
public static void main(String[] args) {
Salary s = new Salary("Mohd Mohtashim", "Ambehta, UP", 3, 3600.00);
Employee e = new Salary("John Adams", "Boston, MA", 2, 2400.00);
System.out.println("Call mailCheck using Salary reference --");
s.mailCheck();
System.out.println("\n Call mailCheck using Employee reference--");
e.mailCheck();
}
}
运行结果:
注意:
- 抽象类不能被实例化(很容易犯的错),如果被实例化,就会报错,编译无法通过。只有抽象类的非抽象子类可以创建对象。
- 抽象类中不一定包含抽象方法,但是有抽象方法的类必定是抽象类。
- 抽象类中的抽象方法只是声明,不包含方法体,就是不给出方法的具体实现也就是方法的具体功能。
- 构造方法,类方法(用 static 修饰的方法)不能声明为抽象方法。
- 抽象类的子类必须给出抽象类中的抽象方法的具体实现,除非该子类也是抽象类。
二、接口
2.1什么是接口
在Java中,接口是一个抽象类型,是抽象方法的集合,一个类通过继承接口的方式来继承接口的抽象方法。接口不是类,它是抽象类的延伸,包含要实现的方法,可以将它看做是纯粹的抽象类。接口无法被实例化,但是可以被实现。一个实现接口的类必须实现接口内所描述的全部方法,否则就必须声明为抽象类。
2.2接口的特性
- 一个接口可以有多个方法,但是接口没有构造方法。
- 接口不能实例化对象
- 接口中的 所有方法必须是抽象方法
- 接口中不能包含成员变量,接口中可以含有变量,但是接口中的变量会被隐式的指定为 public 、static、 final 变量,并且只能是 public,不能用 private
- 接口中每一个方法也是隐式抽象的,接口中的方法会被隐式的指定为 public 、abstract
- 接口支持多继承
- 接口中的方法是不能在接口中实现的,只能由实现接口的类来实现接口中的方法。
- 接口中的方法都是公有的
2.3声明接口
语法:
[修饰符] interface 接口名 extends 父接口1,父接口2...{
//定义常量
//定义方法
}
class 类名extends 父类名 implements 接口1,接口2{
//类成员
}
注意:
- 接口的命名规则和类相同
- 接口中定义常量必须在定义时给定初始值
- 接口中所有的方法都是抽象方法
- 接口中不能有构造方法,接口不能被实例化
- 接口之间可以通过extends关键字实现继承关系,一个接口可以继承多个接口,但是接口不能继承类
- 接口的实现类必须实现接口中的全部方法,否则必须定义为抽象类
- 当类在继承父类的同时又实现了多个接口时,extends关键字必须位于implements之前。
代码示例:
/**
* USB接口
*/
public interface USBInterface {
//USB接口提供服务
void service();
}
/**
* U盘
*/
public class UDisk implements USBInterface{
public void service(){
System.out.println("连接USB接口,开始传输数据!!!");
}
}
/**
* USB充电宝
*/
public class USBCharge implements USBInterface{
public void service(){
System.out.println("连接USB,获得电流,开始充电!!!");
}
}
/**
* 测试接口类
*/
public class TestInterface {
public static void main(String[] args) {
//U盘
USBInterface uDisk = new UDisk();
uDisk.service();
//充电宝
USBInterface usbCharge = new USBCharge();
usbCharge.service();
}
}
运行结果:
以上一个简单的接口实现的例子,下面我们一起来看一下一个相对复杂一点的。
下面的代码是使用接口实现打印机功能!!!
代码示例:
/**
* 墨盒接口
*/
public interface InkBox {
/**
* 得到墨盒颜色
* @return
*/
public String getColor();
}
/**
* 纸张接口
*/
public interface Paper {
/**
* 得到纸张大小
* @return
*/
public String getSize();
}
/**
* 打印机类
*/
public class Printer {
InkBox inkBox; //墨盒
Paper paper; //纸张
/**
* 设置打印机墨盒
*
* @param inkBox
*/
public void setInkBox(InkBox inkBox) {
this.inkBox = inkBox;
}
/**
* 设置打印纸张
*
* @param paper
*/
public void setPaper(Paper paper) {
this.paper = paper;
}
/**
* 使用墨盒在打印机上打印
*/
public void print() {
System.out.println("使用" + inkBox.getColor() +
"墨盒在" + paper.getSize() + "纸张上打印");
}
}
/**
* 彩色墨盒
*/
public class ColorInkBox implements InkBox{
public String getColor(){
return "彩色";
}
}
/**
* 黑白墨盒
*/
public class GrayInkBox implements InkBox{
public String getColor(){
return "黑白";
}
}
/**
* A4纸类
*/
public class A4Paper implements Paper{
public String getSize(){
return "A4";
}
}
/**
* B5纸类
*/
public class B5Paper implements Paper{
public String getSize(){
return "B5";
}
}
/**
* 测试打印机
*/
public class TestPrinter {
public static void main(String[] args) {
//1.定义打印机
InkBox inkBox = null;
Paper paper = null;
Printer printer = new Printer();
//2.使用黑白墨盒在A4纸上打印
inkBox = new GrayInkBox();
paper = new A4Paper();
printer.setInkBox(inkBox);
printer.setPaper(paper);
printer.print();
//3.使用彩色墨盒在B5纸上打印
inkBox = new ColorInkBox();
paper = new B5Paper();
printer.setInkBox(inkBox);
printer.setPaper(paper);
printer.print();
//4.使用彩色墨盒在A4纸上打印
paper = new A4Paper();
printer.setPaper(paper);
printer.print();
}
}
运行结果:
本期文章就到这里,希望有所帮助!!!
关注我,持续更新!!!