目的

为了方便使用 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. selenium实战脚本集(1)——新浪微博发送QQ每日焦点

    背景 很多同学在工作中是没有selenium的实战环境的,因此自学的同学会感到有力无处使,想学习但又不知道怎么练习.其实学习新东西的道理都是想通的,那就是反复练习.这里乙醇会给出一些有用的,也富有挑战 ...

  2. css中position:fixed实现div居中

    上下左右 居中 代码如下 复制代码 div{ position:fixed; margin:auto; left:0; right:0; top:0; bottom:0; width:200px; h ...

  3. Objective-C 资源收藏

    日志 https://github.com/robbiehanson/CocoaLumberjack 反汇编 otool      nm http://stackoverflow.com/questi ...

  4. Cross compiling coreutils and generate the manpages

    When we cross compiling coreutils, there is an problem of generating man pages, because the source s ...

  5. windows系统如何通过Xshell 客户端连接 linux系统(主要介绍ubuntu系统)

    一. 1.查看ubuntu系统的ip地址:ifconfig 在window系统运行窗口下:ping ubuntu系统的IP地址:例如:ping 192.168.163.129 出现下述命令就是ping ...

  6. jeecg中datagrid中获取选定行的字段值

    datagrid代码如下: <t:datagrid name="orderMainList" checkbox="true" pagination=&qu ...

  7. unity, 2d rope

    https://www.youtube.com/watch?v=l6awvCT29yU

  8. 使用git上传项目到码云

    一.git安装 1.首先在官方网站下载git工具,或者根据以下链接进行下载:http://download.csdn.net/detail/qq_27501889/9788879(此链接版本为git- ...

  9. 定时删除elasticsearch的index

    #!/bin/bashfind /data/elasticsearch/data/kz-log/nodes/0/indices/ -type d -mtime +5 |  awk -F"/& ...

  10. Go 源码学习之--net/http

    其实自己不是很会看源码,但是学习优秀的源码是提升自己代码能力的一种方式,也可以对自己以后写代码有一个很好的影响,所以决定在之后的时间内,要有一个很好的习惯,阅读优秀的源码.刚开始自己会觉得看源码很痛苦 ...