SpringMVC中的引數繫結總結

林加欣發表於2017-10-30

眾所周知,springmvc是用來處理頁面的一些請求,然後將資料再通過檢視返回給使用者的,前面的幾篇博文中使用的都是靜態資料,為了能快速入門springmvc,在這一篇博文中,我將總結一下springmvc中如何接收前臺頁面的引數,即springmvc中的引數繫結問題。

1. 引數繫結的過程

  我們可以回憶一下,在struts2中,是通過在Action中定義一個成員變數來接收前臺傳進來的引數,而在springmvc中,接收頁面提交的資料是通過方法形參來接收的。從客戶端請求的key/value資料,經過引數繫結,將key/value資料繫結到controller方法的形參上,然後就可以在controller中使用該引數了。來看一下這個過程: 
引數繫結 
  所以我們知道,是springmvc提供了很多轉換器來將頁面引數繫結到controller方法的形參上,關於自定義converter,我下面會提到。大概瞭解了該過程後,下面開始做具體的總結。

2. 預設支援的型別

  springmvc中,有支援的預設型別的繫結。也就是說,直接在controller方法形參上定義預設型別的物件,就可以使用這些物件。

  1. HttpServletRequest物件
  2. HttpServletResponse物件
  3. HttpSession物件
  4. Model/ModelMap物件

  在引數繫結過程中,如果遇到上面型別就直接進行繫結。也就是說,我們可以在controller的方法的形參中直接定義上面這些型別的引數,springmvc會自動繫結。這裡要說明一下的就是Model/ModelMap物件,Model是一個介面,ModelMap是一個介面實現 ,作用是將Model資料填充到request域,跟ModelAndView類似,關於它的使用,我在下面和簡單型別引數繫結一起說。

3. 簡單型別的繫結

  總結這個還是以需求為例吧,這樣比較容易理解,假設現在有個需求:根據商品的id來修改對應點商品資訊。所以前臺頁面肯定要傳進來該商品的id,然後springmvc的controller進行處理,返回一個修改商品資訊的頁面。關於前臺頁面的東西都很簡單,我就不貼程式碼了,主要部分截個圖,具體的程式碼在文章最後有下載地址。 
  前臺頁面通過url將引數傳遞過來,請求的是editItems.action。 
簡單型別 
  下面寫controller中的editItems方法:

@RequestMapping("/editItems")
public String editItems(Model model, Integer id) throws Exception {
    //根據id查詢對應的Items
    ItemsCustom itemsCustom = itemsService.findItemsById(id);

    model.addAttribute("itemsCustom", itemsCustom);

    //通過形參中的model將model資料傳到頁面
    //相當於modelAndView.addObject方法
    return "/WEB-INF/jsp/items/editItems.jsp";
}

  這是個很簡單的demo,從上面的程式碼中可以看出model可以直接作為引數,springmvc預設會繫結它,然後使用model將查詢到的資料放到request域中,這樣就可以在前臺頁面取出該資料了。 
  要注意一點的是,簡單型別的繫結中,方法形參中的引數名要和前臺傳進來的名一樣才能完成引數的繫結。那有人要問了,如果有特殊需求(比如更好的可讀性?),這裡定義的引數名就是不一樣,那咋整呢?有解決辦法麼?有!我們可以使用註解@RequestParam對簡單的型別進行引數繫結,如下: 
requestParam註解
  所以說,如果不使用@RequestParam,要求request傳入引數名稱和controller方法的形參名稱一致,方可繫結成功。如果使用@RequestParam,不用限制request傳入引數名稱和controller方法的形參名稱一致。通過@RequestParam中的required屬性指定引數是否必須要傳入,如果設定為true,沒有傳入引數就會報錯。

4. pojo型別的繫結

4.1 普通pojo型別

  再來總結下pojo型別的繫結,繼續上面的案例,當頁面展示了商品詳細資訊後,我做了修改,然後點選提交,後臺應該將我提交的這些引數全部更新到資料庫的items表中,也就是說,我提交的這些引數要繫結到Items物件或者其擴充套件物件中。先看一下Items中都有哪些屬性: 
屬性 
  可以看到,有各種型別的屬性,當我們提交後,要將這些屬性全部封裝到一個pojo中,然後去更新資料庫。 
  繫結很簡單,對於基本型別,要求頁面中input標籤的name屬性值和controller的pojo形參中的屬性名稱一致,即可將頁面中資料繫結到pojo。也就是說前臺頁面傳進來的name要和要封裝的pojo屬性名一模一樣,然後就可以將該pojo作為形參放到controller的方法中,如下: 
pojo型別
  這樣就能將前臺表單傳進來的不同屬性值封裝到ItemsCustom中了。但是執行一下就會發現報錯,報錯的資訊是無法將String型別轉換成java.util.Date型別,因為上面Items中有一個屬性是Date型別的createtime。這就需要我們自己定義轉換器了,這也是這部分的重點,下面我們自己定義一個日期轉換器:

//需要實現Converter介面,這裡是將String型別轉換成Date型別
public class CustomDateConverter implements Converter<String, Date> {

