Java防止范型擦除的方法

防止范型擦除的方法

前言

java的范型我最喜欢的东西, 他可以把代码变得更精简, 但是也可能会是代码中的一个小陷阱.

如果让我去比喻一下范型在java是一个什么角色.

如果java项目工程是一个小区(爪哇小区), 那各种class 可能就是小区的住户了. 范型就是这个小区的门卫大爷. 他可以拦住那些看起来就是不法分子的入侵者. 但是还是有办法偷偷进入小区, 或者像外卖小哥进入小区.

作为小区的“物业”的我, 应该对范型的行为有所掌握, 至少出了问题, 我有方向可寻.

范型举例

// 这可能是我最常用的范型应用场景了
List<String> list = new HashList<>(15);

范型擦除场景

故名思义, 范型被干掉了.
范型擦出发生在什么阶段呢??

背景设置

这里有两个class

class A {
    @Override
    public String toString() {
        return "A";
    }
}

class B {
    @Override
    public String toString() {
        return "B";
    }
}
class C {
    @Override
    public String toString() {
        return "C";
    }
}
class D {
    @Override
    public String toString() {
        return "D";
    }
}

以及2个范型使用的class

public class Test<T> {
    T obj;
    Test(T obj) {
        this.obj = obj;
    }
    Test() {}
    public void setObj(Object obj) {
        // 注意该方法的区别
        this.obj = (T)obj;
    }
    @Override
    public String toString() {
        return obj.toString();
    }
}
public class Test2<T> {
    T obj;
    Test(T obj) {
        this.obj = obj;
    }
    Test2() {}
    
    public void setObj(T obj) {
        // 注意该方法的区别
        this.obj = obj;
    }
    @Override
    public String toString() {
        return obj.toString();
    }
}

以及一个测试输出方法

public static void show(String mark, Object o) {
    System.out.println("[ " + mark + " ] " + o);
}

场景1-被忽略的范型

注意方法

B b = new B();
A a = new A();

Test<A> test1 = new Test<>();
test1.setObj(b);
show("1", test1);
Test<A> test2 = new Test<>(a);
test2.setObj(b);
show("2", test2);

结果

[ 1 ] B
[ 2 ] B

编译通过了也没有出现了报错, Test<A> 范型我也写了, 但是B还是被set进去.
与预想中 可能会报错的地方(T)obj;
可以看出 这个过程中并未发生 B->A 的强制转换.
(原因: 发现在编译后的文件被擦除, 该处(T)为 (Object), 验证在后面)

场景2-常规操作

B b = new B();
A a = new A();

Test2<A> test3 = new Test2<>();
/// 编译不通过
// test3.setObj(b);

Test2<A> test4 = new Test2<>(a);
/// 编译不通过
// test4.setObj(b);

以上两种情况均出现了编译不通过的情况. 这就是我前言所说看起来就是不法分子的入侵者
那么有办法, 越过这道防线嘛??
答案是可以的(场景3).

场景3-反射操作

B b = new B();
A a = new A();
Test2<A> test6 = new Test2<>(a);

/// 这一步会发现 这样是找不到方法的
/// Exception in thread "main" java.lang.NoSuchMethodException: cn.net.bale.demo.reflect.Test2.setObj(cn.net.bale.demo.reflect.ModifiedClass$A)
// Test2.class.getMethod("setObj", A.class);

Method method = Test2.class.getMethod("setObj", Object.class);
method.setAccessible(true);
method.invoke(test6, b);
show("6", test6);

输出结果

[ 6 ] B

这种方式就非常暴力的用反射把调用 setObj 方法把 b放了进去.
但是我们可以看到, 反射的是getMethod并不是通过 getMethod("setObj", A.class)
(进一步佐证, 范型在编译过后被擦除, 验证在后面)

这种方式太暴力, 一般的场景用不到, 那么什么方式能“委婉”一些呢?
见 场景4

场景4-多次范型擦除

引入工具类

