目的

为了方便使用 DbUtils,在插入数据时需要传入含有占位符的 SQL 语句和对应占位符的值(数组),封装代码如下:

/**
* 插入实体
*/
public static <T> Long insertEntity(Class<T> entityClass, Map<String, Object> fieldMap) {
if (CollectionUtil.isEmpty(fieldMap)) {
LOGGER.error("can not insert entity: fieldMap is empty");
return null;
}
String sql = "INSERT INTO " + getTableName(entityClass);
StringBuilder colums = new StringBuilder("(");
StringBuilder values = new StringBuilder("("); // 插入实体的字段名,和字段值的占位符
for (String colum : fieldMap.keySet()) {
colums.append(colum).append(", ");
values.append("?, ");
}
colums.replace(colums.lastIndexOf(", "), colums.length(), ")");
values.replace(values.lastIndexOf(", "), values.length(), ")");
sql += colums + " VALUES " + values;
// 插入实体的值
Object[] params = fieldMap.values().toArray();
Long result = null;
try {
result = QUERY_RUNNER.insert(sql, new ScalarHandler<Long>(), params);
} catch (SQLException e) {
LOGGER.error("insert failed : {}", e);
}
return result;
}

调用该方法需要将JavaBean转化成Map<Key, Value>的形式,那不如提供一个JavaBean与Map的装换工具类。

Mybatis 源码中应该也有这种转换过程。

实现

package org.snowflake.framework.util;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.beans.BeanInfo;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map; /**
* Map 与 JavaBean 的相互转换
* Created by zhengbinMac on 2017/4/24.
*/
public class BeanUtil {
private static final Logger LOGGER = LoggerFactory.getLogger(BeanUtil.class); public static void transMap2Bean(Map<String, Object> map, Object obj) {
try {
BeanInfo beanInfo = Introspector.getBeanInfo(obj.getClass());
PropertyDescriptor[] propertyDescriptor = beanInfo.getPropertyDescriptors();
for (PropertyDescriptor pro : propertyDescriptor) {
String key = pro.getName();
if (map.containsKey(key)) {
Object value = map.get(key);
Method method = pro.getWriteMethod();
method.invoke(obj, value);
}
}
} catch (Exception e) {
LOGGER.error("transMap2Bean failed, Exception: {}", e);
}
} public static Map<String, Object> tranBean2Map(Object obj) {
if (obj == null) {
return null;
}
Map<String, Object> resultMap = new HashMap<String, Object>();
try {
BeanInfo beanInfo = Introspector.getBeanInfo(obj.getClass());
PropertyDescriptor[] propertyDescriptors = beanInfo.getPropertyDescriptors();
for (PropertyDescriptor pro : propertyDescriptors) {
String key = pro.getName();
if (!key.equals("class")) { // 过滤 class 的属性
// 得到property对应的getter方法
Method method = pro.getReadMethod();
Object value = method.invoke(obj);
resultMap.put(key, value);
}
}
} catch (Exception e) {
LOGGER.error("tranBean2Map failed, Exception: {}", e);
}
return resultMap;
}
}

测试类:

package com.wxct.test.dao;

import com.wxct.model.Person;
import org.snowflake.framework.util.BeanUtil; import java.util.Map; /**
* Created by zhengbinMac on 2017/4/24.
*/
public class BeanUtilTest {
public static void main(String[] args) {
Person p1 = new Person();
p1.setId(201370034217L);
p1.setName("郑斌");
Map<String, Object> map = BeanUtil.tranBean2Map(p1);
System.out.println(map); Person p2 = new Person();
BeanUtil.transMap2Bean(map, p2);
System.out.println(p2);
}
}

测试结果:

{name=郑斌, id=201370034217}
Person{id=201370034217, name='郑斌'}

反射实现

通过 java.lang.reflect 下的 Field 实现:

public static Map<String, Object> beanToMap(Object obj) {
Map<String, Object> resultMap = new HashMap<String, Object>();
Class cls = obj.getClass();
Field[] fields = cls.getDeclaredFields(); // 返回 Field 对象的一个数组,这些对象反映此 Class 对象所表示的类或接口所声明的所有字段。
try {
for (Field field : fields) {
field.setAccessible(true);
resultMap.put(field.getName(), field.get(obj));
}
} catch (Exception e) {
LOGGER.error("mapToBean failed, Exception: ", e);
}
return resultMap;
} public static void mapToBean(Map<String, Object> map, Object obj) {
try {
Class cla = obj.getClass();
for (String key : map.keySet()) {
Field field = cla.getDeclaredField(key);
field.setAccessible(true);
field.set(obj, map.get(key));
}
} catch (Exception e) {
LOGGER.error("mapToBean failed, Exception: ", e);
}
}

通过测试,反射的方式相比上一种更加的高效。

参考资料

[1] javaBean与Map<String,Object>互转

