1. 程式人生 > 程式設計 >使用java反射將結果集封裝成為物件和物件集合操作

使用java反射將結果集封裝成為物件和物件集合操作

java反射機制是什麼

反射機制是在執行狀態中,可以知道任何一個類的屬性和方法,並且呼叫類的屬性和方法;

反射機制能夠做什麼

1、判斷執行物件的所屬類

2、構造任意一個類的物件

3、獲取任意一個類的屬性和方法

4、呼叫任意屬性和方法

5、生成動態代理

利用反射將結果集封裝成為物件或者集合(實測可用)

package coral.base.util;

import java.beans.IntrospectionException;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import wfc.service.database.RecordSet;

public class ReflectUtils {
  /**
   * 將一個map集合封裝成為bean物件
   * 
   * @param param
   * @param clazz
   * @return
   */
  public static <T> T MapToBean(Map<String,Object> param,Class<?> clazz) {
    Object value = null;

    Class[] paramTypes = new Class[1];

    Object obj = null;

    try {
      obj = clazz.newInstance();

      // 獲取類的屬性
      Field[] declaredFields = clazz.getDeclaredFields();
      // 獲取父類或介面的公有屬性
      Field[] superFields = clazz.getSuperclass().getFields();

      List<Field[]> list = new ArrayList<Field[]>();
      if (declaredFields != null) {
        list.add(declaredFields);
      }
      if (superFields != null) {
        list.add(superFields);
      }
      for (Field[] fields : list) {
        for (Field field : fields) {
          String fieldName = field.getName();
          // 獲取屬性對應的值ֵ
          value = param.get(fieldName);
          // 把值設定進入物件屬性中 這裡可能是有屬性但是沒有相應的set方法,所以要做異常處理
          try {
            PropertyDescriptor pd = new PropertyDescriptor(
                fieldName,clazz);
            Method method = pd.getWriteMethod();
            method.invoke(obj,new Object[] { value });
          } catch (Exception e1) {
          }
        }
      }
    } catch (Exception e1) {
    }
    return (T) obj;
  }
  /**
   * 獲取類的所有屬性,包括父類和介面
   * @param clazz
   * @return
   */
  public static List<Field[]> getBeanFields(Class<?> clazz) {
    List<Field[]> list = new ArrayList<Field[]>();
    Field[] declaredFields = clazz.getDeclaredFields();

    Field[] superFields = clazz.getSuperclass().getFields();
    if (declaredFields != null) {
      list.add(declaredFields);

    }
    if (superFields != null) {
      list.add(superFields);
    }
    return list;
  }
  /**
   * 從結果集中獲取出值
   * @param fieldName
   * @param rs
   * @return
   */
  public static Object getFieldValue(String fieldName,ResultSet rs) {
    Object value = null;
    try {
      //捕獲值不存在的異常
      value = rs.getObject(fieldName);
      return value;
    } catch (SQLException e) {
      //oracle資料庫的列都是大寫,所以才查詢一次
      fieldName = fieldName.toLowerCase();
      try {

        value = rs.getObject(fieldName);
        return value;
      } catch (SQLException e1) {
        //結果集中沒有對應的值,返回為空
        return null;
      }

    }
  }
  /**
   * 方法過載,
   * @param fieldName
   * @param rs 這個是封裝過的結果集
   * @return
   */
  public static Object getFieldValue(String fieldName,RecordSet rs) {
    Object value = null;
    value = rs.getObject(fieldName);
    return value;
  }

  /**
   * 方法過載
   * @param rs 封裝過的結果集
   * @param clazz
   * @return
   */
  public static <T> T RSToBean(RecordSet rs,Class<?> clazz) {
    Object obj = null;
    List<Field[]> list = getBeanFields(clazz);
    try {
      obj = clazz.newInstance();

      for (Field[] fields : list) {
        for (Field field : fields) {
          String fieldName = field.getName();
          // String fieldName = field.getName();ֵ
          Object value = getFieldValue(fieldName,rs);
          try {
            PropertyDescriptor pd = new PropertyDescriptor(
                fieldName,new Object[] { value });
          } catch (Exception e1) {

          }
        }
      }
    } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    return (T) obj;
  }
  /**
   * 把結果集封裝成為bean物件
   * @param rs
   * @param clazz
   * @return
   */
  public static <T> T RSToBean(ResultSet rs,Class<?> clazz) {
    Object obj = null;
    List<Field[]> list = getBeanFields(clazz);
    try {
      while (rs.next()) {
        obj = clazz.newInstance();

        for (Field[] fields : list) {
          for (Field field : fields) {
            String fieldName = field.getName();
            // String fieldName = field.getName();ֵ
            Object value = getFieldValue(fieldName,rs);
            PropertyDescriptor pd = new PropertyDescriptor(
                fieldName,new Object[] { value });
          }
        }
      }
    } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    return (T) obj;
  }