    @Override
    public Date convert(String source) {
        //實現 將日期串轉成日期型別(格式是yyyy-MM-dd HH:mm:ss)
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");

        try {
            //轉成直接返回
            return simpleDateFormat.parse(source);
        } catch (ParseException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
        //如果引數繫結失敗返回null
        return null;

    }
}

  定義好了轉換器後,需要在springmvc.xml中進行如下配置: 
轉換器 
轉換器配置
  現在就可以了,springmvc就能根據這個轉換器將String型別正確轉換成Date型別,然後封裝到ItemsCustom中去了。 
  這裡說一個小小的插曲:修改商品詳細資訊後提交,可能會有中文亂碼問題,表達提交都是post方式,springmvc中關於post方式的中文亂碼問題可以在web.xml中配置一個過濾器來解決,如下: 
post亂碼解決

4.2. 包裝的pojo型別

  這個包裝型別pojo與上面普通的pojo有啥區別呢?包裝型別pojo指的是pojo中有另一個也是pojo的屬性,即pojo套pojo,為什麼會設計這種pojo呢?在前面的博文中我也有提到,這種組合的設計方法對於後期程式的擴充套件很有用,比如複雜的查詢條件就需要包裝到這種包裝型別中。 
  那麼該如何繫結呢?有兩個思路:

  1. 在形參中新增HttpServletRequest request引數,通過request接收查詢條件引數。
  2. 在形參中讓包裝型別的pojo接收查詢條件引數。

  第一種方式就跟原始servlet差不多,這裡使用第二種方法,我們傳進來一個包裝型別的pojo。看一下這個包裝型別的pojo: 
包裝型別pojo 
  這個包裝pojo中還有一個ItemsCustom類,這個類繼承了Items類,並且用來擴充套件與Items相關的User物件中的相關資訊。所以這個ItemsCustom中有name屬性,假如我們要想將前臺傳進來的name屬性封裝到ItemsCustom中的name屬性中,該如何傳入呢?這就是包裝型別的pojo引數繫結問題。 
  很簡單,在前臺我們可以通過這種方式來傳: 
包裝pojo引數 
  然後controller中方法的形參傳入包裝型別的pojo,即ItemsQueryVo,打個斷點,即可檢視值有沒有傳進來。如下: 
後臺接收 
  這樣就能根據使用者傳進來的引數,進行復制的查詢操作了。

5. 集合型別的繫結

5.1 陣列的繫結

  陣列的繫結指的是前臺傳來多個同一型別的資料,我們在controller中使用陣列形參來接收前臺傳來的資料。還是以案例來驅動這部分內容,比如現在我們要批量刪除商品,那麼我們需要勾選好幾個商品,這些商品都有id號,在controller中,我們需要將這些id號全部獲取並放到一個陣列中,然後再根據陣列中的id號挨個刪除資料庫中對應的項。那麼該如何繫結呢?其實也很簡單,如下: 
  controller的方法定義為: 
陣列 
  對應前臺傳入的引數為: 
前臺傳入id 
  這樣就能將前臺傳入的多個id繫結到陣列中,然後我們就可以從陣列中拿出要刪除的商品的id了。

5.2 List的繫結

  通常在需要批量提交資料時,將提交的資料繫結到list<pojo>中,比如:成績錄入(錄入多門課成績,批量提交),在這裡我們假設有需求:批量商品修改,在頁面輸入多個商品資訊,將多個商品資訊提交到controller方法中,即一次性更新多個商品資訊。 
  所以思路是在擴充套件類ItemsQueryVo中新新增一個List<ItemsCustom>,然後將不同商品的資訊都存到這個List中,所以修改如下: 
List 
  controller方法的定義:

1、進入批量商品修改頁面 
2、批量修改商品提交

  所以controller中應該有兩個方法,如下: 
批量更新 
  前臺jsp頁面中是如何傳入引數的呢?這是我們所關心的問題,因為後臺形參中接收資料用的就是包裝類ItemsQueryVo。看下面: 
傳參 
  所以我們知道了,前臺是通過類似於list[i].name這種形式來傳遞引數的。list[i]表示第i個ItemsCustom,然後 list[i].屬性 表示第i個ItemsCustom中對應的屬性。

5.2 Map的繫結

  Map的繫結其實和List的繫結是差不多的,首先也是在包裝的pojo中新新增一個Map型別的屬性,如(我隨便舉個例子,跟本例無關了)

Public class QueryVo {
private Map<String, Student> itemInfo = new HashMap<String, Student>();
  //get/set方法..
}

關鍵是前臺傳參的時候和List不太一樣,Map是這樣傳的,比如:

<tr>
    <td>學生資訊:</td>
    <td>
        姓名:<input type="text" name="itemInfo['name']"/>
        年齡:<input type="text" name="itemInfo['price']"/>
    .. .. ..
    </td>
</tr>

  我們可以看到,Map的引數繫結傳來的是Map中的key,然後value會自動繫結到Map中的那個物件的屬性中。在controller中的方法裡,形參就直接使用QueryVo接收即可,也很簡單。 
  關於springmvc的引數繫結基本就總結到這了,其實原理都差不多,只是針對於不同的型別,繫結的方式有些區別而已,多想想多寫寫,基本就能掌握這些了。 

相關文章