Java学习笔记004——接口、克隆、回调、内部类

接口

1、接口不是类,而是对类的一组需求描述,这些类要遵从接口描述的统一格式进行定义

2、定义接口,如:

public interface Abc{

int aaa(Object other)

}

3、接口中的所有方法自动地属于public,声明在接口中的内部类自动成为staticpublic

4、接口不能含有实例域,也不能在接口中实现方法,提供实例域和方法实现的任务应该由实现接口的那个类完成

5、为了让类实现一个接口,需要如下两个步骤:

① 将类声明为实现给定的接口,需要使用implements关键字,如:class Employee implements Compable

② 对接口中的所有方法进行定义

6、在接口声明中的方法不需要声明为public,但在实现接口时必须把方法声明为public,否则编译器会发出警告

7、如果存在一种通用算法,它能够对两个不同的子类对象进行比较,在应该在超类中提供一个compareTo方法,并将这个方法声明为final

8、接口的特性

① 接口不是类,不能使用new运算符实例化一个接口,但能够声明接口的变量,如:Comparable x;

② 接口变量必须引用实现了接口的类对象:x = new Employee(...)

③ 可以使用instanceof检查一个对象是否实现了某个特定的接口,如:if(anObject instanceof Comparable){...}

④ 接口可以被扩展,如:有一个成为Moveable的接口:

public interface Moveable{

void move(double x, double y);

}  //然后可以以它为基础扩展一个叫做Powered的接口:

public interface Powered entends Moveable{

public milesPerGallon();

}

⑤ 在接口中不能包含实例域或静态方法,但却可以包含常量,如:

public interface Powered entends Moveable{

double milesPerGallon();

double SPEED_LIMIT = 95;

}  //类中的域将被自动设定为public static final

——有些程序员为了提高清晰度的考虑,将接口方法标记为public,将域标记为public static final,虽然这在Java中是允许的,但Java语言规范却建议不要书写这些多余的关键字

⑥ 实现多个接口,如:class Employee implements Clonealbe, Comparable,使用逗号将实现的对个接口隔开

克隆

1、clone方法是Object类的一个proteced方法,该方法只能实现浅拷贝,若浅拷贝对象共享的子对象是不可变的,那就不会出现什么问题,但如果需要拷贝的对象里有可变的子对象,那就需要编写该类自己的clone方法

2、一个类需要实现了Cloneable接口才能实现克隆操作,否则编译不通过,可以使用if(obj instanceof Cloneable)...来检查是否实现了该接口,该接口用来表明这个类需要拥有克隆的能力

3、即使clone的默认实现(浅拷贝)能够满足需求,也应该实现Cloneable接口,将clone重定义为public,并调用super.clone(),如:

class Employee implements Cloneable{

public Employee clone() throws CloneNotSupportedException{

Employee cloned = (Employee)super.clone();

}

}

4、建立深拷贝clone方法的示例:

class Employee implements Cloneable{

public Employee clone() throws CloneNotSupportedException{

Employee cloned = (Employee)super.clone();

cloned.hireDay = (Date)hireDay.clone();

return cloned;

}

}

回调

1、回调是一种程序设计模式,在这种模式中可以指出某个特定事件发生时应该采取的动作

2、java.swing包中有一个Timer类可以使用它在到达个顶的时间间隔时发出通告,需要设置一个时间间隔,将某一个类的对象传递给定时器可以设置定时器定时进行的操作,这要求传递的对象所属的类实现了java.awt.event包的ActionListener接口,该接口如下:

public interface ActionListener{

void actionPerformed(ActionEvent event);

}  //当到达指定的时间间隔时,定时器就调用actionPerformed方法

3、回调的示例:

import java.awt.event.*;

import java.util.*;

import javax.swing.*;

import javax.swing.Timer;

public class TimerTest{

public static void main(String[] args){

ActionListener listener = new TimePrinter();

Timer t = new Timer(1000, listener);

t.start();

JOptionPane.showMessageDialog(null, "quit program?");

System.exit(0);

}

}

class TimePrinter implements ActionListener{

public void actionPerformed(ActionEvent event){

Date now = new Date();

System.out.println("At the time is " + now);

}

}

内部类

1、内部类是定义在另一个类中的类,使用内部类的主要原因有:

① 内部类方法可以访问该类定义所在的作用域中的数据,包括私有的数据

② 内部类可以对同一个包中的其他类隐藏起来

③ 当先要定义一个回调函数且不想编写大量代码时,使用匿名内部类比较便捷

2、内部类既可以访问自身的数据域,也可以访问创建它的外围类对象的数据域

3、若在A类中定义了内部类B,则只有A类的方法才能够生成B对象,只有内部类是私有类,而常规类只可以具有包可见性或者公有可见性

4、表达式OuterClass.this表示外围类引用,如A类是外围类,在内部类B里要使用A类的成员变量可以这样:A.this.xxx,可以使用下列语法格式来编写内部对象的构造器:outerObject.new InnerClass(构造方法变量列表),如this.new B();这里的this限定词是多余的,不过可以通过显式地命名将外围类引用设置为其他的对象,如果B是A的内部类,对于任意的A对象都可以构造一个B对象:

A abc0 = new A();

A.B abc1 = abc0.new B();

5、局部内部类

即在一个方法中定义的类,局部类不能用public或private进行声明,它的作用域被限定在声明这个局部类的块中,其优点有:

① 可以对外部世界外圈地隐蔽起来

② 它不仅可以访问包含它们的外部类,还可以访问被声明为final的局部变量

final关键字可以应用于局部变量、实例变量和静态变量中,在所有这些情况下,它们的含义都是:在创建这个变量后,只能够为之赋值一次,此后,再也不能修改它的值了,在定义final变量的时候,不必进行初始化

6、匿名内部类

① 假如只创建某个类的一个对象,就不必命名了,这种类叫做匿名内部类,如:

public void start(int interval, final boolean beep){

ActionListener listener = new ActionListener(){

public void actionPerformed(ActionEvent event){

Date now = new Date();

System.out.println("At the tone, the time is " + now);

if(beep) Toolkit.getDefaultToolkit().beep();

}

};

Timer t = new Timer(interval, listener);

t.start();

}  //创建一个实现ActionListener接口的新对象,需要实现的方法actionPerformed定义在括号{}内

② 用于构造对象的任何参数都要被放在超类/接口/类名后面的括号()内,通常的语法格式是:

new SuperType(构造方法变量列表){

inner class methods and data

}

③ 由于匿名类没有类名,所以匿名类不能有构造器,取而代之的是将构造器参数传递给超类构造器,尤其是在内部类实现接口的时候不能有任何构造参数,还要加上一组(),如果构造参数的闭圆括号后跟一个花括号,则表示定义的是匿名内部类

7、静态内部类

① 有时候,使用内部类只是为了把一个类隐藏在另外一个类的内部,并不需要内部类引用外围类对象,为此可以将内部类声明为static,以便取消产生的引用

② 只有内部类可以声明为static,静态内部类的对象除了没有对生成它的外围类对象的引用特权外,与其他所有内部类完全一样

③ 注:在内部类不需要访问外围类对象的时候,应该使用静态内部类;声明在接口中的内部类自动成为publicstatic

原文地址:https://www.cnblogs.com/oushihua/p/2933653.html