  /**
   * 把結果集封裝成為List
   * 
   * @param rs
   * @param clazz
   * @return
   */
  public static <T> List<T> RsToList(ResultSet rs,Class<?> clazz) {
    ArrayList<T> objList = new ArrayList<T>();
    // 獲取所有的屬性
    List<Field[]> list = getBeanFields(clazz);
    try {
      while (rs.next()) {
        // 定義臨時變數
        Object tempObeject = clazz.newInstance();

        // 新增到屬性中
        for (Field[] fields : list) {
          for (Field field : fields) {
            String fieldName = field.getName();
            // 獲取屬性值ֵ
            Object value = getFieldValue(fieldName,clazz);
            Method method = pd.getWriteMethod();
            method.invoke(tempObeject,new Object[] { value });
          }
        }
        objList.add((T) tempObeject);
      }
    } catch (Exception e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
    return objList;
  }
}

補充知識:java反射自動封裝值到實體類

1.工具類

package com.util;

import com.entity.Student;
import javax.servlet.ServletRequest;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.text.SimpleDateFormat;
import java.util.Enumeration;

/**
 * Created by wq on 2017/8/30.
 */
public class BeanOperateTools {

  /*
  利用反射進行所以請求引數的設定,要求引數名與屬性名一致
   */

  /**
   *
   * @param obj
   * @param req
   * @param datePatern 將字串變為日期時間格式化的字串
   * @throws Exception
   */
  public static void setValue(Object obj,ServletRequest req,String datePatern)
      throws Exception {//設定屬性的內容
    {
      System.out.println("進了setValue方法.....");
      //取得class物件
      Class<?> cls = obj.getClass();
      //取得輸入的全部引數名稱
      Enumeration<String> enu = req.getParameterNames();
      //迴圈輸入的全部引數名稱
      while (enu.hasMoreElements()) {
        String paramName = enu.nextElement();//取得引數名稱
        String paramValue = req.getParameter(paramName);
      //取得屬性的型別是為了取得引數的型別,以確定method方法和是否轉型

        Field field = cls.getDeclaredField(paramName); //取得指定名稱的屬性 (實體類裡面
        //取得指定的操作方法,以滿足反射呼叫
        Method method = cls.getMethod("set"+StringTools.initcap(paramName),field.getType());
        //根據型別進行資料的轉換,同時呼叫setter設定資料
        //取得型別
        String fieldType=field.getType().getSimpleName();
        if ("string".equalsIgnoreCase(fieldType)) {
          method.invoke(obj,paramValue);
        }else if ("integer".equalsIgnoreCase(fieldType)||"int".equalsIgnoreCase(fieldType)){
          method.invoke(obj,Integer.parseInt(paramValue));
        }else if ("double".equalsIgnoreCase(fieldType)){
          method.invoke(obj,Double.parseDouble(paramValue));
        }else if ("date".equalsIgnoreCase(fieldType)){
          method.invoke(obj,new SimpleDateFormat(datePatern).parse(paramValue));
        }
      }
    }
  }
}

2.servlet中呼叫此方法

package com.servlet;

import com.entity.Student;
import com.sun.xml.internal.bind.v2.runtime.reflect.opt.FieldAccessor_Double;
import com.util.BeanOperateTools;

import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Enumeration;

/**
 * Created by wq on 2017/8/22.
 */
@WebServlet(name = "studentServlet",urlPatterns = {"/StudentServlet"})
public class StudentServlet extends HttpServlet {
  //private Student stu = new Student();
  private Student student=new Student();
  public Student getStudent() {
    return student;
  }

  public void setStudent(Student student) {
    this.student = student;
  }

-------此處模仿依賴注入
  @Override
  protected void doGet(HttpServletRequest req,HttpServletResponse resp) throws ServletException,IOException {
    req.setCharacterEncoding("UTF-8");
    try {
      BeanOperateTools.setValue(this.getStudent(),req,"yyyy-MM-dd");
    } catch (Exception e) {
      e.printStackTrace();
    }
    req.setAttribute("student",this.student);
    req.getRequestDispatcher("student_insert_do.jsp").forward(req,resp);
    // Class<?> cls= this.stu.getClass();
  }

  @Override
  protected void doPost(HttpServletRequest req,IOException {
    this.doGet(req,resp);
  }
  public String initcap(String str) {//首字母大寫
    return str.substring(0,1).toUpperCase().concat(str.substring(1).toLowerCase());
  }
}

3.新增學生的jsp頁面

<%--
 Created by IntelliJ IDEA.
 User: wq
 Date: 2017/8/21
 Time: 23:25
 To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
  <title>Title</title>
</head>
<body>
<form action="StudentServlet" method="post">
  姓名:<input type="text" name="name"><br>
  年齡:<input type="text" name="age"><br>
  成績:<input type="text" name="score"><br>
  <input type="submit" value="輸入">
  <input type="reset" value="重置">
</form>
</body>
</html>

4。展示學生的資訊的頁面

<%--
 Created by IntelliJ IDEA.
 User: wq
 Date: 2017/8/21
 Time: 23:25
 To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
  <title>Title</title>
</head>
<body>
 <h1>姓名:${student.name}</h1>
 <h1>年齡:${student.age}</h1>
 <h1>成績:${student.score}</h1>

</body>
</html>

以上這篇使用java反射將結果集封裝成為物件和物件集合操作就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支援我們。