【java】【泛型】泛型genetic

一、泛型

泛型的由来

        通过Object转型问题引入,早起Object类型可以接收任意的对象类型,但是在实际使用中,会有类型转换的问题,也就存在安全隐患,所以java提供了泛型来解决这个安全问题。

        子类对象在当做参数传递时有时会被自动提升为object类型,但后面想使用该类的方法(子类特有的方法)时,就需要进行类型强制转换。有时若忘记进行转换时,编译也能通过,但是运行时会出错,存在安全隐患。

1、泛型概述

<>:里面放的是引用数据类型,指定集合里只能放指定引用数据类型,包括子类

        泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法。 Java语言引入泛型的好处是安全简单。

2、泛型好处

    * 提高安全性(将运行期的错误转换到编译期,编译器就会给出可能错误的提示)

    * 省去强转的麻烦

        在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。   

        泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,以提高代码的重用率。

3、泛型基本使用

    * <>中放的必须是引用数据类型,后面添加的是该类及子类对象

4、泛型使用注意事项

    * 前后的泛型必须一致,或者后面的泛型可以省略不写(1.7的新特性菱形泛型) 

    * 泛型最好不定义为Object,因为它是所有类的父类,加与不加没什么意义

import java.util.ArrayList;
import java.util.Iterator;
import com.heima.bean.Person;

public class Demo1_Generic {
	public static void main(String[] args) {
		//demo1();
		//int[] arr = new byte[5];				//数组要保证前后的数据类型一致
		//ArrayList<Object> list = new ArrayList<Person>();	//集合的泛型要保证前后的数据类型一致
		//ArrayList<Object> list = new ArrayList<>();		//1.7版本的新特性,菱形泛型
		ArrayList<Object> list = new ArrayList<>();		//泛型最好不要定义成Object,没有意义
		list.add("aaa");
		list.add(true);
	}

	public static void demo1() {
		ArrayList<Person> list = new ArrayList<Person>();
		list.add(new Person("张三", 23));
		list.add(new Person("李四", 24));
		
		Iterator<Person> it = list.iterator();
		while(it.hasNext()) {
		    //System.out.println(it.next());
			
		    //System.out.println(it.next().getName() + "..." + it.next().getAge()); //next方法只能调用一次,如果调用多次会将指针向后移动多次,张三,24
		    Person p = it.next();  //调用一次	 		
		    System.out.println(p.getName() + "..." + p.getAge());
		}
	}
}

二、ArrayList存储字符串和自定义对象并遍历泛型版

import java.util.ArrayList;
import java.util.Iterator;
import com.heima.bean.Person;
public class Demo2_Generic {
	public static void main(String[] args) {
		//demo1();
		ArrayList<Person> list = new ArrayList<>();
		list.add(new Person("张三", 23));
		list.add(new Person("李四", 24));
		list.add(new Person("王五", 25));
		list.add(new Person("赵六", 26));
		
		Iterator<Person> it = list.iterator();
		while(it.hasNext()) {
			Person p = it.next();				//将集合中的每一个元素用Person记录,无需进行类型强制转换
			System.out.println(p.getName() + "..." + p.getAge());
		}
	}
	public static void demo1() {
		ArrayList<String> list = new ArrayList<>();		//创建集合对象
		list.add("a");
		list.add("b");
		list.add("c");
		list.add("d");
		
		Iterator<String> it = list.iterator();
		while(it.hasNext()) {
			System.out.println(it.next());    //无需进行类型强制转换
		}
	}
}


三、泛型类(在类上单独使用,不实现借口

定义类:public class Tool<T>

实例化:Tool<student> t = new Tool<student>();

四、泛型方法

1、非静态方法

定义方法:public  void show(T t)

调用方法:t.show<student>()

方法泛型需要最好与类的泛型一致:因为实例化对象的时候,泛型也相当于通过参数传递进类里,而类里的方法泛型不一致就会导致出错。解决方法:需要在方法上声明另外的泛型,使用时直接传递该泛型即可

定义方法:public<Q>  void show(Q q)


2、静态方法:随着类的加载而加载

方法泛型和类泛型不能一致,因为类名.静态方法时对象可能还没创建,即类泛型还不知道,就会出现错误,此时方法就必须声明自己的泛型,而不能跟类的泛型一样。

此刻的一样多一层意思:

public  void show(Q q) :不行,这里的Q是随着类的加载而加载,这里需要换一个泛型即T,不能跟类一致

public<Q>  void show(Q q):可行,这里的Q是随着静态方法的加载而加载,是在方法上声明我自己的泛型,跟类不一样


五、泛型接口

1、类在实现接口的时候就指定具体泛型

class Demo implements Inter<String> {		//推荐用这种
	@Override
	public void show(String t) {
		System.out.println(t);
	}

2、方法实现接口的时候再自己定义一下泛型

class Demo<T> implements Inter<T> {	//没有必要在实现接口的时候给自己类加泛型
	@Override
	public void show(T t) {
		System.out.println(t);
	}
	
}


六、泛型高级之通配符

* A:泛型通配符<?>

    * 任意类型,如果没有明确,那么就是Object以及任意的Java类了

* B:? extends E

    * 向下限定,E及其子类

* C:? super E

    * 向上限定,E及其父类




评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值