HashMap--comparableClassFor方法及Type体系

HashMap在存储树化过程中,通过比较key的hash值来决定作为左子树还是右子树。hash值大于当前节点的hash值时作为当前节点的右节点;小于时作为左节点;相等时看key是否实现了Comparable接口,决定能够基于compareTo 比较,如果仍未比较出大小,就需要进行仲裁了,仲裁方法为 tieBreakOrder。
看该元素键是否实现了Comparable接口,用到了comparableClassFor方法来获取该元素键的Class。

static Class<?> comparableClassFor(Object x) {
//  先判断是否是Comparable类型的
    if (x instanceof Comparable) {
        Class<?> c; Type[] ts, as; Type t; ParameterizedType p;
        //  对String类型特判,如果是String类型直接返回该对象的Class类
                //  所以大多数人都建议在使用HashMap时key使用String类型
        if ((c = x.getClass()) == String.class) // bypass checks
            return c;
        //  获取该类实现的接口集,包含泛型参数信息
        if ((ts = c.getGenericInterfaces()) != null) {
            for (int i = 0; i < ts.length; ++i) {
               //判断是否实现了泛型接口
                if (((t = ts[i]) instanceof ParameterizedType) &&
                //判断承载泛型接口是否是Comparable
                    ((p = (ParameterizedType)t).getRawType() ==
                     Comparable.class) &&
                //判断泛型实际类型列表是否只有一个且是key的class类型
                    (as = p.getActualTypeArguments()) != null &&
                    as.length == 1 && as[0] == c) // type arg is c
                    return c;
            }
        }
    }
    return null;
}


Type是个空接口,没有定义任何方法,通过多态提高了程序的扩展性,具体实现去看下面的子类;


                                                  Type体系

Type是Java 编程语言中所有类型的公共高级接口(官方解释),也就是Java中所有类型的抽象;其中,“所有类型”的描述尤为值得关注。它并不是我们平常工作中经常使用的 int、String、List、Map等数据类型,而是从Java语言角度来说,对基本类型、引用类型向上的抽象;
Type体系中类型的包括:原始类型(Class)、参数化类型(ParameterizedType)、数组类型(GenericArrayType)、类型变量(TypeVariable)、基本类型(Class);
原始类型,不仅仅包含我们平常所指的类,还包括枚举、数组、注解等;
参数化类型,就是我们平常所用到的泛型List、Map;
数组类型,并不是我们工作中所使用的数组String[] 、byte[],而是带有泛型的数组,即T[] ;
基本类型,也就是我们所说的java的基本类型,即int,float,double等
1.ParameterizedType
参数化类型,即泛型;例如:List<T>、Map<K,V>等带有参数化的对象;

在ParameterizedType接口中,有3个方法,分别是getActualTypeArguments()、 getRawType()、 getOwnerType();
1.1 getActualTypeArguments
获取泛型中的实际类型,可能会存在多个泛型,例如Map<K,V>,所以会返回Type[]数组;

值得注意的是,无论<>中有几层嵌套(List<Map<String,Integer>),getActualTypeArguments()方法永远都是脱去最外层的<>(也就是List<>),将口号内的内容(Map<String,Integer>)返回;
我们经常遇到的List<T>,通过getActualTypeArguments()方法,得到的返回值是TypeVariableImpl对象,也就是TypeVariable类型;
1.2 getRawType
获取声明泛型的类或者接口,也就是泛型中<>前面的那个值;

1.3 getOwnerType
通过方法的名称,我们大概了解到,此方法是获取泛型的拥有者,那么拥有者是个什么意思?
Returns a {@code Type} object representing the type that this type    * is a member of.  For example, if this type is {@code O.I},    * return a representation of {@code O}.  (摘自JDK注释)
通过注解,我们得知,“拥有者”表示的含义--内部类的“父类”,通过getOwnerType()方法可以获取到内部类的“拥有者”;例如: Map  就是 Map.Entry<String,String>的拥有者;


2.TypeVariable
类型变量,即泛型中的变量;例如:T、K、V等变量,可以表示任何类;在这需要强调的是,TypeVariable代表着泛型中的变量,而ParameterizedType则代表整个泛型;泛型的类型变量,指的是List<T>、Map<K,V>中的T,K,V等值,实际的Java类型是TypeVariableImpl(TypeVariable的子类);此外,还可以对类型变量加上extend限定,这样会有类型变量对应的上限;

2.1 getBounds
获得该类型变量的上限,也就是泛型中extend右边的值;例如 List<T extends Number> ,Number就是类型变量T的上限;如果我们只是简单的声明了List<T>(无显式定义extends),那么默认为Object;

无显式定义extends:

值得注意的是,类型变量的上限可以为多个,必须使用&符号相连接,例如 List<T extends Number & Serializable>;其中,& 后必须为接口;
2.2 getGenericDeclaration
获取声明该类型变量实体,也就是TypeVariableTest<T>中的TypeVariableTest;

2.3 getName
获取类型变量在源码中定义的名称;

说到TypeVariable类,就不得不提及Java-Type体系中另一个比较重要的接口---GenericDeclaration;含义为:声明类型变量的所有实体的公共接口;也就是说该接口定义了哪些地方可以定义类型变量(泛型);
通过查看源码发现,GenericDeclaration下有三个子类,分别为Class、Method、Constructor;也就是说,我们定义泛型只能在一个类中这3个地方自定义泛型;

此时,我们不禁要问,我们不是经常在类中的属性声明泛型吗,怎么Field没有实现 GenericDeclaration接口呢?
其实,我们在Field中并没有声明泛型,而是在使用泛型而已!不信,我们实际上代码来看看!
1.首先在Class上定义泛型:

2.我们没有在Class上定义泛型,直接在构造方法上定义泛型

3.同样没有在Class定义泛型,直接在普通方法上定义泛型

4.我们直接在属性上定义

我们看到,如果不在Class上定义,属性上并不能直接使用!所以,这也是我之前说的属性上并不是定义泛型,而是使用泛型,所以Field并没有实现GenericDeclaration接口!

3.GenericArrayType
泛型数组类型,用来描述ParameterizedType、TypeVariable类型的数组;即List<T>[] 、T[]等;

3.1 getGenericComponentType
返回泛型数组中元素的Type类型,即List<String>[] 中的 List<String>(ParameterizedTypeImpl)、T[] 中的T(TypeVariableImpl);     
无论是几维数组,getGenericComponentType()方法都只会脱去最右边的[],返回剩下的值;
4.Class
上三者不同,Class是Type的一个实现类,属于原始类型,是Java反射的基础,对Java类的抽象;
在程序运行期间,每一个类都对应一个Class对象,这个对象保存有这个类的全部信息包含了类的修饰符、方法,属性、构造等,所以我们可以对这个Class对象进行相应的操作,这就是Java的反射;

当我们没有声明泛型的时候,我们普通的对象就是一个Class类型,是Type中的一种;
5.WildcardType
泛型表达式(或者通配符表达式),即? extend Number、? super Integer这样的表达式;WildcardType虽然是Type的子接口,但却不是Java类型中的一种;

5.1 getUpperBounds
获取泛型变量的上边界(extends)

5.2 getLowerBounds
获取泛型变量的下边界(super)


以上,简单介绍了Java-Type的体系;为了解决泛型,JDK1.5版本开始引入Type接口;在此之前,Java中只有原始类型,所有的原始类型都是通过Class进行抽象;有了Type以后,Java的数据类型得到了扩展,从原始类型扩展为参数化类型(ParameterizedType)、数组类型(GenericArrayType)、类型变量(TypeVariable);

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值