zoukankan      html  css  js  c++  java
  • Java抽象类、接口和内部类

    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{ }
    

      

     

     

     

  • 相关阅读:
    最近想读的书想做的事
    syslogng 正确配置udp接受端口
    (转)iPhone重绘机制drawRect
    .Net工具整理
    用系统的网格还是用自己的网格
    将主关键字加强为适应多个
    onblur与onbeforedeactivate的区别
    模式窗口中无法转向
    实现获取客户端的MAC地址(3)
    onkeypress、onkeydown、onkeyup三事件的区别
  • 原文地址:https://www.cnblogs.com/jmwm/p/7011610.html
Copyright © 2011-2022 走看看