public class ObjectUtil {
    private ObjectUtil() {}
    /**
     * 类型转换
     * WARNING: 确保强制转换正确无误方可使用
     *
     * @param object
     * @param <T>
     * @return (T) object
     */
    @SuppressWarnings("unchecked")
    public static <T> T cast(Object object) {
        return (T) object;
    }
}

使用场景

B b = new B();
A a = new A();
Test2<A> test5 = new Test2<>(a);
test5.setObj(ObjectUtil.cast(b));
show("5", test5);

输出结果

[ 5 ] B

我们可以看到, 没有编译出错, 也没有使用反射. 但是就这样, 一路通行的进入了“爪哇小区”??
这个方法在编译的过程中发生了 2次范型擦除.

  1. Test2方法 中的T都被擦除变为 Object
  2. cast()方法中发生了一次范型擦除 变为 return Object.

b 首先进入 cast方法成为Object 伪装自己. 然后两次范型擦除 cast()与setObj()打通, 让b成功被set.

这个过程中, 没有发生我们想象中的, B -> A 发生强制转换报出运行时异常.
(如果没有发生范型擦除, 那么我们设想的“强转报错”的情况就会发生, 但是事实与设想截然相反)

场景5-创建实例

该场景下, 来验证一下, 范型信息会不会存储在object中.

引入输出方法

/**
     * 输出 obj.class
     *
     * @param mark 标记区分
     * @param test
     */
    public static void showObjClass(String mark, Test test) {
        try {
            System.out.println("[ " + mark + " ]Test.class: " + test.getClass());
            System.out.println("[ " + mark + " ]toString: " + test);
            System.out.println("[ " + mark + " ]Test.obj.class: " + test.obj.getClass());
        } catch (Exception e) {
            System.out.println("[ " + mark + " ]Test.obj.class: " + e.getMessage());
        } finally {
            System.out.println();
        }
    }

场景应用

Test test1 = new Test<>("bale");
Test test2 = new Test<>(1);
Test test3 = new Test();
// 强制转换一下
Test test4 = new Test((String) null);
Test.showObjClass("1", test1);
Test.showObjClass("2", test2);
Test.showObjClass("3", test3);
System.out.println("Test1.class == Test2.class: " + test1.getClass().equals(test2.getClass()));

输出结果

[ 1 ]Test.class: class cn.net.bale.demo.reflect.Test
[ 1 ]toString: bale
[ 1 ]Test.obj.class: class java.lang.String

[ 2 ]Test.class: class cn.net.bale.demo.reflect.Test
[ 2 ]toString: 1
[ 2 ]Test.obj.class: class java.lang.Integer

[ 3 ]Test.class: class cn.net.bale.demo.reflect.Test
[ 3 ]Test.obj.class: null

Test1.class == Test2.class: true

从最后的结果看到, Test1.class == Test2.class: true.
Test.class 无论范型是否相同, 但是class都是相同的.

范型擦除验证

通过以上场景, 可以看出, 范型在编译之后的class似乎就不起作用了.
通过反编译来看一下class是否和我们的场景保持一致, 还是在什么地方发生了奇妙的反应, 让范型被擦除.

引入命令

// javap 反编译输出class
javap -v ClassName.class

class 结构参考如下

// todo jvm文档在补ing

验证1-范型类

Test.class 反编译之后筛选出有用的信息

#9 = Utf8               TT;
#19 = Utf8               (TT;)V
// ... 基本参数与常量池省略 ...

