• java反射保存


    前言

    代码是我师父的,代码是我师父的,代码是我师父的,如有需要拿走的时候请标注  copyright by 山人Wu  记录这篇是为了加深理解,前段时间只是当做工具类来用,才有时间好好看一下,加深理解。

    背景

    项目中涉及到从excel表导入,很多模块都涉及到了,从excel里面拿到的值都是字符串,运用起来比较麻烦,所以师父写了一个工具类,可以将excel得到的数据直接映射成对象。得到的excel里面的值第一行是列标题,以下是数据,大概如下图所示↓

    代码

    新建一个类用来接受excel里面你想要的值,将你想要接受的字段设置Annotation属性,这里的名字一定要和excel里面的列标题一样,方法是匹配[类]中的某一列的Annotation和excel里面的列标题,所以双方都可以冗余。

    public class DEnterpriseForExcel {
    
        /**
         * 企业全称
         */
        @ExcelAnnotation(exportName = "FULLNAME")
        private String fullname;
        
        /**
         * 企业全称
         */
        private String test;
    }
    接收类
    @Retention(RetentionPolicy.RUNTIME)
    @Target(ElementType.FIELD)
    public @interface ExcelAnnotation {
        // excel导出时标题显示的名字,如果没有设置Annotation属性,将不会被导出和导入
        public String exportName();
    
    }
    添加字段名称

    如果有需要特殊处理的字段,类似日期之类的,可以通过注入方法来特殊处理那一列

    /**
     * 反射保存
     * 
     * @author wulin
     *
     */
    public class ReflectionUtils<T> {
        Class<T> clazz;
    
        private ReflectionUtilsCallback callback = null;
    
        public ReflectionUtils(Class<T> clazz) {
            this.clazz = clazz;
        }
    
        public void setCallback(ReflectionUtilsCallback callback) {
            this.callback = callback;
        }
    
        // 格式化日期
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");
    
        public List<T> stringsToClass(List<String[]> lstData) throws Exception {
            List<T> rtn = new ArrayList<T>();
            int colNum = 0;
            try {
                /**
                 * 类反射得到调用方法
                 */
                // 得到目标目标类的所有的字段列表
                Field filed[] = clazz.getDeclaredFields();
                // 将所有标有Annotation的字段,也就是允许导入数据的字段,放入到一个map中
                Map<String, Method> fieldmap = new HashMap<String, Method>();
    
                // 循环读取所有字段
                // 自定义Class中的所有变量 都要用 注释
                for (int i = 0; i < filed.length; i++) {
                    Field f = filed[i];
                    // 得到单个字段上的Annotation
    
                    ExcelAnnotation exa = f.getAnnotation(ExcelAnnotation.class);
                    // 如果标识了Annotationd的话
                    if (exa != null) {
                        // 构造设置了Annotation的字段的Setter方法
                        String fieldname = f.getName();
                        String setMethodName = "set" + fieldname.substring(0, 1).toUpperCase() + fieldname.substring(1);
                        // 构造调用的method,
                        Method setMethod = clazz.getMethod(setMethodName, new Class[] { f.getType() });
                        // 将这个method以Annotaion的名字为key来存入。
                        fieldmap.put(exa.exportName(), setMethod);
                    }
                }
    
                // 将标题的文字内容放入到一个map中。
                Map<Integer, String> titlemap = new HashMap<Integer, String>();
                // 第一行是标题
                String[] dataTitle = lstData.get(0);
                for (int i = 0; i < dataTitle.length; i++) {
                    String value = dataTitle[i].replace(""", "").trim();
                    titlemap.put(i, value);
                }
    
                // 从第2行往下是数据区
                for (int nLp = 1; nLp < lstData.size(); nLp++) {
    
                    // 标题下的第一行
                    String[] row = lstData.get(nLp);
                    // 行的所有列
                    // 得到传入类的实例
                    T tObject = clazz.newInstance();
    
                    int k = 0;
                    int nFlushCnt = 0;
                    // 列遍历(遍历一行的所有列)
                    colNum = 0;
                    for (String rec : row) {
    
                        if (nFlushCnt >= 1000) {
                            System.gc();
                        }
                        colNum++;
    
                        // 取得-列标题
                        String titleString = (String) titlemap.get(k);
                        // 如果[列标题]和[类]中的某一列的Annotation相同,那么则调用此类的的set方法,进行设值
                        if (fieldmap.containsKey(titleString)) {
                            // 获得 set方法
                            Method setMethod = (Method) fieldmap.get(titleString);
    
                            if (null != callback) {
                                if (callback.isCallbackColumn(colNum)) {
                                    Object objVal = callback.getCellValue(colNum, rec);
                                    setMethod.invoke(tObject, objVal);
                                    k = k + 1;
                                    continue;
                                }
                            }
    
                            // 得到setter方法的参数
                            Type[] ts = setMethod.getGenericParameterTypes();
                            // 获得参数类型[java.lang.String]
                            String xclass = ts[0].toString();
    
                            if (xclass.equals("class java.lang.String")) {
                                setMethod.invoke(tObject, rec == null ? null : rec.replace(""", ""));
    
                            } else if (xclass.equals("class java.lang.Integer")) {
    
                                setMethod.invoke(tObject,
                                        (rec == null || " ".equals(rec)) ? null : Integer.parseInt(rec.replace(""", "")));
                            } else if (xclass.equals("class java.math.BigDecimal")) {
    
                                setMethod.invoke(tObject,
                                        (rec == null || " ".equals(rec)) ? null : new BigDecimal(rec.replace(""", "")));
                            } else if (xclass.equals("int")) {
    
                                int temp = Integer
                                        .parseInt((rec == null || " ".equals(rec)) ? null : rec.replace(""", ""));
                                setMethod.invoke(tObject, temp);
                            } else if (xclass.equals("class java.util.Date")) {
    
                                setMethod.invoke(tObject,
                                        (rec == null || 0 == rec.length()) ? null : sdf.parse(rec.replace(""", "")));
                            } else if (xclass.equals("class java.lang.Short")) {
                                setMethod.invoke(tObject,
                                        (rec == null || " ".equals(rec)) ? null : Short.parseShort(rec.replace(""", "")));
                            }
    
                        }
                        // 下一列
                        k = k + 1;
                    }
                    rtn.add(tObject);
    
                }
    
            } catch (Exception e) {
                //System.out.println("colNum::" + colNum);
                e.printStackTrace();
                // 将异常抛出去
                throw e;
            }
    
            return rtn;
        }
    }
    工具类
    public interface ReflectionUtilsCallback {
        public boolean isCallbackColumn(int column);
        public Object getCellValue(int column, String strVal);
    }
    特殊处理
  • 相关阅读:
    判断activity是否显示在界面上
    限制EditText的输入字数
    安卓自定义类似TabHost的导航栏
    安卓中加载布局文件的三种方法
    绘图——Android绘图基础:Canvas、Paint等
    使用简单图片
    使用原始资源
    MySQL分表(Partition)学习研究报告
    Docker基础知识介绍
    Python开发系列
  • 原文地址:https://www.cnblogs.com/chanmao--/p/5893353.html
Copyright © 2020-2023  润新知