首页 > 技术文章 > Java抽象类、接口和内部类

jmwm 2017-06-14 23:44 原文

1.抽象方法、抽象类

1)抽象方法:

  • 由abstract修饰
  • 只有方法的定义,没有方法的具体实现(连{}都没有)

  由abstract修饰的方法为抽象方法,抽象方法只有方法的定义,没有方法体实现,用一个分号结尾。即:方法五要素中,抽象方法缺少了一个要素(方法体),也可以将抽象方法理解为不完整的方法。

 

2)抽象类:

  • 由abstract修饰
  • 包含抽象方法的类必须是抽象类,不包含抽象方法的类也可以声明抽象类(意义不大)
  • 抽象类不能被实例化
  • 抽象类是需要被继承的
      • 子类需要重写抽象类的所有抽象方法---常用
      • 子类也可以声明为抽象类---不常用
  • 抽象类的意义:
      • 封装子类所共有的属性和行为---代码复用
      • 为所有子类提供一种统一的类型---向上造型
      • 可以包含抽象方法,为所有子类提供了统一的入口,子类的实现不同,但入口是一致的

  由abstract修饰的类为抽象类,抽象类是不能实例化对象的,而一个类不能实例化是没有意义的,所以需要定义类来继承抽象类,它的子类必须重写所有的抽象方法,除非该类也声明为抽象类。

abstract class Foo{
	private double c;
	public Foo(double c){  //没什么意义,需要被子类重写
		this.c = c;
	}
	public abstract double area();  //抽象方法,没有方法体,大括号也不存在
}
class Sub extends Foo{
	public Sub(double c) {  //需要重写构造方法
		super(c);
	}
	public double area(){
		return 0.0;
	}
}
abstract class Sub2 extends Foo{  //抽象方法继承抽象方法
	public Sub2(double c) {  //需要重写构造方法,不需要重写抽象方法
		super(c);
	}
}

  

3)抽象类不可以被实例化

Foo f = new Foo();  //编译错误,抽象类不能被实例化

  即使一个类中没有抽象方法,也可以将其定义为抽象类,同样,该类不可以实例化。

  注意:abstract和final关键字不能同时修饰一个类,因为final使得类不可继承,而abstract修饰的类如果不可以继承将没有任何意义。

 

4)继承抽象类

  一个类继承抽象类后,必须实现其抽象方法,不同的子类可以有不同的实现。

abstract class Foo{
	private double c;
	public Foo(double c){  //没什么意义,需要被子类重写
		this.c = c;
	}
	public abstract String sayHi();
}
class Sub extends Foo{
	public Sub(double c) {  //需要重写构造方法
		super(c);
	}
	public String sayHi(){
		return "Hello";
	}
}

  

4)抽象类的意义

  • 为其子类提供一个公共的类型(父类引用指向子类对象,即向上造型)
  • 封装子类中的重复内容(成员变量和方法)
  • 定义有抽象方法,子类虽然有不同的实现,但该方法的定义是一致的(子类需要实现此抽象方法)

案例1:抽象类演示

public class AbstractDemo {
	public static void main(String[] args){
//		Shape s = new Shape(); //编译错误,抽象类不能被实例化
		Shape[] shape = new Shape[4];  //创建Shape数组对象
		shape[0] = new Circle(1);  //向上造型
		shape[1] = new Circle(2);
		shape[2] = new Square(1);
		shape[3] = new Square(2);
		maxArea(shape);  //调用求最大面积方法
	}

	private static void maxArea(Shape[] shape) {
		double max = shape[0].area();
		int maxIndex = 0; //最大面积下标
		for(int i=0;i<shape.length;i++){
			double area = shape[i].area();
			if(area>max){
				max = area;
				maxIndex=i;
			}
		}
		System.out.println("最大面积为:"+max+"所在下标为:"+maxIndex);
	}
}
abstract class Shape{ //抽象类---不完整的类
	protected double c; //周长
	public abstract double area(); //抽象方法---不完整
}
class Circle extends Shape{
	public Circle(double c){
		this.c = c;
	}
	public double area(){  //重写抽象方法---变不完整为完整
		return 0.0796*c*c;
	}
}
class Square extends Shape{
	public Square(double c){
		this.c = c;
	}
	public double area(){
		return 0.0625*c*c;
	}
}

  

 

2. 接口

  • 是一个标准、规范,遵循了这个标准就能干某件事
  • 是一种数据类型(引用类型)
  • 由interface定义,只能包含常量和抽象方法,方法默认由public abstract修饰
  • 接口不能被实例化
  • 接口是需要被实现的,通过implements关键字实现,实现类:必须重写接口中的所有抽象方法
  • 一个类可以实现多个接口,逗号分隔,若又继承又实现,必须先继承后实现
  • 接口可以继承一个或多个接口,逗号分隔(extends)

1)定义接口

  接口可以看成是特殊的抽象类。即:只包含抽象方法和常量的抽象类。通过interface关键字来定义接口。

