Java学习----泛型

今日学习内容总结如下:

不引入泛型的问题:

当将一个对象放入集合中,集合不会记住此对象的类型,当再次从集合中取出此对象时,改对象的编译类型变成了Object类型,但其运行时类型任然为其本身类型。

因此取出集合元素时需要人为的强制类型转化到具体的目标类型,但是很容易出现
ClassCastException异常

那么有没有什么办法可以使集合能够记住集合内元素各类型,且能够达到只要编译时不出现问题,运行时就不会出现java.lang.ClassCastException异常呢?

答案就是使用泛型,可以将运行时的类型检查搬到编译期实现

什么是泛型

泛型是jdk5引入的类型机制,就是将类型参数化。泛型作为一种安全机制而产生

泛型机制将类型转换时的类型检查从运行时提前到了编译时,使用泛型编写的代码比杂乱的使用object并在需要时再强制类型转换的机制具有更好的可读性和安全性。

泛型在本质上是指类型参数化。所谓类型参数化,是指用来声明数据的类型本身,也是可以改变的,它由实际参数来决定。在一般情况下,实际参数决定了形式参数的值。而类型参数化,则是实际参数的类型决定了形式参数的类型。

在声明List<E>阶段E是什么类型不确定,这里的E仅仅充当占位符的作用,在具体调用时类型才能确定,而E的所有位置将被指定的类型所替代

使用泛型的定义

public interface List<E> extends Collection<E>
这里的<>中的内容就是类型参数,一般建议使用T或者E之类的全大写
{
         E get(int index);  获取的元素类型就是定义时指定的类型
         void add(E element);
}

List<String> list=new ArrayList<String>();就是将String传递给E,用于替代定义中的EString str=list.get(0); 不需要进行类型转换

List<Date> list=new ArrayList<>();//这里使用菱形语法,支持泛型推导
list.add("abc");//语法报错,编译时就会进行类型检查
list.add(123);//语法报错
list.add(new Date());
for(int i=0;i<list.size();i++){
     Date temp=list.get(i);   直接获取目标类型,不需要进行类型转换
     System.out.println(temp.getYear()+1900);
}

典型场景

获取两个整数中较大的整数
        

Integer max(Integer a, Integer b){
             return a>b?a:b;      
         }

如果需要比较的不是Integer类型,而是Double或是Float类型,那么就需要另外再写max()方法【方法的重载】。

引入泛型的目的实际上就是能够在编写max()方法时,不必确定参数a和b的数据类型,而等到调用的时候再来确定这两个参数的数据类型,那么只需要编写一个max()就可以了,这将大大降低程序员编程的工作量。

Comparable接口---用于类定义中表示当前类型的对象是可比较的,接口中定义了比较方法,要求具体类型提供实现【具体的比较规则】

import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import java.util.Random;

public class Test1 {
	public static void main(String[] args) {
		List list = new ArrayList();
		list.add(new Date());
		list.add(new Random());
		list.add("abcd");
		for (int i = 0; i < list.size(); i++) {
			Object obj = list.get(i);
			if (obj != null) {
				if (obj instanceof String) {
					String ss = obj.toString();
					System.out.println(ss.length());
				} else if (obj instanceof Date) {
					Date dd=(Date)obj;
					System.out.println(dd);
				}
			}
		}
	}
}

定义泛型接口

public interface IA<T>{
   public T pp();
}

定义泛型类上给泛型添加约束

public class MyClass<T extends IA>{}

在泛型类或者泛型接口中不允许直接使用super

// 在这里, Integer 可以认为是 Integer 的 "父类"
List<? super Integer> array = new ArrayList<Integer>();
// Number 是 Integer 的 父类
List<? super Integer> array = new ArrayList<Number>();
// Object 是 Integer 的 父类
List<? super Integer> array = new ArrayList<Object>();

如果要求泛型实现两个或者两个以上的接口,要求传入的类型T必须同时实现了接口1和接口2。

public class MyClass<T extends 接口1 & 接口2>{}

如果要求泛型必须继承B类,同时实现IC接口

public class MyClass<T extends B & IC >{}

如果要求泛型继承一个类并且实现接口

public class MyClass<T extends 类型 & 接口>{}

