当前位置:Gxlcms > 数据库问题 > 利用反射和ResultSetMetaData实现DBUtils的基本功能

利用反射和ResultSetMetaData实现DBUtils的基本功能

时间:2021-07-01 10:21:17 帮助过:17人阅读

  ResultSetMetaData是可以获取ResultSet对象的列类型和属性信息的对象.这个类里面有很多方法,在这个案例中,只用到两个:getColumnCount():获取ResultSet结果集中列的数目.getColumnName(int column):根据指定的列数目获取列名.有了这两个方法就可以自己动手去实现一个简易版的DBUtils啦~下面是我实现的步骤:

  1.编写MyQueryRunner的executeUpdate方法.

  这个方法的编写非常简单,因为可以通过dataSource获取Connection,在方法的内部就是简单的jdbc操作.需要注意的是,需要手动设置传入的参数到PreparedStatement中.代码如下:

public int update(String sql, Object... params) {
        Connection connection=null;
        PreparedStatement preparedStatement=null;
        try {
            connection=dataSource.getConnection();
            preparedStatement=connection.prepareStatement(sql);
            for(int i=0;i<params.length;i++) {
                preparedStatement.setObject((i+1), params[i]);//设置参数.
            }
            int x=preparedStatement.executeUpdate();//执行更新操作.
            return x;
        } catch (SQLException e) {
            throw new RuntimeException(e);
        }
        finally {
            if(preparedStatement!=null) {
                try {
                    preparedStatement.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
            if(connection!=null) {
                try {
                    connection.close();
                } catch (SQLException e) {
                    e.printStackTrace();
                }
            }
        }
    }

  2.编写MyQueryRunner的executeQuery方法.

  这个方法的实现也不难,因为我们将重要的代码交给传入的ResultSetHandler实现对象来处理.代码如下:

//查询会比较麻烦.
    //作出三个实现BeanHandler BeanListHandler MapListHandler
     public <T> T query(String sql, ResultSetHandler<T> rsh, Object... params) {
         Connection connection=null;
            PreparedStatement preparedStatement=null;
            ResultSet rs=null;
            try {
                connection=dataSource.getConnection();
                preparedStatement=connection.prepareStatement(sql);
                if(params!=null) {
                    for(int i=0;i<params.length;i++) {
                        preparedStatement.setObject((i+1), params[i]);
                    }
                }
                rs=preparedStatement.executeQuery();
                return rsh.handle(rs);//交给处理器处理
            } catch (SQLException e) {
                throw new RuntimeException(e);
            }
            finally {
                if(rs!=null) {
                    try {
                        rs.close();
                    } catch (SQLException e) {
                        e.printStackTrace();
                    }
                }
                if(preparedStatement!=null) {
                    try {
                        preparedStatement.close();
                    } catch (SQLException e) {
                        e.printStackTrace();
                    }
                }
                if(connection!=null) {
                    try {
                        connection.close();
                    } catch (SQLException e) {
                        e.printStackTrace();
                    }
                }
            }
     }

  3.3个处理器的编写.

  在MyQueryRunner实现自定义查询的实现中,我实现了三个处理器,它们分别是采用了BeanHandler,BeanListHandler,MapListHandler的实现思想,并且给出了最简单易懂(实际上是因为我水平不够= =)的实现.下面说说BeanHandler的实现流程,Handler的实现基本上都是一样的,无非是采用反射获得元素对象,并且将数据封装进去.

  BeanHandler中有两个成员变量,一个T t用来作为要返回的JavaBean,声明在外面供之后封装数据使用,一个Class type对象,用来获取Class,创建JavaBean对象赋值给t,创建Field对象为t的成员变量赋值.在方法的实现中,先通过getColumnCount方法来获取列的数目,遍历每一列,通过getColumnName(int columnCount)方法获取列名,用Class对象的getField方法获取Field对象(这里JavaBean的取值一定要和数据库中相同!否则会报错),再利用Field对象的set方法赋值.当然Field对象对应的成员变量一定是私有的(JavaBean的特性.)因此需要先调用setAccessable方法才可以.具体的代码如下:

public class MyBeanHandler<T> implements ResultSetHandler{
    private Class<T> type;
    T t;//需要封装的JavaBean
    public MyBeanHandler(Class<T> type) {
        this.type=type;
    }
    
    @Override
    public Object handle(ResultSet rs) throws SQLException {
        try {
            t=type.newInstance();
            ResultSetMetaData metaData = rs.getMetaData();
            int count=metaData.getColumnCount();//获取ResultSet中数据的列数
            rs.next();//移动指针
            //遍历获的每一列的列名,采用反射机制设置值
            for(int i=1;i<=count;i++) {
                String name=metaData.getColumnName(i);
                Object obj=rs.getObject(i);
                Field field = type.getDeclaredField(name);
                field.setAccessible(true);
                field.set(t, obj);//封装数据进入JavaBean
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return t;
    }

}

  MyBeanListHandler的实现步骤,基本和BeanHandler一致.唯一的区别就是采用结果集的next方法遍历每一条记录,而将对象的创建移动到了循环里,这里考虑返回的List对象的增删应该比较少因此采用的是ArrayList.具体实现如下:

public class MyBeanListHandler<T> implements ResultSetHandler<List<T>>{
    private Class<T> type;
    List<T> list=new ArrayList<T>();//可能获取比较多.因此采用ArrayList
    
    public MyBeanListHandler(Class<T> type) {
        this.type = type;
    }

    @Override
    public List<T> handle(ResultSet rs) throws SQLException {
        try {
            ResultSetMetaData metaData = rs.getMetaData();
            int count=metaData.getColumnCount();
            while(rs.next()) {
                T t=type.newInstance();
                for(int i=1;i<=count;i++) {
                    String name=metaData.getColumnName(i);//该方法获取列名.获取一系列字段名称.例如name,age...
                    Object obj=rs.getObject(i);//获取字段值
                    Field field = type.getDeclaredField(name);//获取field对象
                    field.setAccessible(true);
                    field.set(t, obj);//设置值
                }
                list.add(t);
            }
            return list;
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

}

  MapList看上去似乎最复杂,但是实际上实现起来却由于没有采用泛型和反射,是最方便的,代码如下:

public class MyMapListHandler implements ResultSetHandler<List<Map<String, Object>>> {

    private List<Map<String,Object>> data=new ArrayList<>();

    @Override
    public List<Map<String, Object>> handle(ResultSet rs) throws SQLException {
        ResultSetMetaData metaData = rs.getMetaData();
        int count=metaData.getColumnCount();
        while(rs.next()) {
            Map<String,Object> map=new HashMap<>();
            for(int i=1;i<=count;i++) {
                Object value=rs.getObject(i);
                String name=metaData.getColumnName(i);
                map.put(name, value);
            }
            data.add(map);
        }
        return data;
    }

}

 

利用反射和ResultSetMetaData实现DBUtils的基本功能

标签:shm   override   update   time   类型   开发效率   动态代理模式   代码   cti   

人气教程排行