目的

为了方便使用 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. 通过Stetho在Chrome上调试Android App

    引 入依赖包 compile 'com.facebook.stetho:stetho:1.3.1' 初始化一下 public class MyApplication extends Applicati ...

  2. 二进制安装mysql 5.6

    创建用户和组 # groupadd mysql # useradd -r -g mysql mysql 解压压缩包 # tar -xvf mysql-5.6.37-linux-glibc2.12-x8 ...

  3. 8个实用而有趣Bash命令提示行

    很多人都对过命令行提示的重要性不屑一顾,甚至是一点都不关心.但是我却一点都不这么认为,一个好的命令行提示可以改变你使用命令的方式.为此,我在internet上找到一些非常实用,优秀,并有趣的bash的 ...

  4. 跟我学SharePoint 2013视频培训课程——排序、过滤在列表、库中的使用(10)

    课程简介 第10天,SharePoint 2013排序.过滤在列表.库中的使用. 视频 SharePoint 2013 交流群 41032413

  5. 变量命名神器Codelf

    个人感觉,当觉得命名困难的时候,其实是因为还没有想清楚这个变量.这个方法或者这个类是要干什么,还不能用一个或几个词准确描述它的工作,才觉得无法命名,这是命名的最困难的阶段.而只要想清楚了它的任务,命名 ...

  6. apache apr的编译和引用

    各种巧合吧,需要从JAVA转C,经过这一段时间的心理折磨,还是决定先把精力放到C上. 想快速的提高自己,学习相关语言的经典的源码是唯一的“捷径”,从Apache apr开始吧. 一.下载源代码 官网地 ...

  7. 设计模式之策略模式&amp;简单工厂模式

    学习设计模式已经有非常长一段时间了,事实上先前已经敲过一遍了.可是老认为没有学到什么,认识也不够深刻.如今趁着重构机房,再又一次来过,也不晚. 事实上在敲了机房之后,看看模式,事实上,曾经非常难理解. ...

  8. 关于Android NDK中调用第三方的动态库

    因为最近在整合Android 上RTSP播放器的网络库,因需要调用自己编译的网络库,调用一直出现问题,开始时是直接在Android.mk 中加入LOCAL_SHARED_LIBRARIES := li ...

  9. nginx 配置支持URL HTML5 History 模式 与 设置代理

    拾人牙慧:https://segmentfault.com/q/1010000007140360 nginx 配置支持URL HTML5 History 模式 location / { try_fil ...

  10. 摘抄JPA官方文档原文 防呆

    Spring Data JPA - Reference Documentation Oliver GierkeThomas DarimontChristoph StroblMark PaluchVer ...