java 泛型题目讲解

本文详细解释了Java中的泛型知识,包括泛型的使用、类型擦除、泛型限制、静态成员与泛型的关系,以及泛型与数组、原始类型和类型推断的相关内容。通过实例展示了泛型的常见错误和注意事项。
摘要由CSDN通过智能技术生成

泛型的知识点

泛型仅存在于编译时期,编译期间JAVA将会使用Object类型代替泛型类型,在运行时期不存在泛型;且所有泛型实例共享一个泛型类

public class Main{
	public static void main(String[] args){
		ArrayList<String> list1=new ArrayList<String>();
		ArrayList<Integer> list2=new ArrayList<Integer>();
		System.out.println(list1 instanceof ArrayList);//true
		System.out.println(list2 instanceof ArrayList);//true
		//System.out.println(list1 instanceof ArrayList<String>);//编译错误,不存在此类型
	}
}

虽然list1和list2属于不同的类型,但是JVM加载的类仅ArrayList,各泛型实例共享这个类;
泛型的4个限制:
(1)不能new T()
(2)不能创建泛型数组:new T[];
但是可以使用强制类型转换的方法,不过编译器会给出一个警告。
T[] arr=(T[])new ArrayList[10];
(3)在静态环境下不能使用泛型类型的参数
如静态方法、静态变量、静态代码块都不能用泛型类型,原因在于各泛型实例共享一个类,而静态成员时随着类的加载就存在了的。
(4)异常类不能是泛型的。

学习地址

以下是关于Java泛型的一些重要知识点:

泛型类:
使用(或任何字母,但通常使用T、E、K、V等作为约定)来声明泛型类。
在类体内部,可以使用泛型类型参数(如T)来声明属性、方法参数和返回值类型。
泛型参数在编译时会被擦除(类型擦除),因此运行时无法知道具体的泛型类型(除非通过其他方式,如使用instanceof检查对象的具体类型)。
泛型方法:
泛型方法可以在普通类或非泛型类中定义。
使用(或其他字母)在方法返回类型之前声明泛型参数。
泛型参数可以与方法参数或局部变量一起使用。
类型擦除:
在编译时,泛型信息会被擦除,替换为原始类型(如Object)。
这意味着在运行时,泛型参数的实际类型信息是未知的。
为了保持类型安全,编译器会在编译时插入类型转换和类型检查。
泛型限制:
可以使用extends关键字为泛型参数设置上界(即该参数必须是某个类的子类或实现了某个接口)。
类似地,可以使用super关键字为泛型参数设置下界(即该参数必须是某个类的超类或接口)。
静态成员与泛型:
静态成员(包括变量、方法和内部类)不能使用类的泛型参数。
这是因为静态成员是与类关联的,而不是与类的任何特定实例关联的。
如果需要在静态上下文中使用泛型,可以定义泛型静态方法或泛型内部类。
泛型通配符:
使用?表示未知的泛型类型。
? extends T表示未知的类型,但它是T或T的子类。
? super T表示未知的类型,但它是T或T的超类。
泛型与数组:
不能创建泛型数组(如new T[10]),因为类型擦除后无法确保类型安全。
但可以使用泛型集合(如ArrayList)来替代。
泛型与原始类型:
如果没有为泛型类型提供实际的类型参数,编译器会将其视为原始类型(即擦除泛型参数的类型)。
这通常是不安全的,因为它会导致类型检查失效。
泛型与类型推断:
Java编译器可以使用类型推断来自动确定泛型方法的类型参数。
这在使用泛型方法时提高了代码的可读性和简洁性。
泛型与泛型方法的区别:
泛型类是在类声明中定义的,泛型参数用于整个类的所有非静态成员。
泛型方法是在方法声明中定义的,泛型参数仅用于该方法内部。
希望这些知识点能帮助你更好地理解和使用Java泛型。

题目一

