《Java核心技术 卷I》学习笔记24:接口

本文详细介绍了Java中的接口概念,包括接口的定义、属性、静态和私有方法、默认方法、接口与回调、Comparator接口的使用,以及对象克隆与接口的关系。接口作为类行为规范的定义,使得类能实现多个接口,提供了代码的灵活性和可扩展性。Java 8引入的默认方法允许接口演进,而静态方法和私有方法的添加则进一步增强了接口的功能。此外,通过Comparator接口可以自定义对象的比较方式,实现灵活的排序。对象克隆时,若实现Cloneable接口,可以创建对象的副本,但需注意深拷贝和浅拷贝的区别。
摘要由CSDN通过智能技术生成

1 接口的概念

接口不是类,而是对希望符合这个接口的类的一组需求。接口用来描述类应该做什么,而不指定它们具体应该如何做,由类具体实现接口中的方法。一个类可以实现一个或多个接口。

使用关键字interface定义接口。例如,Comparable接口的代码如下:

public interface Comparable<T>
{
	int compareTo(T other);
}

接口中的所有方法都自动是public方法。因此,在接口中声明方法时,不必提供关键字public

接口中可以包含多个方法,也可以定义常量,但不能有实例字段。提供实例字段和方法实现的任务由实现接口的那个类来完成。

类实现接口时,要在类定义处使用关键字implements指定要实现的接口,并实现接口中的方法。例如,Employee类实现Comparable接口的代码如下:

public class Employee implements Comparable<Employee>
{
	private String name;
	private double salary;

	public Employee(String name, double salary)
	{
		this.name = name;
		this.salary = salary;
	}

	public String getName()
	{
		return name;
	}

	public double getSalary()
	{
		return salary;
	}

	public void raiseSalary(double byPercent)
	{
		double raise = salary * byPercent / 100;
		salary += raise;
	}

	public int compareTo(Employee other) // 实现接口中的方法
	{
		return Double.compare(salary, other.salary);
	}
}

这里使用了Double类中的compare方法,它的签名如下:

static int compare(double x, double y)
	// 如果 x < y 返回一个负数,如果 x==y 返回0,如果 x > y 返回一个正整数

Arrays类中的sort方法可以对对象数组进行排序,但要求对象所属的类必须实现Comparable接口。这是因为不同的类有不同的比较方式,必须向sort方法提供对象的比较方式。上面已经让Employee类实现了Comparable接口,就可以使用sort方法对Employee数组进行排序,例如:

Employee staff = new Employee[3];
staff[0] = new Employee("Harry", 35000);
staff[1] = new Employee("Carl", 75000);
staff[2] = new Employee("Tony", 38000);
Arrays.sort(staff);

2 接口的属性

不能使用new运算符实例化一个接口,但可以声明接口的变量,接口变量必须引用实现了这个接口的类对象。例如:

Comparable x = new Employee("Harry", 35000);

可以使用instanceof检查一个对象是否实现了某个特定的接口。例如:

if (obj instanceof Comparable) { ... }

接口也可以像类一样继承。例如:

public interface Moveable
{
	void move(double x, double y);
}

public interface Powered extends Moveable
{
	double milesPerGallon();
}

接口中可以包含常量,接口中的字段自动被设置为public static final,声明字段时可以省略这些关键字。

接口与抽象类很类似,它们都有未实现的方法,由另外一个类提供抽象方法的实现,但它们并不相同。每个类只能有一个超类,但可以实现多个接口,这就为定义类的行为提供了极大的灵活性。

3 静态和私有方法

在Java 8中,允许在接口中增加静态方法。此前,通常将静态方法放在伴随类中,构成成对出现的接口和实用工具类,例如Collection/CollectionsPath/Paths。如果可以将静态方法写在接口中,就不需要提供伴随类了。

在Java 9中,接口中的方法可以是private。接口中的私有方法只能在接口本身的方法中使用,所以它们用法很有限,只能作为接口中其他方法的辅助方法。

4 默认方法

可以为接口方法提供一个默认实现,必须用default修饰符标记这样的方法。例如:

public interface Comparable<T>
{
	default int compareTo(T other) { return 0; }
}

在实现这个接口的类中提供方法的具体实现后,默认实现就会被覆盖。

默认方法可以调用其他方法。例如:

public interface Collection
{
	int size();
	default boolean isEmpty() { return size() == 0; }
}

默认方法的一个重要用法是“接口演化”。假设类C实现了接口F,后来接口F添加了一个新方法,如果新方法不是默认方法,类C中又没有实现这个新方法,就不能通过编译。如果将新方法实现为默认方法,就不需要修改类C的代码。

如果先在一个接口中将一个方法定义为默认方法,然后又在超类或另一个接口中定义同样的方法,会产生二义性。这种情况下的处理规则为:

  1. 超类优先。如果超类提供了一个具体方法,接口中同名且有相同参数类型的默认方法会被忽略。
  2. 接口冲突。如果一个接口提供了一个默认方法,另一个接口提供了一个同名且参数类型(不论是否是默认参数)相同的方法,必须覆盖这个方法解决冲突。

5 接口与回调

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

javax.swing包中有一个Timer类,每经过一个时间间隔就发出一个通知。构造Timer对象时,需要设置一个时间间隔,并告诉定时器经过这个时间间隔时需要做什么。Timer类的部分方法如下:

