<五>JDBC_利用反射及JDBC元数据编写通用的查询方法

时间:2022-08-27 21:08:28

此类针对javaBean类写了一个通用的查询方法,List<javaBean> 通用查询更新中...;通过学习,深刻体会到学会反射就等于掌握了java基础的半壁*!

一、使用JDBC驱动程序处理元数据:
 *   .java通过JDBC获得连接后得到一个Connection对象,可以从这个对象获得有关数据库管理系统的各种信息,
 *   包括数据库的各个表,表中的各个列,数据类型,触发器,存储过程等各方面的信息。根据这些信息,JDBC可以访问
 *   一个事先并不了解的数据库。
 *   .获取这些信息的方法都是在DataBaseMetaData类的对象上实现的,而DataBaseMetaData对象是在
 *   Connection对象上获得的。
二、ResultSetMetaData类
 *   .是描述ResultSet的元数据对象,可用于获取ResultSet对象中列的类型和属性信息的对象:
 *    ---getColumnName(int column) :获取指定列的名称
 *    ---getColumnCount() : 返回当前ResultSet对象中的列数
 *    ---getColumnLabel(int column) : 获取指定列的别名(索引从1开始)
 *    ---getColumnTypeName() : 检索指定列的数据库特定的类型名称
 *    ---getColumnDisplaySize() : 指示指定列的最大标准宽度,以字符为单位
 *    ---isNullable(int column) : 指示指定列中的值是否可以为null
 *    ---isAutoIncrement() : 指示是否自动为指定列进行编号,这样这些列仍然是只读的。
 *    
 *    调用ResultSet的getMetaData()方法得到ResultSetMetaData对象。

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.ResultSetMetaData;
import java.util.HashMap;
import java.util.Map;

import org.junit.Test;

public class JDBCTest2 {
 
 @Test
 public void testGet(){
    String sql="select id,name,email,birth from customers where id=?";
  
    Customer customer=get(Customer.class,sql,2);
    System.out.println(customer);
    /*
     * **FlowID对应数据库表属性字段,要求别名(flow_Id)对应javaBean的属性名**
     *   解析sql语句中列的别名、值,再通过反射技术把值赋给对应的属性,详细过程如下:
     *     1、先利用SQL语句进行查询,得到结果集
     *     2、利用反射创建实体类的对象
     *     3、获取结果集列的别名
     *      如何获取?
     *     4、再获取结果集的每一列的值,结合3得到一个Map,键:列的别名,值:列的值
     *     5、再利用反射为2的对应的属性赋值,属性即为Map的键,值即为Map的值。
     * */
    sql="select FlowId flow_Id,Type,IDCard,ExamCard,StudentName from examstudent where FlowId=?";
    Student stu=get(Student.class,sql,1);
    System.out.println(stu);
  
   }
    public <T> T get(Class<T> clazz, String sql,Object... args){
  
      /* 0、得到ResultSet对象
       * 1、得到ResultSetMetaData的对象
       * 2、创建一个Map<String,Object>对象
       * 3、处理结果集,利用ResultSetMetaData填充2对应的Map对象
       * 4、若Map不为空集,利用反射创建clazz对应的对象
       * 5、遍历Map对象,利用反射为Class对象对应的属性赋值
       *
       * */
  
      T entity=null;
  
      Connection con=null;
      PreparedStatement ps=null;
      ResultSet rs=null;
  
      try {
   
         con=JDBCTools.getConnection();
         ps=con.prepareStatement(sql);
         for (int i = 0; i < args.length; i++) {
            ps.setObject(i+1, args[i]);
         }
         rs=ps.executeQuery();
   
         ResultSetMetaData rsmd=rs.getMetaData();
   
         Map<String,Object> values=new HashMap<>();
         //因获取的是一个类...,所以if非while
         if(rs.next()) {
            for (int i = 0; i < rsmd.getColumnCount(); i++) {
               String columnLabel=rsmd.getColumnLabel(i+1);
               Object columnValue=rs.getObject(i+1);
               values.put(columnLabel, columnValue);
            }
    
         }
         //若Map不为空
         if (values.size()>0) {
            entity=clazz.newInstance();
            //为javaBean的属性赋值
            for (Map.Entry<String, Object> entry:values.entrySet()) {
                 String fieldName=entry.getKey();
                 Object value=entry.getValue();
                  ReflectionUtils.setFieldValue(entity, fieldName, value);
            }
         }
   
   
      } catch (Exception e) {
         e.printStackTrace();
      }finally{
         JDBCTools.release(rs, ps, con);
      }
  
      return entity;
   }
 }

<五>JDBC_利用反射及JDBC元数据编写通用的查询方法

帮助工具类

import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;

/**
 * 反射的 Utils 函数集合
 * 提供访问私有变量, 获取泛型类型 Class, 提取集合中元素属性等 Utils 函数
 * @author Administrator
 *
 */