下面五条语句中,错误的有_______________________________________________。
(1ArrayList<String> lists = new ArrayList<String>();2ArrayList<Object> lists = new ArrayList<String>();3ArrayList<String> lists = new ArrayList<Object>();4ArrayList<String> lists = new ArrayList();5ArrayList lists = new ArrayList<String>();
ArrayList<Object> lists = new ArrayList<String>(); // 编译错误

这是错误的。它尝试将lists声明为ArrayList类型,但试图用ArrayList的实例来初始化它。在Java中,泛型不是协变的,所以ArrayList不是ArrayList的子类型。

ArrayList<String> lists = new ArrayList<Object>(); // 编译错误

这同样是错误的。它尝试将lists声明为ArrayList类型,但试图用ArrayList的实例来初始化它。这会导致编译时类型不匹配错误,因为ArrayList不是ArrayList的子类型。

题二
  1. 下面泛型定义中不正确的是_D_。
    A. class Test1 {}
    B. interface Test2 {}
    C. class Test3{ void test () {}}
    D. class Test4{void test () {}}

D. class Test4{void test () {}} 是不正确的。在Java中,你不能在方法内部定义泛型参数 。泛型参数应该在类、接口或方法的返回类型之前定义。正确的泛型方法定义应该是这样的:

class Test4 {  
    <T> void test() {}  
}

题三

8. 下列语句编译时不出错的是___________。 
A.	List<?> c1 = new ArrayList<String> (); c1.add (new Object ());
B.	List<?> c2 = new ArrayList<String> (); c2.add (new String ("1"));
C.	List<?> c3 = new ArrayList<String> (); c3.add ("1"); 
D. 	List<?> c4 = new ArrayList<String> (); c4.add(null);

在Java中,List<?> 是一个未知类型的列表,这被称为通配符捕获(wildcard capture)。由于它是未知的,你不能向这样的列表中添加任何元素(除了null),因为你不能保证列表的实际类型能够容纳你试图添加的元素。

现在,我们来看每一个语句:

  • List<?> c1 = new ArrayList (); c1.add (new Object ());
    这个会编译失败。因为 c1 是一个未知类型的列表,你不能向其添加任何具体的对象(除了null)。

  • List<?> c2 = new ArrayList (); c2.add (new String (“1”));
    这个也会编译失败。同样的原因,c2 是一个未知类型的列表,你不能向其添加 String 对象。

  • List<?> c3 = new ArrayList (); c3.add (“1”);
    这个也会编译失败。c3 是一个未知类型的列表,你不能向其添加 String 对象。

  • List<?> c4 = new ArrayList (); c4.add(null);
    这个不会编译失败。因为 null 可以被赋予任何类型的变量,包括未知类型的列表。

因此,只有c4的语句不会编译失败。

题四

  1. 给定下列代码:
class Shape {}
class Circle extends Shape {}
class Triangle extends Shape {}
public class Test2_9 {
    public static void main (String [] args) {
        List<? extends Shape> list1 = new ArrayList< Triangle> ();
        List<? extends Shape> list2 = new ArrayList<Circle> ();

        System.out.println(list1 instanceof List< Triangle>);		①
        System.out.println(list2 instanceof List);				②
        System.out.println(list1.getClass() == list2.getClass());}
}

则关于语句①②③说法正确的是:___________。
A. ①②③输出结果为true、false、false
B. ①②③输出结果为true、true、true
C. ①编译出错,②③输出结果为false、false
D. ①编译出错,②③输出结果为true、true

选 D ,运行的时候不存在泛型,list1 list2 和 list3运行的时候都是ArrayList

题五

对于泛型类class A { … },T在A类里可以用作不同的地方,在A类类体内,下面语句正确的有_____________________________________________________________。
A. T x;
B. T m1() {return null;}
C. static T y;
D. void m2(T i) {}
E. static T s1() {return null;}
F. static void s2(T i) {}
G. static void s3(T1 i, T1 j){}

对于泛型类 A,我们可以分析每个选项来确定哪些在类体内是正确的。

  • A. T x;
    这是正确的。在泛型类的实例成员变量中,我们可以使用泛型参数 T 来声明一个变量。

  • B. T m1() {return null;}
    这也是正确的。在泛型类的方法中,我们可以使用泛型参数 T 作为返回类型或参数类型。但是,这里需要注意的是,如果 T 是一个非可空类型(例如一个具体的类或接口),那么直接返回 null 可能会产生编译错误,除非 T 被声明为可以包含 null(例如 T extends SomeType | null,在Java 8及以上版本使用)。但仅从语法的角度看,这个方法是合法的。

  • C. static T y;
    这是不正确的。静态成员变量不能是泛型的,因为静态成员是与类关联的,而不是与类的任何特定实例关联的,因此它们不能在声明时引用实例的类型参数。

  • D. void m2(T i) {}
    这是正确的。在泛型类的方法中,我们可以使用泛型参数 T 作为参数类型。

  • E. static T s1() {return null;}
    这是不正确的。与C选项类似,静态方法不能引用泛型参数,因为它们是类级别的,而不是实例级别的。

  • F. static void s2(T i) {}
    这也是不正确的。静态方法不能引用泛型参数。

  • G. static void s3(T1 i, T1 j){}
    这实际上是正确的,但它有点误导,因为它定义了一个新的类型参数 T1,而不是使用类的泛型参数 T。不过,从语法的角度看,这个静态方法声明是合法的,它定义了一个泛型静态方法。

总结:
正确的选项是 A, B, 和 D(如果忽略B选项的 null 返回可能导致的类型不匹配问题)。而G选项虽然技术上也是正确的,但它引入了一个新的类型参数 T1,这可能不是题目所期望的。如果考虑仅使用类声明的泛型参数 T,则G选项不应被视为正确。

题目六

List<? extends Comparable<Double>> x8 = new ArrayList<Double> ();

这个句子是没有问题的
在这里插入图片描述

题目七

  1. 关于java泛型,下面描述正确的是___________________________。
    A. 泛型的类型参数只能是类类型(包括自定义类),不能是基本类型
    B. 泛型的类型参数可以有多个
    C. 不能对泛型的具体实例类型使用instanceof操作,如 o instanceof ArrayList,否则编译时会出错。
    D. 不能创建一个泛型的具体实例类型的数组,如 new ArrayList[10],否则编译时会出错。

关于Java泛型的描述,正确的选项是:

A. 泛型的类型参数只能是类类型(包括自定义类),不能是基本类型
这个描述是正确的。在Java中,泛型类型参数不能是基本类型(如int, float, char等),而只能是类类型(包括自定义类、接口、数组类型或另一个泛型类型)。

B. 泛型的类型参数可以有多个
这个描述也是正确的。Java支持定义带有多个类型参数的泛型类和方法。例如:class MyClass<T, U> { … }。

C. 不能对泛型的具体实例类型使用instanceof操作,如 o instanceof ArrayList,否则编译时会出错。
这个描述不完全正确。在Java中,你不能直接使用带有泛型参数的instanceof操作,因为类型擦除导致在运行时泛型信息不可用。但是,你可以省略泛型参数,只检查原始类型,如 o instanceof ArrayList。然而,这并不会导致编译错误,只是这样的检查通常没有太多意义,因为它不会告诉你列表中元素的类型。

D. 不能创建一个泛型的具体实例类型的数组,如 new ArrayList[10],否则编译时会出错。
这个描述是正确的。在Java中,你不能创建泛型类型的数组,因为类型擦除和数组协变性的结合会导致运行时类型安全问题。如果你尝试这样做,编译器会报错。

综上所述,正确的选项是A、B和D。

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wniuniu_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值