JavaSE 泛型

泛型是用于解决安全问题的,用于解决安全机制

好处
1.将运行时期出现问题ClassCastException,转移到了编译时期。
方便于程序员解决问题。让运行时期问题减少,安全。
2.避免了强制转换的麻烦。

泛型格式:通过<>来定义要操作的引用数据类型。

在使用java提供的对象时,什么时候写泛型呢?

通常在集合框架中很常见,只要见到<>就要定义泛型

其实<>就是用来接收类型的。当使用集合时,将集合中要存储的数据类型作为参数传递到<>中即可。

class GenericDemo{
    public static void main(String[] args) {
        ArrayList<String>a1 = new ArrayList<String>();       //强制只能存储字符串类型

        a1.add("abc01");
        a1.add("abc02");
        a1.add("abc03");

        Iterator<String> it = a1.iterator();
        while(it.hasNext()){
            String s = it.next();       
            System.out.println(s+":"+s.length());
        }
    }
}

泛型类

什么时候定义泛型类?
当类中要操作的引用数据类型不确定的时候,定义泛型来完成扩展。
定义格式
* public class 类名<泛型类型1,…>

class Worker{
}

class Utils<QQ> {
    private QQ q;
    public void setObject(QQ q) {
        this.q = q;
    }
    public QQ getObject() {
        return q;
    }
}

class GenericDemo3 {
    public static void main(String[] args)  {
        Utils<Worker> u = new Utils<Worker>();      //只能操作Worker类型

        u.setObject(new Worker());
        Worker w = u.getObject();
    }
}

泛型方法

泛型类定义的泛型,在整个类中有效。如果被方法使用,那么泛型类的对象明确要操作的具体类型后,所有要操作的类型就已经固定了。

为了让不同方法可以操作不同类型,而且类型还不确定。那么可以将泛型定义在方法上。
定义格式
* public <泛型类型> 返回类型 方法名(泛型类型 变量名)

class Demo {
    public <T> void show(T t) {
        System.out.println("show:"+t);
    }

    public <Q> void print(Q q) {
        System.out.println("print:"+q);
    }
}

class GenericDemo4 {
    public static void main(String[] args) {
        Demo d = new Demo();
        d.show("haha");             //String类型
        d.show(new Integer(4));     //          
        d.print("heihei");          //String类型
        d.print(5);
    }
}

特殊之处:
静态方法不可以访问类上定义的泛型,如果静态方法操作的引用数据类型不确定,可以将泛型定义在方法上。

class Demo<T> {
    public <T> void show(T t){
        System.out.println("show:"+t);
    }

    public <Q> void print(Q q){
        System.out.println("print:"+q);
    }

    /*                                          
    public static void method(T t) {                   这样错误
        System.out.println("method:"+t);
    } 
    */

    public static <W> void method(W w) {
        System.out.println("method:"+w);
    }
}
class GenericDemo4{
    public static void main(String[] args) {
        Demo<String> d = new Demo<String>();
        d.show("haha");
        //d.show(4);            只能打印T  也就是String类型
        d.print("heihei");
        d.print(5);
        Demo.method("hahahhaha");
    }
}

泛型接口

泛型定义在接口上
看如下代码:

interface Inter<T> {
    void show(T t);
}

calss InterImpl implements Inter<String> {  //传入String类型
    public void show(String t) {
        System.out.println("show:"+t)
    }
}

calss GenericDemo5 {
    public static void main(String[] args) {
        InterImpl i = new InterImpl();
        i.show("haha");                             //只能输入String类型
    }
}
interface Inter<T> {
    void show(T t);
}

calss InterImpl<T> implements Inter<T> {
    public void show(T t) {
        System.out.println("show:"+t)
    }
}

calss GenericDemo5 {
    public static void main(String[] args) {
        InterImpl<Integer> i = new InterImpl<Integer>();      //这样可以由调用者决定操作类型            
        i.show(4);                             
    }
}

泛型限定