<? extends T>只能用于方法返回,告诉编译器此返参的类型的最小继承边界为T,
T和T的父类都能接收,但是入参类型无法确定,只能接受null的传入
<? super T>只能用于限定方法入参,告诉编译器入参只能是T或其子类型,而返参只能用Object类接收

  • `?`既不能用于入参也不能用于返参
  • 频繁往外读取内容的,适合用上界extends
  • 经常往里插入的,适合用下界Super
import java.util.ArrayList;
import java.util.List;

public class Test1 {
	public void pp(List<?> list) {
//		list.add("123"); // 语法报错
		
//		Object aa="123";
//		list.add(aa);
		
		for(Object tmp:list) {
			System.out.println(tmp);
		}
	}
	
	public void bb(List<Object> list) {}

	public static void main(String[] args) {
		List<String> list = new ArrayList<>();
		list.add("bbbb");
		
		Test1 t = new Test1();
		t.pp(list);
		
//		t.bb(list);   //List<String>和List<Object>之间没有任何关系
		
		List<Object> alist=new ArrayList<>();
		alist.add("1234");
		alist.add(1234);
		t.bb(alist);
	}
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: jmu-java-05集合()-10-generalstack是关于Java中的通用栈的学习内容。通用栈是一种可以存储任意类数据的栈结构,通过的方式实现。在学习中,我们可以了解到通用栈的实现原理、使用方法以及注意事项等内容,帮助我们更好地理解和应用Java。 ### 回答2: JMU-Java-05集合()-10-GeneralStack是一个Stack(栈)的实现类,使用Java中的进行定义,可以存储任何类的数据。 在该类中,使用一个Object类的数组进行存储元素,并通过一个整变量top来表示当前栈顶元素的下标。在push(入栈)方法中,先将top加一,然后将元素存储在数组中,实现了入栈的功能。在pop(出栈)方法中,先判断栈是否为空,若为空则抛出栈空异常,否则将top减一,并返回数组中相应的元素,实现了出栈的功能。其他方法如isEmpty(判断栈是否为空)、isFull(判断栈是否已满)和size(获取栈中元素个数)也在该类中实现。 该类的定义使得我们可以使用该类存储任何类的数据,而不需要在定义类时指定数据类,提高了其灵活性和可复用性。在使用时,我们只需要在创建对象时传入相应的数据类,如:GeneralStack<Integer> stack = new GeneralStack<Integer>();即可。 该类还实现了Iterable接口,使得该类可以使用foreach循环进行遍历操作,方便了我们对栈中元素的操作。同时,该类还通过对数组的动态扩容,解决了数组固定大小的限制问题,从而提高了该栈类的通用性和易用性。 总之,JMU-Java-05集合()-10-GeneralStack是一个使用实现的通用栈类,可以存储各种类的数据,并提供了常用的栈操作方法,具有较高的可复用性与适用性。 ### 回答3: jmu-java-05集合()-10-generalstack,意为基于的栈实现。 首先我们需要了解什么是Java SE 5引入的一个新特性,它可以让我们在编写代码时定义一些未知的类参数,以达到代码的复用和类安全的目的。对于集合类或者其他容器类而言,适用于任何类的容器的需求是普遍存在的,这时就可以应用。 在Java使用<>标识,其语法格式如下: ```java public class 类名<类参数列表> { //成员变量、方法等 } ``` 类参数列表是由逗号隔开的参数列表,可以理解为未知类的占位符。这样的好处是可以在编写集合类时,避免出现类转换等一系列问题,提高程序的可读性。 而在jmu-java-05集合()-10-generalstack中,则是基于实现的数据结构——栈。栈是一种后进先出(LIFO)的数据结构,它只允许在栈顶添加或删除元素,因此操作非常简单、快速并且高效。 在该实现中,栈的元素可以是任意类,数据元素入栈则是通过 push() 方法实现的,出栈则是通过 pop() 方法实现的。其中,push() 方法用于向栈中添加新的元素,pop() 方法用于弹出并返回栈顶的元素。同时,还提供了 isEmpty() 方法、size() 方法等基本的栈操作方法。 总之,jmu-java-05集合()-10-generalstack是一种基于实现的栈,提供了复用和类安全的目的。其实现使用简便、高效、易于扩展等优点,广应用在Java开发中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值