在路上

 找回密码
 立即注册
在路上 站点首页 学习 查看内容

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

2016-12-20 13:17| 发布者: zhangjf| 查看: 484| 评论: 0

摘要: 遇到了若干问题: 1.从oracle返回的列名都是大写,再用反射,就找不到相对应得 名字 2.oracle 中number类型 返回来,就变成了BigDecimal public static void main(String args){ String sql ...

遇到了若干问题:

1.从oracle返回的列名都是大写,再用反射,就找不到相对应得 名字

2.oracle 中number类型 返回来,就变成了BigDecimal

  1. public static void main(String[] args){
  2. String sql = "SELECT IDCARD , examcard , "
  3. + "studentname ,"
  4. + "lacation LoCATION,grade "
  5. + " FROM student WHERE IDCARD = ?";
  6. Student s = get(Student.class, sql, 7);
  7. System.out.println(s);
  8. }
  9. //String sql = "SELECT id, name, email, birth "
  10. <span style="white-space:pre"> </span>//<span style="white-space:pre"> </span>+ "FROM customers WHERE id = ?";
  11. public static <T> T get(Class<T> clazz, String sql, Object... args) {
  12. T entity = null;
  13. Connection connection = null;
  14. PreparedStatement preparedStatement = null;
  15. ResultSet resultSet = null;
  16. try {
  17. //1. 得到 ResultSet 对象
  18. connection = JDBC_Tools.getConnection();
  19. preparedStatement = connection.prepareStatement(sql);
  20. for (int i = 0; i < args.length; i++) {
  21. preparedStatement.setObject(i + 1, args[i]);
  22. }
  23. resultSet = preparedStatement.executeQuery();
  24. //2. 得到 ResultSetMetaData 对象
  25. ResultSetMetaData rsmd = resultSet.getMetaData();
  26. //3. 创建一个 Map<String, Object> 对象, 键: SQL 查询的列的别名,
  27. //值: 列的值
  28. Map<String, Object> values = new HashMap<>();
  29. //4. 处理结果集. 利用 ResultSetMetaData 填充 3 对应的 Map 对象
  30. if(resultSet.next()){
  31. for(int i = 0; i < rsmd.getColumnCount(); i++){
  32. //从 ResultSetMetaData 获取列的别名
  33. String columnLabel = rsmd.getColumnLabel(i + 1);
  34. //从 结果集 中获取列的值
  35. Object columnValue = resultSet.getObject(i + 1);
  36. values.put(columnLabel, columnValue);
  37. }
  38. }
  39. //5. 若 Map 不为空集, 利用反射创建 clazz 对应的对象
  40. if(values.size() > 0){
  41. entity = clazz.newInstance();
  42. //5. 遍历 Map 对象, 利用反射为 Class 对象的对应的属性赋值.
  43. for(Map.Entry<String, Object> entry: values.entrySet()){
  44. String fieldName = entry.getKey();
  45. Object value = entry.getValue();
  46. //System.out.println(fieldName+":"+value);
  47. ReflectionUtils.setFieldValue(entity, fieldName, value); //出问题
  48. //System.out.println(ReflectionUtils.getDeclaredField(entity,fieldName));
  49. }
  50. }
  51. } catch (Exception e) {
  52. e.printStackTrace();
  53. } finally {
  54. JDBC_Tools.relaseSource(resultSet, connection, preparedStatement);
  55. }
  56. return entity;
  57. }
  58. }
复制代码

ReflectionUtils.setFieldValue(entity, fieldName, value);出问题

java.lang.IllegalArgumentException: Can not set int field xuezaipiao3.Student.GRADE to java.math.BigDecimal
at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:164)
at sun.reflect.UnsafeFieldAccessorImpl.throwSetIllegalArgumentException(UnsafeFieldAccessorImpl.java:168)
at sun.reflect.UnsafeIntegerFieldAccessorImpl.set(UnsafeIntegerFieldAccessorImpl.java:98)
at java.lang.reflect.Field.set(Field.java:741)
at xuezaipiao3.ReflectionUtils.setFieldValue(ReflectionUtils.java:156)
at xuezaipiao3.ThinkInJDBC.get(ThinkInJDBC.java:77)
at xuezaipiao3.ThinkInJDBC.main(ThinkInJDBC.java:45)
Student [IDCARD=0, EXAMCARD=0, STUDENTNAME=null, LOCATION=7, GRADE=0]


Oracle 中 number类型 通过ResultSet 的 getObject() 返回的是 BigDecimal ,无法强制转化,而且 ResultSetMetaData 的 getColumnLabel() 取回的 列的别名是大写

MySQL不存在这样的问题

用的 Oracle 10g