/* javax.swing.Timer */
Timer(int interval, ActionListener listener) // 构造一个定时器,每经过 interval 毫秒通知 listener一次
void start() // 启动定时器。一旦启动,定时器将调用监听器的 actionPerformed
void stop() // 停止定时器。一旦停止,定时器将不再调用监听器的 actionPerformed

ActionListenerjava.awt.event包中的一个接口,这个接口的定义如下:

public interface ActionListener
{
	void actionPerformed(ActionEvent event);
}

可以定义一个类实现ActionListener接口,向定时器传入这个类的对象。每经过指定的时间间隔,定时器就调用实现的actionPerformed方法。例如:

import java.awt.*;
import java.awt.event.*;
import java.time.*;
import javax.swing.*;

public class TimerTest
{
	public static void main(String[] args)
	{
		TimePrinter listener = new TimePrinter();
		Timer timer = new Timer(1000, listener);
		timer.start();

		JOptionPane.showMessageDialog(null, "Quit program?");
		System.exit(0);
	}
}

class TimerPrinter implements ActionListener
{
	public void actionPerformed(ActionEvent event)
	{
		System.out.println("At the tone, the time is " + Instant.ofEpochMilli(event.getWhen()));
		Toolkit.getDefaultToolkit().beep();
	}
}

上面的例子中用到的方法有:

/* javax.swing.JOptionPane */
static void showMessageDialog(Component parent, Object message)
	// 显示一个包含一条提示信息和 OK 按钮的对话框
	// 对话框位于 parent 组件的中央。如果 parent 为 null,对话框将显示在屏幕中央

/* java.time.Instant */
static Instant ofEpochMilli(long epochMilli)
	// 使用从 1970 年 1 月 1 日 00:00:00 开始的毫秒数获得一个 Instant 实例

/* java.awt.event.ActionEvent */
long getWhen()
	// 返回发生此事件的时间戳,表示为从 1970 年 1 月 1 日 00:00:00 开始到现在的毫秒数

/* java.awt.Toolkit */
static Toolkit getDefaultToolkit()
	// 获得默认的工具箱。工具箱包含有关GUI环境的信息
void beep()
	// 发出一声铃响

6 Comparator接口

Arrays类中的sort方法有另一个版本,通过Comparator接口指定排序方式:

/* java.util.Arrays */
static <T> void sort(T[] a, Comparator<? super T> c)
static <T> void sort(T[] a, int fromIndex, int toIndex, Comparator<? super T> c)

Comparator接口的定义如下:

public interface Comparator<T>
{
	int compare(T o1, T o2);
	// 默认方法在此省略
}

例如,要按长度比较字符串,可以如下定义一个实现Comparator<String>的类:

import java.util.*;

public class ComparatorTest
{
	public static void main(String[] args)
	{
		String[] friends = {"Peter", "Paul", "Mary"};
		Arrays.sort(friends, new LengthComparator());
	}
}

class LengthComparator implements Comparator<String>
{
	public int compare(String o1, String o2)
	{
		return o1.length() - o2.length();
	}
}

7 对象克隆

对象变量直接赋值只是让两个变量引用同一个对象,改变其中一个变量,另一个变量也会受到影响,这并不能达到克隆的目的。如果想让两个变量互不影响,就要使用clone方法。例如:

Employee original = new Employee("John", 50000);
Employee copy = original.clone()

clone方法是Object类的一个protected方法。在使用clone方法时,要求对象所属的类必须实现Cloneable接口,重新实现clone方法,并指定public访问修饰符。例如:

class Employee implements Cloneable
{
	public Employee clone() throws CloneNotSupportedException
	{
		return (Employee) super.clone();
	}
	//...
}

Cloneable接口是一个标记接口,它不包含任何方法,只是作为一个标记。如果在一个对象上调用clone方法,但这个对象的类没有实现Cloneable接口,就会抛出CloneNotSupportedException异常。

上面的Employee类中实现的clone方法使用它的默认操作,将整个对象逐个字段地进行拷贝。如果对象中所有的字段都是基本类型,拷贝不会有问题;但是如果有对象类型的字段,拷贝字段会得到引用同一个对象的另一个变量,相当于直接赋值,这样原对象和克隆对象仍然会共享部分信息。默认的克隆操作是“浅拷贝”,并没有克隆对象中引用的其他对象。如果原对象和浅克隆对象共享的子对象是不可变的,这种共享就是安全的;否则,必须重新定义clone方法建立深拷贝,同时克隆所有子对象。例如,Employee类的深拷贝实现如下:

class Employee implements Cloneable
{
	private String name; // 不可变类对象
	private double salary; // 基本类型字段
	private Date hireDay; // 可变类对象
	
	public Employee clone() throws CloneNotSupportedException
	{
		Employee cloned = (Employee) super.clone();
		cloned.hireDay = (Date) hireDay.clone(); // 可变类对象单独处理
		return cloned;
	}
	//...
}

所有数组类型都有一个publicclone方法,可以用这个方法建立新数组,包含原数组所有元素的副本。例如:

int[] luckyNumbers = { 2, 3, 5, 7, 11, 13 };
int[] cloned = luckyNumbers.clone();
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值