java

前言

Java里面的泛型在实际开发中运用的很多,学过C++的同学一定知道C++的模板,而Java中的泛型,一定程度上和它还是挺像的。

相信写Java的人,大都有用过List的实现类ArrayList。在Java没有泛型之前,它的内部是一个Object的数组实现的。这也导致一个问题,每次使用里面的元素的时候需要向下转型,而且很明显,如果是Object的话,意味着我们可以丢任何对象进去。自动转型成Object,这样在使用的时候就很容易出问题,不知道里面存的是什么。如:

ArrayList list = new ArrayList();

list.add("string1");

list.add("string2");

String str = (String) list.get(0);

list.add(new File("test.txt"));

然而使用泛型容易,我们就经常使用List的泛型,但是如果我们要写一个泛型的类其实并不那么容易。

最简单的泛型

package io.ilss.advanced.generic;

/**

* className MyObject

* description MyObject

*

* @author feng

* @version 1.0

* @date 2019-01-24 18:32

*/

public class MyObject<T> extends BaseData {

    private T valueOne;

    private T valueTwo;

    public MyObject(T valueOne, T valueTwo) {

        this.valueOne = valueOne;

        this.valueTwo = valueTwo;

    }

    public T getValueOne() {

        return valueOne;

    }

    public void setValueOne(T valueOne) {

        this.valueOne = valueOne;

    }

    public T getValueTwo() {

        return valueTwo;

    }

    public void setValueTwo(T valueTwo) {

        this.valueTwo = valueTwo;

    }

    public static void main(String[] args) {

        MyObject<String> object = new MyObject<>("String one!", "String two");

        System.out.println("value one " + object.valueOne + " value two " + object.valueTwo);

    }

}

在MyObject中引入一个类型变量T,用尖括号<>括起来,放在类名的后面。如上!在定义的时候可以有多个类型的变量,在<>中以逗号,分隔例如public class MyObject<T, U>{…},在类中如果要使用类型 就直接把定义的类型变量看做类名使用即可。

用泛型做一个接口返回类

相信写Java的程序员都写过Web接口,那我们用泛型来封装一个统一响应返回的类ResponseMsg。

package io.ilss.advanced.generic;

/**

* className ResponseMsg

* description ResponseMsg

*

* @author feng

* @version 1.0

* @date 2019-01-24 18:47

*/

public class ResponseMsg<T extends BaseData> {

    public static int SUCCESS_CODE = 1;

    public static int ERROR_CODE = 0;

    public static int OTHER_CODE = -1;

    private int code;

    private String msg;

    private T data;

    public static <U extends BaseData> ResponseMsg sendSuccess(U data) {

        ResponseMsg<U> responseMsg = new ResponseMsg<>();

        responseMsg.code = SUCCESS_CODE;

        responseMsg.data = data;

        responseMsg.msg = "Remote Call Success!";

        return responseMsg;

    }

    public static <U extends BaseData> ResponseMsg sendError(U data, String msg) {

        ResponseMsg<U> responseMsg = new ResponseMsg<>();

        responseMsg.code = ERROR_CODE;

        responseMsg.data = data;

        responseMsg.msg = "Remote Call Error";

        return responseMsg;

    }

    public static <U extends BaseData> ResponseMsg sendOther(U data, String msg) {

        ResponseMsg<U> responseMsg = new ResponseMsg<>();

        responseMsg.code = OTHER_CODE;

        responseMsg.data = data;

        responseMsg.msg = msg;

        return responseMsg;

    }

    public static void main(String[] args) {

        System.out.println(ResponseMsg.<MyObject>sendSuccess(new MyObject<String>("asdf","asfd")));

    }

    @Override

    public String toString() {

        return "ResponseMsg{" +

                "code=" + code +

                ", msg='" + msg + '\'' +

                ", data=" + data +

                '}';

    }

}

上面的重点是上的<U>我利用静态方法封装了ResponseMsg的构建,只需要提供静态方法让调用的人传入一个类,也可以不用写Getter Setter方法,或者直接变成私有方法。

方法如果要用泛型,则只需要在定时的时候在返回类型前面加上<T>即可使用。调用的时候直接在调用方法前面用<>传入你想用的类即可。如上所示,也可以将方法用的泛型直接传入自己类定义的泛型。

此外还可以通过extends限定你是某某类的子类,或者实现了某个接口。如果有多个接口可以用&连接,比如<T extends Comparable & Serializable>,如果有多个泛型可以是这样<T extends OneObject, U extends TwoObject>

注意

泛型不能直接new,需要外部传入。

如:

T data = new T();    //这是不被

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 夜莺2517阅读 127,761评论 1 9
  • 版本:ios 1.2.1 亮点: 1.app角标可以实时更新天气温度或选择空气质量,建议处女座就不要选了,不然老想...
    我就是沉沉阅读 6,963评论 1 6
  • 我是黑夜里大雨纷飞的人啊 1 “又到一年六月,有人笑有人哭,有人欢乐有人忧愁,有人惊喜有人失落,有的觉得收获满满有...
    陌忘宇阅读 8,606评论 28 53
  • 兔子虽然是枚小硕 但学校的硕士四人寝不够 就被分到了博士楼里 两人一间 在学校的最西边 靠山 兔子的室友身体不好 ...
    待业的兔子阅读 2,652评论 2 9