Java -- 泛型之通配符(五)

本文详细介绍了Java泛型中的通配符概念,包括子类型限定和超类型限定的通配符,以及如何利用它们提高代码的复用性和灵活性。通过具体的示例代码展示了如何正确使用这些通配符。

Java -- 泛型之通配符(五)

 

Java泛型给设计人员提供了一种很有用的通配符:"?",使用它能够使我们设计出复用性更高的代码。通配符不是类型变量,因此,不能在编写代码中使用“?”作为一种类型。

下面就简单介绍一下这部分内容。

 

先看要使用的示例代码:

class Employee {
	
}

class Manager extends Employee{
	
}

class HumanResource extends Employee{
	
}

class Basic {
	
	public static void f(List<Employee> list){
		System.out.println("");
	}
}

如果有这种调用:

		List<Manager> list = new ArrayList<>();
		Basic.f(list);//error

编译过程是无法通过的。原因在上一篇文章讲过,虽然Manager和Employee有继承关系,但List<Manager>和List<Employee>没有继承关系;不能将list做参数调用Basic.f()方法,这一点很受限制。那我们应该怎么解决呢?其实方法很简单,就是使用我们将要介绍的通配符,改造Basic.f()方法:

	public static void f(List< ? extends Employee> list) {
		System.out.println("f() int Basic");
	}

就可以完成上面的调用了。

那List< ? extends Employee>(子类型限定)代表什么呢?其实通配符代“?”表示“任意类型”,它不同于类型变量(T、E等);结合起来看就是:表示任何的泛型List,它的类型参数是任何Employee的子类型,如这里的List<Manager>。

再加入Pair<T>:

class Pair<T> {

	private T first;
	private T second;

	public Pair(T first, T second) {
		super();
		this.first = first;
		this.second = second;
	}

	public T getFirst() {
		return first;
	}

	public void setFirst(T first) {
		this.first = first;
	}

	public T getSecond() {
		return second;
	}

	public void setSecond(T second) {
		this.second = second;
	}
}

看如下调用:

		Pair<Manager> manager = new Pair<>(new Manager(),new Manager());		
		Pair<? extends Employee> subEmploy = manager;
		subEmploy.setFirst(new Manager());//error

最后的方法调用存在错误。在Pair中,将"? extends Employee"替换类型参数T:setFirst(? extends Employee),它的参数类型是不确定的,只知道它会接收一个Employee的子类类型。?代表不确定的类型,编译器并不接受一个函数的参数类型是不确定的;它拒绝传递任何特定类型,因为?不能用来匹配,通配符并不是类型变量。

但对于? extends Employee getFirst(),我们知道返回值一定是某个Employee的子类型,所以getFirst()可以赋值给一个Employee引用:

		Pair<Manager> manager = new Pair<>(new Manager(),new Manager());		
		Pair<? extends Employee> subEmploy = manager;
		Employee employee = subEmploy.getFirst();//legal

 

既然我们可以实现子类型限定的通配符,那么同时我们也可以实现超类型限定的通配符(超类型限定):

?  super HumanResource,限制范围是HumanResource的所有超类型;

看示例调用代码:

class Basic {

	public static void f(List< ? extends Employee> list) {
		System.out.println("f() int Basic");
	}
	
	public static void g(List < ? super HumanResource> list) {
		System.out.println("g() int Basic");
	}
}
		List<Employee> list = new ArrayList<>();
		Basic.g(list);

Output:g() int Basic。

 

同上,我们再考虑:void setFirst(? super HumanResource)、?  super HumanResource getFirst();

对于setFirst(...),编译器知道它的参数类型是HumanResource的所有父类型,并不知道确切类型;但我们可以使用任意HumanResource对象调用该方法,而不会报错。

但对于getFirst(...),返回的对象类型我们并不能确定,只知道是HumanResource的某个父类型;此时,我们只能将返回值赋给一个Object对象,才能应对所有情景。

《Java core》中有一条结论:直观地讲,带有超类型限定的通配符可以向泛型对象写入,带有子类型限定的通配符可以从泛型对象读取。它总结了上述的两种类型限定。

 

最后,我们还可以使用无限定的通配符,例如Pair<?>。Pair<?>和Pair的原始类型是有区别的。

对于Pair<?>:

  1. void setFirst(?):该方法不能被调用,即使我们传入Object对象;可以调用setFirst(null);
  2. ? getFirst():返回值只能赋给一个Object引用;
  3. Pair<?>和Pair(原始类型)的本质不同在于:可以用任意Object对象调用原始的Pair类的setObject()方法,来自《Java core》;

 

 

