前言

代码是我师父的,代码是我师父的,代码是我师父的,如有需要拿走的时候请标注  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. dedecms代码研究五

    上一次留几个疑问: 1)DedeTagParse类LoadTemplet方法. 2)MakeOneTag到底在搞什么. 从DedeTagParse开始前面,我们一直在dedecms的外围,被各种全局变 ...

  2. php中好用的时间函数

    //查询数据30天的数据$y=date("Y",time());$m=date("m",time());$d=date("d",time() ...

  3. java学习第6天

    今天主要是学习下static静态变量的了解 ,主要是用于多个对象相同的成员变量,用以节省空间.它是随着类的加载而加载可以是方法也可以是对象.直接通过类名调用.比如main方法就是,可以直接调用. ma ...

  4. js分辨浏览器类别和版本

    function BrowserInfo() { var ua = navigator.userAgent.toLowerCase(); var Sys = {}; var s; (s = ua.ma ...

  5. Firefox 23中的新特性(新陷阱)

    话说有一天突然发现我们的网站页面上的JQuery功能都失效了,Firebug中显示如下的错误 Blocked loading mixed active content "http://xxx ...

  6. sqlserver无法连接

    以下是我的检查信息及结果:1.telnet 192.168.1.100 1433 通过  telnet 116.3.15.198 1433 不通,提示“……无法打开连接,连接失败”的错误.2.通过端口 ...

  7. node.js入门及express.js框架

    node.js介绍 javascript原本只是用来处理前端,Node使得javascript编写服务端程序成为可能.于是前端开发者也可以借此轻松进入后端开发领域.Node是基于Google的V8引擎 ...

  8. mysql convert

    SELECT id,boshidianshu,boshidianshu_shuzi,CONVERT(REPLACE(boshidianshu, '个', ''),SIGNED) aaa from lg ...

  9. 灭顶之灾之网络电视精灵——S2 2.8

    从前,有一个神奇的东西叫做搞搞精灵 关于他,有一段历史. 哎呀!我去!写不下去了. -.-以上玩笑 首先需求分析 TreeView显示两种频道 TypeA和TypeB 所以创建三个类 ChannelB ...

  10. UVA 753 A Plug for UNIX(二分图匹配)

    A Plug for UNIX You are in charge of setting up the press room for the inaugural meeting of the Unit ...