[2] 基于java反射的javabean和map相互转换的工具类

Map 与 JavaBean 的相互装换的更多相关文章

  1. int和string的相互装换 (c++)

    int和string的相互装换 (c++) int转换为string 第一种方法 to_string函数,这是c++11新增的函数 string to_string (int val); string ...

  2. [CSAPP笔记]Binary , Unsigned , Signed 之间的相互装换

    LaTex+MarkDown+Pandoc组合套件写博客的处女作,试试效果.各自的分工为:Latex下编辑公式,在Sublime Text 2下使用Markdown排版,最后用Pandoc导出. 摘要 ...

  3. DOM对象 与 jQuery对象 之间的相互装换

    示例代码: //jQuery对象转DOM对象 //因为jQuery对象是一个数组对象,所以转换为DOM对象时要用索引的形式 var $div1 = $("#div1"); //jQ ...

  4. xml与java代码相互装换的工具类

    这是一个java操作xml文件的工具类,最大的亮点在于能够通过工具类直接生成xml同样层次结构的java代码,也就是说,只要你定义好了xml的模板,就能一键生成java代码.省下了自己再使用工具类写代 ...

  5. python基础知识-集合,列表,元组间的相互装换

    在python中列表,元祖,集合间可以进行相互转化, def main(): set1={'hello','good','banana','zoo','Python','hello'} print(l ...

  6. opencv之深拷贝及浅拷贝,IplImage装换为Mat

    一.(1)  浅拷贝: Mat B; B = image  // 第一种方式 Mat C(image); // 第二种方式 这两种方式称为浅copy,是由于它们有不同的矩阵头,但是它们共享内存空间,即 ...

  7. 【计算机视觉】【图像处理】【VS开发】【Qt开发】opencv之深拷贝及浅拷贝,IplImage装换为Mat

    原文:opencv之深拷贝及浅拷贝,IplImage装换为Mat  一.(1) 浅拷贝: Mat B; B = image // 第一种方式 Mat C(image); // 第二种方式 这两种方式称 ...

  8. C# DataSet装换为泛型集合

    1.DataSet装换为泛型集合(注意T实体的属性其字段类型与dataset字段类型一一对应) #region DataSet装换为泛型集合 /// <summary> /// 利用反射和 ...

  9. oracle中的装换函数

    日期装换成字符的函数:TO_CHAR(date[,fmt[,params]]) 默认格式:DD-MON-RR 参数说明: date:将要装换的日期 fmt:装换的格式 params:日期的语言(可以不 ...

随机推荐

  1. tp数据库表大写命名的一些问题

    在使用thinkphp时,如果数据库表命名有大写,会被转换成小写加下划线(可以使用$model->_sql())来查看实际执行的sql是什么 这个问题,看了一下源代码,在 Thinkphp/Co ...

  2. ORA-14404: partitioned table contains partitions in a different tablespace

    SQL> drop tablespace nn_data including contents and datafiles; drop tablespace nn_data including ...

  3. 自定义View之圆形水波扩散动效

    这个效果做出来以后,真的美极了!放在你的应用中,无疑增添了光彩! 效果图    其实,第一种效果,才是产品的需求要的效果.第三种效果,是不是很熟悉?支付宝的咻一咻!哈哈,无意中,我就写出来了. 实现步 ...

  4. 使用JDBC改变Oracle的session參数 NLS_DATE_FORMAT

    近期项目除了一个问题,场景大概是这种,项目在国外开发.在项目开发过程中使用了大量的Oracle函数TO_DATE,可是开发者没有写第二个參数. 所以项目在国外的server上能够正常执行.但是在国内的 ...

  5. 10个优秀的jQuery Mobile主题

    原文链接:http://caibaojian.com/10-best-free-jquery-mobile-theme.html jQuery Mobile 是一个伟大的框架,而每个伟大的产品都需要一 ...

  6. 【转载并整理】mysql 1293错误 建表两个timestamp

    http://www.jb51.net/article/50878.htm 这里要使用到mysql的触发器

  7. git clone 问题 fatal: unable to access

    git clone 遇到问题 Cloning into 'warp-ctc'...fatal: unable to access 'https://github.com/SeanNaren/warp- ...

  8. C#基础课程之四集合(ArrayList、List<泛型>)

    list泛型的使用 ArrayList list = new ArrayList(); ArrayList list = ); //可变数组 list.Add("我"); //Ad ...

  9. 转:zTree高级入门:如何通过扩展节点的属性来达到是否显示节点的删除编辑等图标(按钮)

    当我们在使用ztree树组件的节点编辑功能时,只要我们引入了ztree相关节点编辑的js脚本文件: <script type="text/javascript" src=”/ ...

  10. html中的a标签

    <a> 标签定义超链接,用于从一张页面链接到另一张页面.最重要的属性是 href 属性,它指示链接的目标,<href="#">表示跳转到自己.我们通常通过C ...