JavaJava 5引入的新特性,可以提高代码的可读性和安全性,降低代码的耦合度。是将类型参数化,实现代码的通用性。 一、的基本语法 在声明类、接口、方法时可以使用的声明方式为在类名、接口名、方法名后面加上尖括号<>,括号中可以声明一个或多个类型参数,多个类型参数之间用逗号隔开。例如: ```java public class GenericClass<T> { private T data; public T getData() { return data; } public void setData(T data) { this.data = data; } } public interface GenericInterface<T> { T getData(); void setData(T data); } public <T> void genericMethod(T data) { System.out.println(data); } ``` 其中,`GenericClass`是一个类,`GenericInterface`是一个接口,`genericMethod`是一个方法。在这些声明中,`<T>`就是类型参数,可以用任何字母代替。 二、的使用 1. 类的使用 在使用类时,需要在类名后面加上尖括号<>,并在括号中指定具体的类型参数。例如: ```java GenericClass<String> gc = new GenericClass<>(); gc.setData("Hello World"); String data = gc.getData(); ``` 在这个例子中,`GenericClass`被声明为一个类,`<String>`指定了具体的类型参数,即`data`字段的类型为`String`,`gc`对象被创建时没有指定类型参数,因为编译器可以根据上下文自动推断出类型参数为`String`。 2. 接口的使用 在使用接口时,也需要在接口名后面加上尖括号<>,并在括号中指定具体的类型参数。例如: ```java GenericInterface<String> gi = new GenericInterface<String>() { private String data; @Override public String getData() { return data; } @Override public void setData(String data) { this.data = data; } }; gi.setData("Hello World"); String data = gi.getData(); ``` 在这个例子中,`GenericInterface`被声明为一个接口,`<String>`指定了具体的类型参数,匿名内部类实现了该接口,并使用`String`作为类型参数。 3. 方法的使用 在使用方法时,需要在方法名前面加上尖括号<>,并在括号中指定具体的类型参数。例如: ```java genericMethod("Hello World"); ``` 在这个例子中,`genericMethod`被声明为一个方法,`<T>`指定了类型参数,`T data`表示一个类型为`T`的参数,调用时可以传入任何类型的参数。 三、通配符 有时候,我们不知道的具体类型,可以使用通配符`?`。通配符可以作为类型参数出现在方法的参数类型或返回类型中,但不能用于声明类或接口。例如: ```java public void printList(List<?> list) { for (Object obj : list) { System.out.print(obj + " "); } } ``` 在这个例子中,`printList`方法的参数类型为`List<?>`,表示可以接受任何类型的`List`,无论是`List<String>`还是`List<Integer>`都可以。在方法内部,使用`Object`类型来遍历`List`中的元素。 四、的继承 类和接口可以继承或实现其他类或接口,可以使用子类或实现类的类型参数来替换父类或接口的类型参数。例如: ```java public class SubGenericClass<T> extends GenericClass<T> {} public class SubGenericInterface<T> implements GenericInterface<T> { private T data; @Override public T getData() { return data; } @Override public void setData(T data) { this.data = data; } } ``` 在这个例子中,`SubGenericClass`继承了`GenericClass`,并使用了相同的类型参数`T`,`SubGenericInterface`实现了`GenericInterface`,也使用了相同的类型参数`T`。 的限定 有时候,我们需要对类型参数进行限定,使其只能是某个类或接口的子类或实现类。可以使用`extends`关键字来限定类型参数的上限,或使用`super`关键字来限定类型参数的下限。例如: ```java public class GenericClass<T extends Number> { private T data; public T getData() { return data; } public void setData(T data) { this.data = data; } } public interface GenericInterface<T extends Comparable<T>> { T getData(); void setData(T data); } ``` 在这个例子中,`GenericClass`的类型参数`T`被限定为`Number`的子类,`GenericInterface`的类型参数`T`被限定为实现了`Comparable`接口的类。 六、擦除Java中,信息只存在于代码编译阶段,在编译后的字节码中会被擦除。在运行时,无法获取的具体类型。例如: ```java public void genericMethod(List<String> list) { System.out.println(list.getClass()); } ``` 在这个例子中,`list`的类型为`List<String>`,但是在运行时,`getClass`返回的类型为`java.util.ArrayList`,因为信息已经被擦除了。 七、类型推断 在Java 7中,引入了钻石操作符<>,可以使用它来省略类型参数的声明。例如: ```java List<String> list = new ArrayList<>(); ``` 在这个例子中,`ArrayList`的类型参数可以被编译器自动推断为`String`。 八、总结 Java是一个强大的特性,可以提高代码的可读性和安全性,降低代码的耦合度。在使用时,需要注意它的基本语法、使用方法、通配符、继承、限定、擦除类型推断等问题。
评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值