前言

代码是我师父的,代码是我师父的,代码是我师父的,如有需要拿走的时候请标注  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);
}

特殊处理

java反射保存的更多相关文章

  1. java 反射的应用 以及通过反射 用到的工厂模式

    java反射详解 本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解. 下面开始正文. [案 ...

  2. JAVA反射实践

    Java反射机制在我的理解当中就是下面几点: 1. 对一个给定的类名(以字符串形式提供)能动态构建一个对象实例 2. 对于任意一个类,都能够知道这个类的所有属性和方法     3. 对于任意一个对象, ...

  3. [转]java反射机制

    原文地址:http://www.cnblogs.com/jqyp/archive/2012/03/29/2423112.html 一.什么是反射机制         简单的来说,反射机制指的是程序在运 ...

  4. Java反射机制的学习

    Java反射机制是Java语言被视为准动态语言的关键性质.Java反射机制的核心就是允许在运行时通过Java Reflection APIs来取得已知名字的class类的相关信息,动态地生成此类,并调 ...

  5. java反射机制浅谈

    一.Java的反射机制浅谈 最近研究java研究得很给力,主要以看博文为学习方式.以下是我对java的反射机制所产生的一些感悟,希望各位童鞋看到失误之处不吝指出.受到各位指教之处,如若让小生好好感动, ...

  6. JAVA反射机制学�

    JAVA反射机制:对于随意一个类,都可以知道这个类的全部属性和方法:对于随意一个对象,都可以调用它的随意一个方法和属性:这样的动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. J ...

  7. 【设计模式】学习笔记17:代理模式之保护代理与Java反射

    本文出自   http://blog.csdn.net/shuangde800 本笔记内容: 1. Java动态代理,反射机制 2. 保护代理 3. 应用保护代理实现的约会系统 ----------- ...

  8. java 反射(reflect)总结,附对象打印工具类

    java反射机制认知 java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取类的信息以及动态调用对象的方法的 ...

  9. 再回首,Java温故知新(十一):Java反射

    最近继续回顾Java基础,进行到了Java反射这一部分,个人感觉这部分应该算是Java的高级特性了,在日常开发中使用的并不多,应用人员主要是工具构建人员,所以这次学习中以了解为主,Java反射主要应用 ...

随机推荐

  1. Inside The C++ Object Model - 03

    object Lessons 1.C++中布局以及存取时间上的的额外负担是由virtual引起的:virtual function.virtual base class.或是由于多继承引起的. 2.C ...

  2. Python自动化 【第四篇】:Python基础-装饰器 生成器 迭代器 Json & pickle

    目录: 装饰器 生成器 迭代器 Json & pickle 数据序列化 软件目录结构规范 1. Python装饰器 装饰器:本质是函数,(功能是装饰其它函数)就是为其他函数添加附加功能 原则: ...

  3. information_schema系列之字符集校验(CHARACTER_SETS,COLLATIONS,COLLATION_CHARACTER_SET_APPLICABILITY)

    1:CHARACTER_SETS 首先看一下查询前十条的结果: root@localhost [information_schema]>select * from CHARACTER_SETS ...

  4. 为窗体设置背景图片-UI界面编辑器(SkinStudio)教程

    1.1.   为窗体设置背景图片 在窗体的Background属性中选择图片设置为窗体背景图片

  5. cs0006 未能找到元数据文件 C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\Temporary ASP.NET Files

    翻阅了一些资料后发现是需要重新注册IIS服务扩展,在“开始”-“运行”里输入如入命令,回车,搞定 C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\aspne ...

  6. (转载)iOS系统Crash文件分析方法

    转自: http://ios-iphone.diandian.com/post/2012-05-18/19440182 Xcode 4.3的symbolicatecrash的位置和老版本的不一致了. ...

  7. 转 苹果的新编程语言 Swift 简介

    苹果官方文档地址 https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Pro ...

  8. 创建solr集群简述

    综述: 用两台服务器,每台服务器上启动两个solr实例(端口分别为8983.7574),即一共有2x2=4个节点.4个节点分散在两个分片上,每台机器上存放两个分片的各一个replica,这样等于每台机 ...

  9. 关于ajax提交表单参数序列化和时间戳转换

    ajax提交form表单, 序列化表单的参数 //var a = $("#addfm").serialize(); //将表单的内容序列化成为一个字符串 var a = $(&qu ...

  10. DB2不记录事务日志

    1. DB2大数据处理不记录事务日志步骤:  建表需要添加属性“NOT LOGGED INITIALLY”  在大批量更改操作的同一个事务开始时执行:“ALTER TABLE tabname ACTI ...