interface Demo{
	public static int x = 100;
	public int y = 50;
	double area();   //默认会加上public abstract修饰
	public abstract void test();
}

  

2)实现接口

  与继承不同,一个类可以实现多个接口,实现的接口直接用逗号分隔,该类需要实现这些接口中定义的所有方法。通过implements关键字实现接口。接口可以作为一种类型声明变量,一个接口类型的变量可以引用实现了该接口的类的对象,通过该变量可以调用该接口中定义的方法(具体的实现类提供了方法的实现)。一个接口类型变量,引用了子类的对象,调用时,调用的是子类对象的具体的实现。

public class Test {
	public static void main(String[] args){
		Demo d = new Aoo();   //一个接口类型变量,引用了子类的对象,调用时,调用的是子类对象的具体的实现
		d.test();   //This is Aoo
	}
}

interface Demo{
	public static int x = 100;
	public int y = 50;
	double area();   //默认会加上public abstract修饰
	public abstract void test();
}
class Aoo implements Demo{
	public double area(){
		return 0.0;
	}
	public void test(){
		System.out.println("This is Aoo");
	}
}

  

3)接口的继承

  接口间可以存在继承关系,一个接口可以通过extends关键字继承另外一个接口,子接口继承了父接口中定义的所有方法。

interface Foo{
	public void funFoo();
}
interface Goo{
	public void funGoo();
}
interface Hoo extends Foo,Goo{  //接口可以继承一个或多个接口,逗号分隔
	public void funHoo();
}
interface Ioo{
	void funIoo();
}

class Joo implements Hoo,Ioo{  //一个类可以继承多个接口
	public void funFoo(){}  //重写父类的父类的抽象方法
	public void funGoo(){}  //重写父类的父类的抽象方法
	public void funHoo(){}  //重写父类的抽象方法
	public void funIoo(){}   //重写父类的抽象方法
}

  

4)接口和抽象类的区别

  • 一个类只能继承一个抽象类,但可以实现多个接口
  • 抽象类中可以包含抽象方法和非抽象方法,而接口中的所有方法均为抽象的
  • 子类继承抽象类必须实现抽象类中所有抽象方法,否则子类也必须是抽象类。而子类实现接口则必须实现接口中所有的抽象方法。

 

案例2:接口的演示

public class InterfaceDemo {
	public static void main(String[] args){
//		Inter6 o1= new Inter6();   //编译错误,接口不能被实例化
		Inter6 o2 = new Moo();  //向上造型
		Inter5 o3 = new Moo();
	}
}
//演示接口语法
interface Inter1{
	public static final int NUM = 5;
	public abstract void show();
	double PI = 3.14; //默认public static final修饰
	void say();  //默认public abstract修饰
//	int count; //编译错误,常量必须声明同时初始化
//	void test(){}  //编译错误,抽象方法没有方法体
}

//演示接口实现
interface Inter2{
	void show();
	void say();
}
class Joo implements Inter2{
	public void show(){}
	public void say(){}
}

//演示接口的多实现、继承
interface Inter3{
	void show();
}
interface Inter4{
	void say();
}
abstract class Koo{
	abstract void test();
}
class Loo extends Koo implements Inter3,Inter4{
	public void show(){}
	public void say(){}
	void test(){}
}

//演示接口继承接口
interface Inter5{
	void show();
}
interface Inter6 extends Inter5{
	void say();
}
class Moo implements Inter6{
	public void show(){}
	public void say(){}
}

  

 3. 多态

1)意义:

  1.1)同一类型的引用指向不同的对象时,有不同的实现:行为的多态:cut()、run()

  1.2)同一个对象被造型为不同的类型时,有不同的功能:对象的多态:水

2)向上造型:

  2.1)父类型的引用指向子类的对象

  2.2)能造型成为的类有:父类、所实现的接口

  2.3)能点出来什么,看引用的类型

3)强制类型转换,成功的条件有两种:

  3.1)引用所指向的对象,就是该子类型

  3.2)引用所指向的对象,实现了该接口

4)若不满足以上两个条件,则发生ClassCastException类型转换异常

  建议:在强转之前先通过instanceof判断引用指向的对象是否是该类型

 

4. 成员内部类:应用几率不大

1)类中套类,里面的成为Inner,外面的成为Outer

2)内部类通常只服务于外部类,对外不具备可见性

3)内部类对象通常是在外部类中创建的

4)内部类中可以直接访问外部类的成员,包括私有的

   一般情况下,Inner对象会在Outer对象中创建(构造方法或其他方法),Inner对象中会有一个隐式的引用指向创建它的Outer类对象,外部类名.this.

 

5. 匿名内部类

1)若想创建一个类(子类)的对象,并且对象只被创建一次,此时该类不必命名,称之为匿名内部类

2)匿名内部类中访问外部的变量,该变量必须是final的

 

常见面试题:内部类有独立的.class吗? 答案:有

 

5. 面向对象三大特征

1)封装:

  1.1)类:封装的是对象的属性和行为

  1.2)方法:封装的是业务逻辑功能的实现

  1.3)访问控制修饰符:封装的是访问的权限

