Java_如何编写自己的泛型类

泛型

泛型,即参数化类型,类似于方法中将变量参数化,泛型是将原来定义的具体的类型参数化。

使用泛型的需求

Java中为什么要使用泛型,是因为泛型使用起来非常之方便,泛型类/方法针对于面向复用的开发。当我们想让一个类/方法同时适配于多种数据类型,这将大大省去我们编写重复代码的时间。

举一个常见的例子说明:

List<Integer> ilist = new ArrayList<Integer>();
List<String>  slist = new ArrayList<String>();

其中List接口以及类ArrayList均使用了泛型,从而可以构造出接收以类型Integer, String为参数的list;
所以下面的add操作中可以用不同类型的变量作为参数(用泛型可以省去重载方法的开销):

ilist.add(1);
ilist.add(2);
System.out.println(ilist);
slist.add("a");
slist.add("b");
System.out.println(slist);

分别输出为[1, 2] [a, b]

泛型应用场景

在这里插入图片描述
可以看到,除了部分底层的实现类,上层的类/抽象类/接口基本都是用泛型实现的,用泛型实现能够很好地进行复用,比如这里的L可以是Employee,Course,Process,从而派生出三个具体子类DutyIntervalSet,CourseIntervalSet,ProcessIntervalSet。

下面我们将以上图为例介绍如何编写一个自己的泛型类。

如何编写自己的泛型类

我们以MultiIntervalSet为例进行分析:
编写面向泛型的类时需要注意以下几点:
①泛型类的声明:需要在类名后加上<L>

public class MultiIntervalSet<L> implements IMultiIntervalSet<L>

②需要用到泛型的地方,全部用L代替

protected final Set<L> elabel = new HashSet<>();
protected final Map<L, List<Interval>> emap = new HashMap<>();

③只能对定义为泛型的变量做一些通用的操作
对于一个L label的泛型变量label,可以对它进行list.add(label),set.contains(label)等操作,由于它在具体传入的时候已经变成了一个具体的类。但是想要显式地调用它自己的方法只能调用如下几种(此时感觉label就是一个原始的Object类):
在这里插入图片描述
要小心,不要代入某个具体的类型,调用它特有的方法;
所以这也是泛型的一个局限性,对于通用的操作可以复用,但如果要用到某个具体类的方法,则不能将该类型作为参数传递。

④泛型类中可以有非泛型方法,也可以有非泛型的成员变量。
这点很好理解,泛型类中不一定每个方法都需要用到泛型,用到泛型时才是泛型方法;同时泛型类中当然也可以有具体类型的成员变量。

protected final Set<L> elabel = new HashSet<>();
protected final Map<L, List<Interval>> emap = new HashMap<>();
protected Interval duration;

最后给出一个泛型方法的例子
public boolean remove(L label) {
		if(!elabel.contains(label)) return false;
		else {
			emap.remove(label);
			elabel.remove(label);
			checkRep();
			return true;
		}
	}

总结

总之,写泛型类的时候要保持平常心,将L当作一个普通的类型去编写代码,开始时不要因为觉得L的写法陌生,而不敢对它进行操作,或者不愿意自定义泛型类/方法。


补充介绍:类型通配符

类型通配符一般是使用?代替具体的类型参数。例如 List<?> 在逻辑上是List<String>,List<Integer> 等所有List<具体类型实参>的父类。
①在某些情况下,我们必须使用到类型通配符,否则静态编译时会报错:
比如:重写equals函数时用到instanceof运算符

@Override public boolean equals(Object obj) {
    	if(obj instanceof Edge<L>) {
			Edge<L> e = (Edge<L>)obj;
		//...后续逻辑省略	

编译报错:Cannot perform instanceof check against parameterized type Edge. Use the form Edge<?> instead since further generic type information will be erased at runtime

修改为下图所示,编译通过。

if(obj instanceof Edge<?>) {
			Edge<?> e = (Edge<?>)obj;

②类型通配符还可以用于特定的情形
比如限定类型的上/下限:

List<? extends Number>
限定接收类型的上限为Number,接受Number及其下层子类类型

List<? super Number>
限定接收类型的下限为Number,接受Number及其三层父类类型,如 Object 类型的实例。

### 回答1: 好的,下面是一个简单的泛型类Java 代码示例: ``` public class MyGenericClass<T> { private T myVar; public MyGenericClass(T myVar) { this.myVar = myVar; } public T getMyVar() { return myVar; } public void setMyVar(T myVar) { this.myVar = myVar; } } ``` 在这个示例中,我们定义了一个泛型类 `MyGenericClass`,它可以接受任何类的参数。在这个类中,我们定义了一个私有变量 `myVar`,类参数 `T`。我们还实现了两个公共方法 `getMyVar` 和 `setMyVar`,用于获取和设置 `myVar` 的值。 泛型类可以让我们编写更加通用的代码,可以适用于不同类的数据。当我们需要创建一个特定类的对象时,只需要将该类作为泛型类的参数即可。 ### 回答2: 泛型类是指在类的定义中使用了一个或多个类参数,这样我们就可以在创建对象的时候指定具体的类。在Java中,我们使用关键字<T>来表示类参数。 下面是一个简单的例子,展示了如何使用Java编写一个泛型类: ```java public class GenericClass<T> { private T data; public GenericClass(T data) { this.data = data; } public T getData() { return data; } public void setData(T data) { this.data = data; } public static void main(String[] args) { GenericClass<Integer> integerObj = new GenericClass<>(10); System.out.println("Integer data: " + integerObj.getData()); GenericClass<String> stringObj = new GenericClass<>("Hello"); System.out.println("String data: " + stringObj.getData()); } } ``` 在上述代码中,我们定义了一个泛型类`GenericClass<T>`,使用类参数`<T>`表示类可以接受任意类的数据。我们使用`data`成员变量来存储泛型类的数据,在构造函数中初始化,并通过`getData()`和`setData()`方法对其进行访问和修改。 在`main`方法中,我们创建了两个`GenericClass`对象:一个接收整数类,另一个接收字符串类。然后通过`getData()`方法获取和打印对象中的数据。 这个例子只是展示了泛型类的基本用法,泛型类还有更多高级用法,比如限制类参数的范围,通配符等。在实际开发中,我们可以根据需要使用不同的泛型类来提高代码的可复用性和灵活性。 ### 回答3: 泛型类是使用一个或多个类参数作为其属性或方法的数据类的类。在Java中,我们可以使用泛型类来增加代码的灵活性和重用性。下面是一个使用Java编写的简单泛型类示例: ```java public class GenericClass<T> { private T data; public GenericClass(T data) { this.data = data; } public T getData() { return data; } public void setData(T data) { this.data = data; } public static void main(String[] args) { // 创建整泛型类对象 GenericClass<Integer> intObj = new GenericClass<>(10); System.out.println("整数据为: " + intObj.getData()); // 创建字符串泛型类对象 GenericClass<String> stringObj = new GenericClass<>("Hello, World!"); System.out.println("字符串为: " + stringObj.getData()); } } ``` 在上述代码中,我们定义了一个泛型类`GenericClass<T>`,其中`T`是一个参数。我们在类的构造函数和属性中使用类参数`T`。在`main`方法中,我们创建了一个一个字符串泛型类对象,并分别打印出其数据。 通过使用泛型类,我们能够在不同的类中使用相同的代码,只需改变类参数即可适应不同的数据类。这大大提高了代码的可重用性和灵活性。 该示例只是一个简单的用法示例,实际上,泛型类可以应用于更复杂的情况,例如集合类、容器类等,以实现更高级的功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值