{
  T obj;
    descriptor: Ljava/lang/Object;
    flags: (0x0000)
    Signature: #9                           // TT;

  cn.net.bale.demo.reflect.Test2(T);
    descriptor: (Ljava/lang/Object;)V
    flags: (0x0000)
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: aload_0
         5: aload_1
         6: putfield      #2                  // Field obj:Ljava/lang/Object;
         9: return
      // 省略临时变量区
            0      10     1   obj   TT;
    Signature: #19                          // (TT;)V

  // 省略 无参数构造器

  public void setObj(T);
    descriptor: (Ljava/lang/Object;)V
    flags: (0x0001) ACC_PUBLIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0
         1: aload_1
         2: putfield      #2                  // Field obj:Ljava/lang/Object;
         5: return
      LineNumberTable:
        line 19: 0
        line 20: 5
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0       6     0  this   Lcn/net/bale/demo/reflect/Test2;
            0       6     1   obj   Ljava/lang/Object;
      LocalVariableTypeTable:
        Start  Length  Slot  Name   Signature
            0       6     0  this   Lcn/net/bale/demo/reflect/Test2<TT;>;
            0       6     1   obj   TT;
    Signature: #19                          // (TT;)V

  // 省略 toString方法
}
Signature: #24                          // <T:Ljava/lang/Object;>Ljava/lang/Object;
SourceFile: "Test2.java"

我们通过反编译结果可以看到 所有范型出现的地方 都是 Ljava/lang/Object;

T obj;
    descriptor: Ljava/lang/Object; // Object属性
    flags: (0x0000)
    Signature: #9 // TT;
    
cn.net.bale.demo.reflect.Test2(T);  
    descriptor: (Ljava/lang/Object;)V // Object参数无返回值的构造器方法
    Signature: #19 // (TT;)V
    
public void setObj(T);  
    descriptor: (Ljava/lang/Object;)V // Object参数无返回值的setObject方法
    Signature: #19 // (TT;)V

可以看出, 编译后的结果, 所有范型的位置都被Object擦除

验证2-调用范型

引入class用于反编译观察Test<T>的调用情况

public class NewClass {
    public static void main(String[] args) {
        Test test1 = new Test<>("bale");
        Test test2 = new Test<>(1);
        Test test3 = new Test();
    }
}

反编并省略无关紧要的信息