public class ReflectionUtils {

/**
  * 通过反射, 获得定义 Class 时声明的父类的泛型参数的类型
  * 如: public EmployeeDao extends BaseDao<Employee, String>
  * @param clazz
  * @param index
  * @return
  */
 @SuppressWarnings("unchecked")
 public static Class getSuperClassGenricType(Class clazz, int index){
  Type genType = clazz.getGenericSuperclass();
  
  if(!(genType instanceof ParameterizedType)){
   return Object.class;
  }
  
  Type [] params = ((ParameterizedType)genType).getActualTypeArguments();
  
  if(index >= params.length || index < 0){
   return Object.class;
  }
  
  if(!(params[index] instanceof Class)){
   return Object.class;
  }
  
  return (Class) params[index];
 }
 
 /**
  * 通过反射, 获得 Class 定义中声明的父类的泛型参数类型
  * 如: public EmployeeDao extends BaseDao<Employee, String>
  * @param <T>
  * @param clazz
  * @return
  */
 @SuppressWarnings("unchecked")
 public static<T> Class<T> getSuperGenericType(Class clazz){
  return getSuperClassGenricType(clazz, 0);
 }
 
 /**
  * 循环向上转型, 获取对象的 DeclaredMethod
  * @param object
  * @param methodName
  * @param parameterTypes
  * @return
  */
 public static Method getDeclaredMethod(Object object, String methodName, Class<?>[] parameterTypes){
  
  for(Class<?> superClass = object.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()){
   try {
    //superClass.getMethod(methodName, parameterTypes);
    return superClass.getDeclaredMethod(methodName, parameterTypes);
   } catch (NoSuchMethodException e) {
    //Method 不在当前类定义, 继续向上转型
   }
   //..
  }
  
  return null;
 }
 
 /**
  * 使 filed 变为可访问
  * @param field
  */
 public static void makeAccessible(Field field){
  if(!Modifier.isPublic(field.getModifiers())){
   field.setAccessible(true);
  }
 }
 
 /**
  * 循环向上转型, 获取对象的 DeclaredField
  * @param object
  * @param filedName
  * @return
  */
 public static Field getDeclaredField(Object object, String filedName){
  
  for(Class<?> superClass = object.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()){
   try {
    return superClass.getDeclaredField(filedName);
   } catch (NoSuchFieldException e) {
    //Field 不在当前类定义, 继续向上转型
   }
  }
  return null;
 }
 
 /**
  * 直接调用对象方法, 而忽略修饰符(private, protected)
  * @param object
  * @param methodName
  * @param parameterTypes
  * @param parameters
  * @return
  * @throws InvocationTargetException
  * @throws IllegalArgumentException
  */
 public static Object invokeMethod(Object object, String methodName, Class<?> [] parameterTypes,
   Object [] parameters) throws InvocationTargetException{
  
  Method method = getDeclaredMethod(object, methodName, parameterTypes);
  
  if(method == null){
   throw new IllegalArgumentException("Could not find method [" + methodName + "] on target [" + object + "]");
  }
  
  method.setAccessible(true);
  
  try {
   return method.invoke(object, parameters);
  } catch(IllegalAccessException e) {
   System.out.println("不可能抛出的异常");
  }
  
  return null;
 }
 
 /**
  * 直接设置对象属性值, 忽略 private/protected 修饰符, 也不经过 setter
  * @param object
  * @param fieldName
  * @param value
  */
 public static void setFieldValue(Object object, String fieldName, Object value){
  Field field = getDeclaredField(object, fieldName);
  
  if (field == null)
   throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + object + "]");
  
  makeAccessible(field);
  
  try {
   field.set(object, value);
  } catch (IllegalAccessException e) {
   System.out.println("不可能抛出的异常");
  }
 }
 
 /**
  * 直接读取对象的属性值, 忽略 private/protected 修饰符, 也不经过 getter
  * @param object
  * @param fieldName
  * @return
  */
 public static Object getFieldValue(Object object, String fieldName){
  Field field = getDeclaredField(object, fieldName);
  
  if (field == null)
   throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + object + "]");
  
  makeAccessible(field);
  
  Object result = null;
  
  try {
   result = field.get(object);
  } catch (IllegalAccessException e) {
   System.out.println("不可能抛出的异常");
  }
  
  return result;
 }
}

总结:

1、why:如果只有一个结果集。但不知道结果集中有多少列,列的名字都是什么?

1>编写通用的查询方法时需要使用;

public <T> T get(Class<T> clazz,String sql,Object...sql)

2、what:用于描述ResultSet的对象。

3、how

1>得到ResultSetMetaData对象:调用ResultSet的getMetaData()方法

  ResultSetMetaData rsmd=resultSet.getMetaData();

2>ResultSetMetaData有哪些好用的方法:

  >int getColumnCount()结果集中包含哪些列

  >String getColumnLabel(int column):获取指定列的别名,其中索引从1开始。