2)继承:

  2.1)作用:代码复用,通过extends来实现继承

  2.2)父类/基类:共有的

    子类/派生类:特有的

  3.3)单一继承、多接口实现

3)多态:

  3.1)意义:行为的多态、对象的多态 

  3.2)向上造型、强制类型转换(instanceof)

  3.3)多态的表现形式:重写、重载

 

设计规则:

1)将所有子类共有的属性和行为,抽象到父类中

2)所有子类行为都一样,设计为普通方法

  所有子类行为不一样,设计为抽象方法

3)将部分子类共有的行为,抽象到接口中

  符合既是也是的原则,使用接口

  接口是对继承的单根性的扩展——实现多继承

 

案例3:多态的演示

public class MultiTypeDemo {
	public static void main(String[] args){
		Aoo o1 = new Boo();  //向上造型(自动类型转换)
		Boo o2 = (Boo)o1;     //o1指向的对象就是Boo类型(强制类型转换)
		Inter o3 = (Inter)o1;    //o1指向的对象实现了Inter接口(强制类型转换)
//		Coo o4 = (Coo)o1;    //java.lang.ClassCastException:类型转换异常
		if(o1 instanceof Coo){  //强转建议使用instanceof判断
			Coo o5 = (Coo)o1;
		}
	}
}

interface Inter{}
class Aoo{}
class Boo extends Aoo implements Inter{}
class Coo extends Aoo{}

 

案例4:多态的演示2

public class UnionPayTest {
	public static void main(String[] args) {
		ABCATM atm = new ABCATM(); //atm机对象
		UnionPay card = new ABCImpl(); //银联卡
		atm.insertCard(card); //插卡
		atm.payTelBill(); //支付电话费
	}
}
class ABCATM{ //农行ATM机类
	private UnionPay card; //银联卡
	public void insertCard(UnionPay card){ //插卡
		this.card = card;
	}
	public void payTelBill(){ //支付电话费按钮
		if(card instanceof ABC){ //是农行卡
			ABC abcCard = (ABC)card; //强转为农行卡
			abcCard.payTelBill("12345679845", 500);
		}else{ //不是农行卡
			System.out.println("不是农行卡,不能支付电话费");
		}
	}
}

interface UnionPay{ //银联接口
	public double getBalance(); //查询余额
	public boolean drawMoney(double number); //取款
	public boolean checkPwd(String input); //验证密码
}
interface ICBC extends UnionPay{ //工行接口
	public void payOnline(double number); //在线支付
}
interface ABC extends UnionPay{ //农行接口
	public boolean payTelBill(String phoneNum,double sum); //支付电话费
}

class ICBCImpl implements ICBC{ //工行卡
	public double getBalance(){return 0.0;}
	public boolean drawMoney(double number){return true;}
	public boolean checkPwd(String input){return true;}
	public void payOnline(double number){}
}
class ABCImpl implements ABC{ //农行卡
	public double getBalance(){return 0.0;}
	public boolean drawMoney(double number){return true;}
	public boolean checkPwd(String input){return true;}
	public boolean payTelBill(String phoneNum,double sum){
		System.out.println("支付电话费成功");
		return true;
	}
}

 

案例5:成员内部类的演示

public class InnerClassDemo {
	public static void main(String[] args){
		Outer outer = new Outer(100);
		outer.printTime();
		//		Inner g = new Inner();   //编译错误
	}
}

class Outer{
	private int time;
	private Inner inner;
	Outer(int time){
		this.time = time;
		inner = new Inner();
		inner.timeInc();
	}
	public void printTime(){
		System.out.println(Outer.this.time);   //101
		System.out.println(++time);  //102
	}
	class Inner{
		public void timeInc(){
			time++;
		}
	}
}

  在Outer构造方法中创建的Inner对象会有一个隐式的引用指向创建它的Outer对象,调用Inner对象的timeInc方法,即会对Outer的time属性进行操作。

 

 

案例6:匿名内部类的演示

 

public class NstInnerClassDemo {
	public static void main(String[] args){
		/*
		 * 1)创建了Inter2的一个子类,但是没有名字
		 * 2)为该子类创建了一个对象,叫o1
		 * 3)大括号中的为子类的类体
		 */
		Inter2 o1 = new Inter2(){	};
		
		/*
		 * 1)创建了Inter2的一个子类,但是没有名字
		 * 2)为该子类创建了一个对象,叫o2
		 * 3)大括号中的为子类的类体
		 */
		Inter2 o2 = new Inter2(){ };
		
		int num = 5;
		/*
		 * 1)创建了Inter1的一个子类,但是没有名字
		 * 2)为该子类创建了一个对象,叫o3
		 * 3)大括号中的为子类的类体
		 */
		Inter1 o3 = new Inter1(){
			public void show(){
				System.out.println("show");
				System.out.println(num);
			}
		};
		o3.show();
	}
}

interface Inter1{
	void show();
}
interface Inter2{ }

 

  

 

 

 

 

推荐阅读