[05]——简单接触泛型与反射

作者: junerver | 来源:发表于2016-11-12 17:18 被阅读123次

场景:存在着两个不同的实体类(ParentEntity、ChildEntity),这两个实体类中都有头像这一个属性,对应着服务器的图片保存位置。需要一个方法,可以获取这一属性,然后将图片下载保存到本地指定位置,同时将路径保存到该实体类,然后更新数据库。

因为需要执行的操作基本相同,都是调用实体类的相应方法获取图片保存地址,然后发起请求,保存到本地。由于存在着两个实体类,为了简化方法,理应使用泛型方法。

之前在学习Java的时候没有好好看过泛型,今天用到了不禁有点丈二和尚摸不着头脑。

翻阅一番资料后有了一定的了解,完成了这个方法的设计,记录如下:

不同于一般的方法声明,泛型方法需要声明成如下格式:
public <T> void func(T t){}

也就是需要在作用域关键字后边声明关键字<T>,这样方法才能接受一个泛型作为参数。

现在我们的方法可以接收任意类型的对象作为参数了,但是存在一个问题,那就是这个t对象是一个Object对象,如下图所示:

传入的泛型对象是一个Object

对于方法而言,方法不管传入的对象是什么类型的实例,只对传入的对象做统一处理。那么首先我们应该先知道,传入的对象到底是ParentEntity、ChildEntity中的哪个?

我们可以使用Class clazz = t.getClass();来获取传入对象的“类”,使用String clazzName=clazz.getName();可以得知传入的对象的类的名字,这样我们就可以判断传入的对象是不是我们需要的了。

if (clazzName.equals(ParentEntity.class.getName())) {   

 } else if (clazzName.equals(ChildEntity.class.getName())) {

 }

那我们怎么才能调用我们传入的实体的方法呢?当然我们可以直接使用强制转型,但是这样就不够优雅了。选择反射则优雅很多,网上很多关于反射的介绍,我就不多赘述了。

反射获取方法与调用:

try {
    Method method1 = clazz.getMethod("method1");
    Method method2 = clazz.getMethod("method1",String.class);
    method1.invoke(t);
    method2.invoke(t, "xxx");
} catch (NoSuchMethodException e) {
    e.printStackTrace();
} catch (InvocationTargetException e) {
    e.printStackTrace();
} catch (IllegalAccessException e) {
    e.printStackTrace();
}

注意这里会抛出NoSuchMethodException异常,表示无法通过反射获取这个方法。

反射获取的方法需要通过method.invoke(odj,obj....args)来调用,使用这个方法会抛出两个异常,InvocationTargetException、IllegalAccessException

完整代码如下:

    /**
     * @author Junerver
     * @param entity 幼儿或者家长实例
     * @param <T>
     */
    public <T> void syncAvatarToLocal(final T entity) {
        try {
            Class entityClass = entity.getClass();
            Method getAvatar = null;
            //获取本地路径方法、设置同步位方法
            final Method setLocalAvatarPath = entityClass.getMethod("setLocalAvatarPath", String.class);
            final Method setSync = entityClass.getMethod("setSync", boolean.class);
            //由于服务器字段名的问题不能使用统一方法名
            if (entityClass.getName().equals(ParentEntity.class.getName())) {
                getAvatar = entityClass.getMethod("getParentsavatar");
            } else if (entityClass.getName().equals(ChildEntity.class.getName())) {
                getAvatar = entityClass.getMethod("getAvatar");
            }
            if (getAvatar != null && setLocalAvatarPath != null && setSync != null) {
                String avatarPath = (String) getAvatar.invoke(entity);
                if (avatarPath.startsWith("/Public")) {
                    //拼接url
                    String avatarUrl = Constants.getServer() + avatarPath;
                    String picName = avatarPath.substring(avatarPath.lastIndexOf("/") + 1);
                    OkHttpUtils
                            .get()
                            .url(avatarUrl)
                            .build()
                            .execute(new FileCallBack(Constants.AVATAR_IMAGES_LOCATION, picName) {
                                @Override
                                public void onError(Call call, Exception e, int id) {
                                    // TODO: 2016/11/12
                                }

                                @Override
                                public void onResponse(File response, int id) {
                                    Log.e("NetApi", "onResponse :" + response.getAbsolutePath());
                                    try {
                                        //将本地文件名写入实例中,标记同步位
                                        setLocalAvatarPath.invoke(entity, response.getAbsolutePath());
                                        setSync.invoke(entity, true);
                                        if (entity instanceof ParentEntity)
                                            PunchApplication.sParentEntityDao.update((ParentEntity) entity);
                                        if (entity instanceof ChildEntity)
                                            PunchApplication.sChildEntityDao.update((ChildEntity) entity);
                                    } catch (IllegalAccessException e) {
                                        e.printStackTrace();
                                    } catch (InvocationTargetException e) {
                                        e.printStackTrace();
                                    }
                                }
                            });
                }
            }
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }

相关文章

  • [05]——简单接触泛型与反射

    场景:存在着两个不同的实体类(ParentEntity、ChildEntity),这两个实体类中都有头像这一个属性...

  • 泛型与反射

    引入泛型的目的 通过泛型使得在编译阶段完成一些类型转换的工作,避免在运行时强制类型转换而出现ClassCastEx...

  • 反射与泛型

    原文: https://lsy.iteye.com/blog/220264 反射是Java语言中很重要的一个组成部...

  • 泛型 & 注解 & Log4J日志组件

    掌握的知识 : 基本用法、泛型擦除、泛型类/泛型方法/泛型接口、泛型关键字、反射泛型(案例) 泛型 概述 : 泛型...

  • (转).NET面试题系列[8] - 泛型

    泛型相比反射,委托等较为抽象的概念要更接地气得多,而且在平常工作时,我们几乎时刻都和泛型有接触。大部分人对泛型都是...

  • 泛型

    与Java泛型相同,Kotlin同样提供了泛型支持。对于简单的泛型类、泛型函数的定义,Kotlin 与 Java ...

  • 泛型、型变与投影

    简单泛型 kotlin 对于简单泛型的支持与java类似, 可以通过通配符,提升代码的灵活度 限定型泛型 在编码实...

  • Java泛型

    本文介绍的知识点 泛型是什么? 泛型的使用在反射中使用泛型在集合类中使用泛型 关于泛型擦除如何理解?如何避免泛型擦...

  • 关于反射的使用

    反射中获取泛型参数信息

  • java注解与反射,泛型与反射

    一、反射与注解 内置注解 java内置了3种注解,用来为编译器提供检查。 自定义注解 元注解 元注解是用来修饰注解...

网友评论

    本文标题:[05]——简单接触泛型与反射

    本文链接:https://www.haomeiwen.com/subject/tqnfuttx.html