ReflectionUtils

  1. package xuezaipiao3;
  2. import java.lang.reflect.Field;
  3. import java.lang.reflect.InvocationTargetException;
  4. import java.lang.reflect.Method;
  5. import java.lang.reflect.Modifier;
  6. import java.lang.reflect.ParameterizedType;
  7. import java.lang.reflect.Type;
  8. /**
  9. * 反射的 Utils 函数集合
  10. * 提供访问私有变量, 获取泛型类型 Class, 提取集合中元素属性等 Utils 函数
  11. * @author Administrator
  12. *
  13. */
  14. public class ReflectionUtils {
  15. /**
  16. * 通过反射, 获得定义 Class 时声明的父类的泛型参数的类型
  17. * 如: public EmployeeDao extends BaseDao<Employee, String>
  18. * @param <T>
  19. * @param clazz
  20. * @param index
  21. * @return
  22. */
  23. @SuppressWarnings("unchecked")
  24. public static Class getSuperClassGenricType(Class clazz, int index){
  25. Type genType = clazz.getGenericSuperclass();
  26. if(!(genType instanceof ParameterizedType)){
  27. return Object.class;
  28. }
  29. Type [] params = ((ParameterizedType)genType).getActualTypeArguments();
  30. if(index >= params.length || index < 0){
  31. return Object.class;
  32. }
  33. if(!(params[index] instanceof Class)){
  34. return Object.class;
  35. }
  36. return (Class) params[index];
  37. }
  38. /**
  39. * 通过反射, 获得 Class 定义中声明的父类的泛型参数类型
  40. * 如: public EmployeeDao extends BaseDao<Employee, String>
  41. * @param <T>
  42. * @param clazz
  43. * @return
  44. */
  45. @SuppressWarnings("unchecked")
  46. public static<T> Class<T> getSuperGenericType(Class<T> clazz){
  47. return getSuperClassGenricType(clazz, 0);
  48. }
  49. /**
  50. * 循环向上转型, 获取对象的 DeclaredMethod
  51. * @param object
  52. * @param methodName
  53. * @param parameterTypes
  54. * @return
  55. */
  56. public static Method getDeclaredMethod(Object object, String methodName, Class<?>[] parameterTypes){
  57. for(Class<?> superClass = object.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()){
  58. try {
  59. //superClass.getMethod(methodName, parameterTypes);
  60. return superClass.getDeclaredMethod(methodName, parameterTypes);
  61. } catch (NoSuchMethodException e) {
  62. //Method 不在当前类定义, 继续向上转型
  63. }
  64. //..
  65. }
  66. return null;
  67. }
  68. /**
  69. * 使 filed 变为可访问
  70. * @param field
  71. */
  72. public static void makeAccessible(Field field){
  73. if(!Modifier.isPublic(field.getModifiers())){
  74. field.setAccessible(true);
  75. }
  76. }
  77. /**
  78. * 循环向上转型, 获取对象的 DeclaredField
  79. * @param object
  80. * @param filedName
  81. * @return
  82. */
  83. public static Field getDeclaredField(Object object, String filedName){
  84. for(Class<?> superClass = object.getClass(); superClass != Object.class; superClass = superClass.getSuperclass()){
  85. try {
  86. return superClass.getDeclaredField(filedName);
  87. } catch (NoSuchFieldException e) {
  88. //Field 不在当前类定义, 继续向上转型
  89. }
  90. }
  91. return null;
  92. }
  93. /**
  94. * 直接调用对象方法, 而忽略修饰符(private, protected)
  95. * @param object
  96. * @param methodName
  97. * @param parameterTypes
  98. * @param parameters
  99. * @return
  100. * @throws InvocationTargetException
  101. * @throws IllegalArgumentException
  102. */
  103. public static Object invokeMethod(Object object, String methodName, Class<?> [] parameterTypes,
  104. Object [] parameters) throws InvocationTargetException{
  105. Method method = getDeclaredMethod(object, methodName, parameterTypes);
  106. if(method == null){
  107. throw new IllegalArgumentException("Could not find method [" + methodName + "] on target [" + object + "]");
  108. }
  109. method.setAccessible(true);
  110. try {
  111. return method.invoke(object, parameters);
  112. } catch(IllegalAccessException e) {
  113. System.out.println("不可能抛出的异常");
  114. }
  115. return null;
  116. }
  117. /**
  118. * 直接设置对象属性值, 忽略 private/protected 修饰符, 也不经过 setter
  119. * @param object
  120. * @param fieldName
  121. * @param value
  122. */
  123. public static void setFieldValue(Object object, String fieldName, Object value){
  124. Field field = getDeclaredField(object, fieldName);
  125. if (field == null)
  126. throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + object + "]");
  127. makeAccessible(field);
  128. try {
  129. field.set(object, value);
  130. } catch (IllegalAccessException e) {
  131. System.out.println("不可能抛出的异常");
  132. }
  133. }
  134. /**
  135. * 直接读取对象的属性值, 忽略 private/protected 修饰符, 也不经过 getter
  136. * @param object
  137. * @param fieldName
  138. * @return
  139. */
  140. public static Object getFieldValue(Object object, String fieldName){
  141. Field field = getDeclaredField(object, fieldName);
  142. if (field == null)
  143. throw new IllegalArgumentException("Could not find field [" + fieldName + "] on target [" + object + "]");
  144. makeAccessible(field);
  145. Object result = null;
  146. try {
  147. result = field.get(object);
  148. } catch (IllegalAccessException e) {
  149. System.out.println("不可能抛出的异常");
  150. }
  151. return result;
  152. }
  153. }
复制代码

来自:http://blog.csdn.net/wjw0130/article/details/43760487

最新评论

小黑屋|在路上 ( 蜀ICP备15035742号-1 

;

GMT+8, 2025-7-8 14:14

Copyright 2015-2025 djqfx

返回顶部