class GenericDemo6 {
    public static void main(String[] args) {
        ArrayList<String> a1 = new ArrayList<String>();
        a1.add("abc1");
        a1.add("abc2");
        a1.add("abc3");

        ArrayList<Integer> a2 = new ArrayList<Integer>();
        a2.add("4");
        a2.add("5");
        a2.add("6");

        printColl(a1);
        printColl(a2);
    }

    public static void printColl(ArrayList a1) {
        Iterator it = a1.iterator();

        while(it.hasNext()) {
            System.out.println(it.next());
        }
    }
}

都可以输出,但不能用类型的特有方法,比如length输出长度,这是String类型的特有方法。

class GenericDemo6{
    public static void main(String[] args) {
        ArrayList<Person> a1 = new ArrayList<String>();
        a1.add(new Person("abc1"));
        a1.add(new Person("abc2"));
        a1.add(new Person("abc3"));
        printColl(a1);

        ArrayList<Student> a2 = new ArrayList<Student>();
        a2.add(new Student("abc4"));
        a2.add(new Student("abc5"));
        a2.add(new Student("abc6"));

        //printColl(a2);  //ArrayList<Person> a1 = new ArrayList<Student>();  错误,必须保证左右相同
    }

    public static void printColl(ArrayList<Person> a1) {
        Iterator<Person> it = a1.iterator();

        while(it.hasNext())  {
            System.out.println(it.next().getName());
        }
    }
}

如果我只想打印Person和Person的子类,又不想打印任何类型,希望有一个类型范围,该如何操作呢?

如下,泛型限定

class GenericDemo6{
    public static void main(String[] args){
        ArrayList<Person> a1 = new ArrayList<String>();
        a1.add(new Person("abc1"));
        a1.add(new Person("abc2"));
        a1.add(new Person("abc3"));
        printColl(a1);

        ArrayList<Student> a2 = new ArrayList<Student>();
        a2.add(new Student("abc4"));
        a2.add(new Student("abc5"));
        a2.add(new Student("abc6"))
        printColl(a2);     //这样既可以打印Person类,也可以打印Person的子类
    }

    public static void printColl(ArrayList<? extends Person> a1) {  //泛型限定
        Iterator<? extends Person> it = a1.iterator();

        while(it.hasNext()){
            System.out.println(it.next().getName());
        }
    }
}

? 通配符。也叫做占位符。
泛型通配符 ? :任意类型,如果没有明确的,那么就是Object以及任意的Java类了
泛型的限定:
? extends E :可以接受E类型或者E的子类型 。 上限。
? super E : 可以接受E类型或者E的父类型。 下限。

class Person {
    private String name;
    Person(String name) {
        this.name = name
    }
    public String name getName() {
        return name;
    }
    public String toString() {
        return "person :"+name;
    }
}

class Student extends Person {
    Student(String name) {
        super(name);
    }
}

class Worker extends Person {
    Worker(String name) {
         super(name);
    }
}

class GenericDemo7{
    public static void main(String[] args) {
        TreeSe<Student> ts = new TreeSet<Student>(new comp());
        ts.add(new Student("abc01"));
        ts.add(new Student("abc06"));
        ts.add(new Student("abc02"));

        Iterator<Student> it = ts.iterator();

        while(it.hasNext()) {
            System.out.println(it.next.getName());
        }

        TreeSe<Worker> ts1 = new TreeSet<Worker>(new comp());
        ts1.add(new Worker("wabc01"));
        ts1.add(new Worker("wabc06"));
        ts1.add(new Worker("wabc02"));

        Iterator<Worker> it1 = ts1.iterator();

        while(it1.hasNext()) {
            System.out.println(it1.next.getName());
        }
    }
}

/*
class StuComp implements Comparator<Student> {
    public int compara(Student s1,Student s2) {
        return s1.getName().compareTo(s2.getName());
    }
}
class WorkerComp implements Comparator<Worker> {
    public int compara(Worker w1,Worker w2) {
        return w1.getName().compareTo(w2.getName());
    }
}
*/                            太麻烦,每一个类型都要写一个比较器

class Comp implements Comparator<Person>  {                 //写一个其父类的比较器  其子类都可以使用
    public int compara(Person p1,Person p2) {
        return p1.getName().compareTo(p2.getName());
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值