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);
}
特殊处理
java反射保存的更多相关文章
- java 反射的应用 以及通过反射 用到的工厂模式
java反射详解 本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解. 下面开始正文. [案 ...
- JAVA反射实践
Java反射机制在我的理解当中就是下面几点: 1. 对一个给定的类名(以字符串形式提供)能动态构建一个对象实例 2. 对于任意一个类,都能够知道这个类的所有属性和方法 3. 对于任意一个对象, ...
- [转]java反射机制
原文地址:http://www.cnblogs.com/jqyp/archive/2012/03/29/2423112.html 一.什么是反射机制 简单的来说,反射机制指的是程序在运 ...
- Java反射机制的学习
Java反射机制是Java语言被视为准动态语言的关键性质.Java反射机制的核心就是允许在运行时通过Java Reflection APIs来取得已知名字的class类的相关信息,动态地生成此类,并调 ...
- java反射机制浅谈
一.Java的反射机制浅谈 最近研究java研究得很给力,主要以看博文为学习方式.以下是我对java的反射机制所产生的一些感悟,希望各位童鞋看到失误之处不吝指出.受到各位指教之处,如若让小生好好感动, ...
- JAVA反射机制学�
JAVA反射机制:对于随意一个类,都可以知道这个类的全部属性和方法:对于随意一个对象,都可以调用它的随意一个方法和属性:这样的动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制. J ...
- 【设计模式】学习笔记17:代理模式之保护代理与Java反射
本文出自 http://blog.csdn.net/shuangde800 本笔记内容: 1. Java动态代理,反射机制 2. 保护代理 3. 应用保护代理实现的约会系统 ----------- ...
- java 反射(reflect)总结,附对象打印工具类
java反射机制认知 java反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法和属性:这种动态获取类的信息以及动态调用对象的方法的 ...
- 再回首,Java温故知新(十一):Java反射
最近继续回顾Java基础,进行到了Java反射这一部分,个人感觉这部分应该算是Java的高级特性了,在日常开发中使用的并不多,应用人员主要是工具构建人员,所以这次学习中以了解为主,Java反射主要应用 ...
随机推荐
- git branch 管理常用命令
查看本地分支 git branch * dev master *代表当前位于dev分支 查看远程分支 git branch --remote origin/dev origin/master 查看远程 ...
- SQL Server 磁盘空间告急(磁盘扩容)转载
一.背景 在线上系统中,如果我们发现存放数据库文件的磁盘空间不够,我们应该怎么办呢?新买一个硬盘挂载上去可以嘛?(linux下可以直接挂载硬盘进行扩容),但是我们的SQL Server是运行在Wind ...
- javascript对象属性的赋值解析
代码如下: function Animal(){} function Dog (age){ this.name = 'fuck you' ; this.age = age } var dog = ne ...
- web form 控件
webform的简单控件1.Label:文本显示,编译后是span属性:颜色,样式,边线宽度2.Literal:纯文本显示,编译后是text 3.TextBox:文本框TextMode-密码框.文本域 ...
- linux在yum下安装mysql
1:查看软件是否安装 yum list installed | grep mysql 2:卸载CentOS系统自带mysql数据库 yum -y remove mysql-libs.x86_64,若有 ...
- 移动前端头部标签(HTML5 meta)
在移动前端开发中添加一些webkit专属的HTML5头部标签,帮助浏览器更好解析html代码 <!DOCTYPE html> 使用 HTML5 doctype,不区分大小写 <he ...
- 关于YUV格式数据
(1) YUV格式有两大类:planar和packed.对于planar的YUV格式,先连续存储所有像素点的Y,紧接着存储所有像素点的U,随后是所有像素点的V.对于packed的YUV格式,每个像素点 ...
- web初学之jdbc连接数据库
一直没有找到bug,初次接触jdbc找问题也是比较生疏.. 抛出异常: 出错点是没有在build path下导入mysql驱动jar包,发现问题,导入,成功解决问题.
- CSS声明顺序
CSS对元素样式进行声明,虽然一条规则中的声明可以按照任何顺序写出来,但是还是应该有个优先级的顺序. 如下的声明顺序摘抄自<CSS设计指南>,优先级顺序如下: display及相关声明 p ...
- Java值传递和引用传递详细解说
前天在做系统的时候被Java中参数传递问题卡了一下,回头查阅了相关的资料,对参数传递问题有了新的了解和掌握,但是有个问题感觉还是很模糊,就是 Java中到底是否只存在值传递,因为在查阅资料时,经常看到 ...