gson中new TypeToken>(){}

参考:
JAVA匿名内部类(Anonymous Classes)
gson中对构造方法TypeToken()的探究
Gson中TypeToken如何实现获取参数类型
情景引入:
在使用GSON解析一段JSON数组时,需要借助TypeToken将期望解析成的数据类型传入到fromJson()方法中,如下

   List mblist = gson.fromJson (indexInfoList, new TypeToken> () {
            }.getType ());

疑问:不清楚为什么JSON转换为对象的时候,new TypeToken()后面还要跟着一个大括号?
通常是通过 new 构造方法().方法名() 来调用某个类中的方法
但截图中使用了 new 构造方法(){}.方法名() ,构造方法后面多了一个大括号?

1. new TypeToken>(){}是一个匿名内部类

匿名类表达式包含以下内部分:
操作符:new;
一个要实现的接口或要继承的类,案例一中的匿名类实现了HellowWorld接口,案例二中的匿名内部类继承了Animal父类;
一对括号,如果是匿名子类,与实例化普通类的语法类似,如果有构造参数,要带上构造参数;如果是实现一个接口,只需要一对空括号即可;
一段被"{}"括起来类声明主体;
末尾的";"号(因为匿名类的声明是一个表达式,是语句的一部分,因此要以分号结尾)。

public class AnimalTest {

    private final String ANIMAL = "动物";

    public void accessTest() {
        System.out.println("匿名内部类访问其外部类方法");
    }

    class Animal {
        private String name;

        public Animal(String name) {
            this.name = name;
        }

        public void printAnimalName() {
            System.out.println(bird.name);
        }
    }

    // 鸟类,匿名子类,继承自Animal类,可以覆写父类方法
    Animal bird = new Animal("布谷鸟") {

        @Override
        public void printAnimalName() {
            accessTest();           // 访问外部类成员
            System.out.println(ANIMAL);  // 访问外部类final修饰的变量
            super.printAnimalName();
        }
    };

    public void print() {
        bird.printAnimalName();
    }

    public static void main(String[] args) {

        AnimalTest animalTest = new AnimalTest();
        animalTest.print();
    }
}

2. 问题探索

猜测这里使用了“匿名内部类”,但不明白为什么要这么做,于是做个实验,删除构造方法后面的{}


image.png

报错提示:‘TypeToken()’ has protected access in ‘com.google.gson.reflect.TypeToken’
关键词:protected

前往TypeToken这个类的源码处看一看:


image.png

构造方法TypeToken()被protected修饰,有如下特点:

protected
如果构造函数是protected,那么该类可以继承,可以在被包内其他类中产生实例,但是无法在包外或者子类以外的地方产生实例

划重点:如果构造函数是protected,无法在包外或者子类以外的地方产生实例

因此在使用构造方法TypeToken()进行实例化时,需要先通过匿名内部类继承TypeToken这个类,然后才能进行实例化,进而继续调用getType()方法。

new TypeToken>(){}是一个匿名内部类,其等价MyTypeToken> extends TypeToken(){}.

为什么要用protected来修饰构造方法TypeToken()呢?
为了拿持有泛型。protected修饰的构造方法,对于非同包需要先用一个类来继承父类才能new,然后通过继承的那个类来拿持有泛型。alibaba的fastjson里也有个类似的类,套路相似。

看代码:

 /**
   * Constructs a new type literal. Derives represented class from type
   * parameter.
   *
   * 

Clients create an empty anonymous subclass. Doing so embeds the type * parameter in the anonymous class's type hierarchy so we can reconstitute it * at runtime despite erasure. 构造一个新的类型文本。从类型派生表示的类参数。 * 客户端创建一个空的匿名子类。这样做会嵌入类型匿名类的类型层次结构中的参数,以便我们在运行时删除后可以重新构造它。 */ @SuppressWarnings("unchecked") protected TypeToken() { this.type = getSuperclassTypeParameter(getClass()); this.rawType = (Class) $Gson$Types.getRawType(type); this.hashCode = type.hashCode(); } /** * Returns the type from super class's type parameter in {@link $Gson$Types#canonicalize * canonical form}. 返回从规范化超类的类型参数 */ static Type getSuperclassTypeParameter(Class subclass) { //获取到子类的父类Type(/感觉是返回类 所有的参数信息 ??) Type superclass = subclass.getGenericSuperclass(); if (superclass instanceof Class) { throw new RuntimeException("Missing type parameter."); } ////将Type类型向下转型为参数化类型ParameterizedType ParameterizedType parameterized = (ParameterizedType) superclass; ////这里getActualTypeArguments()返回的是一个数组,由于只有一个泛型参数,直接[0]。 return $Gson$Types.canonicalize(parameterized.getActualTypeArguments()[0]); } Type: 其是一个接口java.lang.reflect.Type,主要有5类: raw types:一般类型,例如:String,Collections ,Math,Number… parameterized types : 参数化类型,例如:`List`集合中常用… array types : 数组类型 type variables :类型变量,不确定其类型,例如`List` primitive types : 基本类型,int,float… 详细参见:[http://blog.csdn.net/kaka123ac/article/details/4470813](http://blog.csdn.net/kaka123ac/article/details/4470813) getSuperClass() 与 getGenericSuperclass()区别: 前者,返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的超类的 Class(由于编译擦除,没有显示泛型参数:在运行期间,泛型参数类型一律为Object类型)。 后者,返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的直接超类的 Type(包含泛型参数)。 详细参见: [http://www.cnblogs.com/maokun/p/6773203.html](http://www.cnblogs.com/maokun/p/6773203.html)

你可能感兴趣的:(gson中new TypeToken>(){})