<五>JDBC_利用反射及JDBC元数据编写通用的查询方法的更多相关文章

  1. Java -- JDBC&lowbar;利用反射及 JDBC 元数据编写通用的查询方法

    先利用 SQL 进行查询,得到结果集: 利用反射创建实体类的对象:创建对象: 获取结果集的列的别名: 再获取结果集的每一列的值, 结合 3 得到一个 Map,键:列的别名,值:列的值: 再利用反射为 ...

  2. JDBC学习笔记&lpar;5&rpar;——利用反射及JDBC元数据编写通用的查询方法

    JDBC元数据 1)DatabaseMetaData /** * 了解即可:DatabaseMetaData是描述数据库的元数据对象 * 可以由Connection得到 */ 具体的应用代码: @Te ...

  3. 【转】JDBC学习笔记&lpar;5&rpar;——利用反射及JDBC元数据编写通用的查询方法

    转自:http://www.cnblogs.com/ysw-go/ JDBC元数据 1)DatabaseMetaData /** * 了解即可:DatabaseMetaData是描述数据库的元数据对象 ...

  4. MYSQL 之 JDBC(六): 增删改查(四)利用反射及JDBC元数据编写通用的查询

    1.先利用SQL进行查询,得到结果集2.利用反射创建实体类的对象:创建Student对象3.获取结果集的列的别名:idCard.studentName4.再获取结果集的每一列的值,结合3得到一个Map ...

  5. 利用反射及jdbc元数据实现通用的查询方法

    ---------------------------------------------------------------------------------------------------- ...

  6. JDBC--利用反射及JDBC元数据编写通用的查询方法

    1.JDBC元数据(ResuleSetMetaData):描述ResultSet的元数据对象,可以从中获取到结果集中的列数和列名等: --使用ResultSet类的getMetaData()方法获得R ...

  7. java攻城狮之路--复习JDBC&lpar;利用BeanUtils、JDBC元数据编写通用的查询方法&semi;元数据&semi;Blob&semi;事务&semi;批量处理&rpar;

    1.利用BeanUtils的前提得要加入以下两个jar包: commons-beanutils-1.8.0.jar commons-logging-1.1.1.jar package com.shel ...

  8. 利用反射及JDBC元数据编写通用查询方法

    元数据:描述数据的数据,ResultSetMetaData是描述ResultSet的元数据对象,从它可以得到数据集有多少了,每一列的列名... ResultSetMetaData可以通过ResultS ...

  9. JDBC课程5--利用反射及JDBC元数据&lpar;ResultSetMetaData&rpar;编写通用的查询方法

    /**-利用反射及JDBC元数据编写通用的查询方法 * 1.先利用SQl语句进行查询,得到结果集--> * 2.查找到结果集的别名:id--> * 3.利用反射创建实体类的对象,创建aut ...

随机推荐

  1. 如何利用 Visual Studio 自带工具提高开发效率

    Visual Stuido 是一款强大的Windows 平台集成开发工具,你是否好好地利用了它呢? 显示行号 有些时候(比如错误定位)的时候,显示行号将有利于我们进行快速定位. 如何显示 1. 工具 ...

  2. 教你摸清 Linux PC 的性能底细?

    导读 基准测试是一项测试或一系列测试,用来确定某个计算机硬件运行起来的状况有多好.在许多情况下,“基准测试”实际上等同于“压力测试”.通过测试硬件的极限,然后可以将测得的结果与其他硬件测得的结果作一番 ...

  3. odoo 10 生产自动领料

    分析源码 当 原材料的 补货规则 的 "补货位置" location_id 是 生产单 的 原材料 "目标位置 ", 并且 原材料的 补货规则 的 " ...

  4. 判断是否是有效的IPV4地址

    参考链接: https://blogs.msdn.microsoft.com/oldnewthing/20060522-08/?p=31113 http://www.cnblogs.com/txw19 ...

  5. Android项目打包成APK文件

    第一步:右键单击该项目选择Export项目 显演示样例如以下界面:选择Exprot Android Application 第二步:输入项目名称,默认的情况下是原始的项目名称 下一步: 点击 Crea ...

  6. javascript中new Date&lpar;&rpar;存在的兼容性问题

    问题:通过new Date()创建的时间对象在Chrome能正常工作,但在IE浏览器却显示NaN 代码: var time = new Date(date + ' 00:00:00'); //NaN ...

  7. poj3666 线性dp

    要把一个序列变成一个不严格的单调序列,求最小费用 /* 首先可以证明最优解序列中的所有值都能在原序列中找到 以不严格单增序列为例, a序列为原序列,b序列为升序排序后的序列 dp[i][j]表示处理到 ...

  8. 【转】web&period;xml配置项详解

    史上最全web.xml配置文件元素详解   一.web.xml配置文件常用元素及其意义预览 1 <web-app> 2 3 <!--定义了WEB应用的名字--> 4 <d ...

  9. Spring Log4j2 log4j2&period;xml

    <dependency> <groupId>org.apache.logging.log4j</groupId> <artifactId>log4j-c ...

  10. Sed&amp&semi;awk笔记之sed篇&lpar;转&rpar;

    Sed是什么 <sed and awk>一书中(1.2 A Stream Editor)是这样解释的: Sed is a "non-interactive" strea ...