// 省略基础信息以及常量池
{
  // 省略构造器
  public static void main(java.lang.String[]);
    descriptor: ([Ljava/lang/String;)V
    flags: (0x0009) ACC_PUBLIC, ACC_STATIC
    Code:
      stack=3, locals=4, args_size=1
         0: new           #2                  // class cn/net/bale/demo/reflect/Test
         3: dup
         4: ldc           #3                  // String bale
         6: invokespecial #4                  // Method cn/net/bale/demo/reflect/Test."<init>":(Ljava/lang/Object;)V
         9: astore_1
        10: new           #2                  // class cn/net/bale/demo/reflect/Test
        13: dup
        14: iconst_1
        15: invokestatic  #5                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
        18: invokespecial #4                  // Method cn/net/bale/demo/reflect/Test."<init>":(Ljava/lang/Object;)V
        21: astore_2
        22: new           #2                  // class cn/net/bale/demo/reflect/Test
        25: dup
        26: invokespecial #6                  // Method cn/net/bale/demo/reflect/Test."<init>":()V
        29: astore_3
        30: return
      // ...省略...
}
SourceFile: "NewClass.java"

重点关注内容如下


// 调用1, 直接调用 Test构造器, 参数为Object
6: invokespecial #4                  // Method cn/net/bale/demo/reflect/Test."<init>":(Ljava/lang/Object;)V

// 调用 valueOf 将数字包装成包装类
15: invokestatic  #5                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
// 调用2, 直接调用 Test构造器, 参数为Object
18: invokespecial #4                  // Method cn/net/bale/demo/reflect/Test."<init>":(Ljava/lang/Object;)V

可以看到调用方法使用范型的部分也都被擦除为Object
(感兴趣的同学可以观察 场景4cast()方法的范型擦除情况, 不过也都大同小异)

范型防止擦除的应用

通过以上的场景观察, 以及反编译验证可以了解到范型在编译之后会被擦除, 那如和防止这种情况的发生呢?
换一种说法, 范型我们如何去使用它, 就目前来看, 范型仅仅做到类型限制的作用, 如何让范型变得很灵活呢??
(要不我凭什么喜欢范型)
如果只做到这一步, 岂不是直接用Object也没什么问题, 只要开发的时候注释写好一些不就可以了??
不是这样的的, 见 应用1

应用1-extends指定擦除类型

使用 extend 关键字来指定擦除类

在不使用extends的时候, 默认使用 Object 作为擦除类(毕竟 超类是万能的)

引入class

class C {
}
public class Test3<T extends C> {
    T obj;
}

反编译结果

// 省略基础信息和常量池
{
  T obj;
    descriptor: Lcn/net/bale/demo/reflect/C;
    flags: (0x0000)
    Signature: #7                           // TT;

  // 省略构造器
}
Signature: #17                          // <T:Lcn/net/bale/demo/reflect/C;>Ljava/lang/Object;
SourceFile: "Test3.java"

结果很明显, extends 关键字之后的范型擦除的位置不在是 Ljava/lang/Object; 而变为指定的 C

应用2-extend保存范型

!!! 该处extends 不同于 应用1
此处为class的 extend继承

此处应用场景最为广泛且实用

应用如下

public class ExtendTest extends Test<C>{
}

此处通过 extends Test<C> 将范型的信息以 标签的形式存储在 ExtendTest中

那么如何使用它?

在普通的情况下 Who extends Test<C> 那么在Who 中一定就知道C是谁. 如何灵活使用这个关系呢?

引入帮助接口

package cn.net.bale.demo.reflect;

import cn.net.bale.demo.util.ObjectUtil;

import java.lang.reflect.ParameterizedType;

/**
 * TClass, 用于辅助范型获取 class
 *
 * @author bale 2019-06-11
 */
public interface Clazz<T> {
    /**
     * 获取 ParameterizedType 辅助获取 T class
     *
     * @return ParameterizedType
     */
    default ParameterizedType getParameterizedType() {
        return ObjectUtil.cast(getClass().getGenericSuperclass());
    }
    /**
     * 获取 T class
     *
     * @return Class<T>
     */
    default Class<T> getTClass() {
        // [0]为<T> 复杂情况, 按需求改写
       return ObjectUtil.cast(getParameterizedType().getActualTypeArguments()[0]);
    }
}

假设一个场景, 对mongoTemplate 二次封装, 每一个mongoCollection都对应一个 class entity
这样我们可以通过 class entity的属性名,生成通用的查询方法

代码如下
引入辅助注解

/**
 * Mongo json集(表)
 *
 * @author wentao.liu01@hand-china.com 2019-05-22
 */
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
public @interface MongoCollection {
    String name() default "";
}
public abstract class BaseMongoRepositoryImpl<T> implements Clazz<T> {
    // 这里可以根据情况改写 spring 中可以 @Autowired
    private final MongoTemplate mongoTemplate;
    /**
     * 获取 mongo 集合 name
     *
     * @return String
     */
    protected String getCollectionName() {
        Class<T> tClass = getTClass();
        MongoCollection mongoCollection = tClass.getAnnotation(MongoCollection.class);
        if (Objects.nonNull(mongoCollection)) {
            return mongoCollection.name();
        }
        return null;
    }
     /**
     * mongo find 
     * (封装一个最简单的find方法, 更多的黑科技可以在这里使用)
     * @param query
     * @return
     */
    public List<T> find(final Query query) {
        List<T> list = this.mongoTemplate.find(query, getTClass(), getCollectionName());
        setMulFieldFunction(list);
        return list;
    }
}
对应接口 
public interface BaseMongoRepositoryI<T> {
    List<T> find(final Query query);
}

使用情况:

entity:

@MongoCollection(name = "集合名称")
public class ClassName {
    // ...
}

impl:

public class Impl extends BaseMongoRepositoryImpl<ClassName> implements BaseMongoRepository {
}

使用:

Impl impl = 想办法获取到 Impl
Query query= new Query();
impl.find(Query)

这里已经实现最简单的版本, 让mongoTempalte 自带collectionName, 而不用每次查询都手动指定collectionName

还有更多的功能, 比如多语言, 根据属性自动生成查询方法等通用的功能, 来方便接下来的开发.

总结

范型是一个好东西, 了解范型的使用与擦除, 那么在编写通用的代码的时候是非常有意义的.
并且可以极大的缩短代码的行数, 让代码更有效.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351