springmvc是用來處理頁面的一些請求,然后將數據再通過視圖返回給用戶的,前面的幾篇博文中使用的都是靜態數據,為了能快速入門springmvc,在這一篇博文中,我將總結一下springmvc中如何接收前台頁面的參數,即springmvc中的參數綁定問題。
1. 參數綁定的過程
我們可以回憶一下,在struts2中,是通過在Action中定義一個成員變量來接收前台傳進來的參數,而在springmvc中,接收頁面提交的數據是通過方法形參來接收的。從客戶端請求的key/value數據,經過參數綁定,將key/value數據綁定到controller方法的形參上,然后就可以在controller中使用該參數了。來看一下這個過程:
所以我們知道,是springmvc提供了很多轉換器來將頁面參數綁定到controller方法的形參上,關於自定義converter,我下面會提到。大概了解了該過程后,下面開始做具體的總結。
2. 默認支持的類型
springmvc中,有支持的默認類型的綁定。也就是說,直接在controller方法形參上定義默認類型的對象,就可以使用這些對象。
- HttpServletRequest對象
- HttpServletResponse對象
- HttpSession對象
- Model/ModelMap對象
在參數綁定過程中,如果遇到上面類型就直接進行綁定。也就是說,我們可以在controller的方法的形參中直接定義上面這些類型的參數,springmvc會自動綁定。這里要說明一下的就是Model/ModelMap對象,Model是一個接口,ModelMap是一個接口實現 ,作用是將Model數據填充到request域,跟ModelAndView類似,關於它的使用,我在下面和簡單類型參數綁定一起說。
3. 簡單類型的綁定
總結這個還是以需求為例吧,這樣比較容易理解,假設現在有個需求:根據商品的id來修改對應點商品信息。所以前台頁面肯定要傳進來該商品的id,然后springmvc的controller進行處理,返回一個修改商品信息的頁面。關於前台頁面的東西都很簡單,我就不貼代碼了,主要部分截個圖,具體的代碼在文章最后有下載地址。
前台頁面通過url將參數傳遞過來,請求的是editItems.action。
下面寫controller中的editItems方法:
3. 簡單類型的綁定
總結這個還是以需求為例吧,這樣比較容易理解,假設現在有個需求:根據商品的id來修改對應點商品信息。所以前台頁面肯定要傳進來該商品的id,然后springmvc的controller進行處理,返回一個修改商品信息的頁面。關於前台頁面的東西都很簡單,我就不貼代碼了,主要部分截個圖,具體的代碼在文章最后有下載地址。
前台頁面通過url將參數傳遞過來,請求的是editItems.action。
下面寫controller中的editItems方法:
這是個很簡單的demo,從上面的代碼中可以看出model可以直接作為參數,springmvc默認會綁定它,然后使用model將查詢到的數據放到request域中,這樣就可以在前台頁面取出該數據了。
要注意一點的是,簡單類型的綁定中,方法形參中的參數名要和前台傳進來的名一樣才能完成參數的綁定。那有人要問了,如果有特殊需求(比如更好的可讀性?),這里定義的參數名就是不一樣,那咋整呢?有解決辦法么?有!我們可以使用注解@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的方法中,如下:
這樣就能將前台表單傳進來的不同屬性值封裝到ItemsCustom中了。但是運行一下就會發現報錯,報錯的信息是無法將String類型轉換成java.util.Date類型,因為上面Items中有一個屬性是Date類型的createtime。這就需要我們自己定義轉換器了,這也是這部分的重點,下面我們自己定義一個日期轉換器:
定義好了轉換器后,需要在springmvc.xml中進行如下配置:
現在就可以了,springmvc就能根據這個轉換器將String類型正確轉換成Date類型,然后封裝到ItemsCustom中去了。
這里說一個小小的插曲:修改商品詳細信息后提交,可能會有中文亂碼問題,表達提交都是post方式,springmvc中關於post方式的中文亂碼問題可以在web.xml中配置一個過濾器來解決,如下:
4.2. 包裝的pojo類型
這個包裝類型pojo與上面普通的pojo有啥區別呢?包裝類型pojo指的是pojo中有另一個也是pojo的屬性,即pojo套pojo,為什么會設計這種pojo呢?在前面的博文中我也有提到,這種組合的設計方法對於后期程序的擴展很有用,比如復雜的查詢條件就需要包裝到這種包裝類型中。
那么該如何綁定呢?有兩個思路:
- 在形參中添加HttpServletRequest request參數,通過request接收查詢條件參數。
- 在形參中讓包裝類型的pojo接收查詢條件參數。
第一種方式就跟原始servlet差不多,這里使用第二種方法,我們傳進來一個包裝類型的pojo。看一下這個包裝類型的pojo:
這個包裝pojo中還有一個ItemsCustom類,這個類繼承了Items類,並且用來擴展與Items相關的User對象中的相關信息。所以這個ItemsCustom中有name屬性,假如我們要想將前台傳進來的name屬性封裝到ItemsCustom中的name屬性中,該如何傳入呢?這就是包裝類型的pojo參數綁定問題。
很簡單,在前台我們可以通過這種方式來傳:
然后controller中方法的形參傳入包裝類型的pojo,即ItemsQueryVo,打個斷點,即可查看值有沒有傳進來。如下:
這樣就能根據用戶傳進來的參數,進行復制的查詢操作了。
5. 集合類型的綁定
5.1 數組的綁定
數組的綁定指的是前台傳來多個同一類型的數據,我們在controller中使用數組形參來接收前台傳來的數據。還是以案例來驅動這部分內容,比如現在我們要批量刪除商品,那么我們需要勾選好幾個商品,這些商品都有id號,在controller中,我們需要將這些id號全部獲取並放到一個數組中,然后再根據數組中的id號挨個刪除數據庫中對應的項。那么該如何綁定呢?其實也很簡單,如下:
controller的方法定義為:
對應前台傳入的參數為:
這樣就能將前台傳入的多個id綁定到數組中,然后我們就可以從數組中拿出要刪除的商品的id了。
List的綁定
通常在需要批量提交數據時,將提交的數據綁定到list<pojo>
中,比如:成績錄入(錄入多門課成績,批量提交),在這里我們假設有需求:批量商品修改,在頁面輸入多個商品信息,將多個商品信息提交到controller方法中,即一次性更新多個商品信息。
所以思路是在擴展類ItemsQueryVo中新添加一個List<ItemsCustom>
,然后將不同商品的信息都存到這個List中,所以修改如下:
controller方法的定義:
1、進入批量商品修改頁面
2、批量修改商品提交
所以controller中應該有兩個方法,如下:
前台jsp頁面中是如何傳入參數的呢?這是我們所關心的問題,因為后台形參中接收數據用的就是包裝類ItemsQueryVo。看下面:
所以我們知道了,前台是通過類似於list[i].name這種形式來傳遞參數的。list[i]表示第i個ItemsCustom,然后 list[i].屬性 表示第i個ItemsCustom中對應的屬性。
所以我們知道了,前台是通過類似於list[i].name這種形式來傳遞參數的。list[i]表示第i個ItemsCustom,然后 list[i].屬性 表示第i個ItemsCustom中對應的屬性。
Map的綁定
Map的綁定其實和List的綁定是差不多的,首先也是在包裝的pojo中新添加一個Map類型的屬性,如(我隨便舉個例子,跟本例無關了)
關鍵是前台傳參的時候和List不太一樣,Map是這樣傳的,比如:
我們可以看到,Map的參數綁定傳來的是Map中的key,然后value會自動綁定到Map中的那個對象的屬性中。在controller中的方法里,形參就直接使用QueryVo接收即可,也很簡單。
關於springmvc的參數綁定基本就總結到這了,其實原理都差不多,只是針對於不同的類型,綁定的方式有些區別而已,多想想多寫寫,基本就能掌握這些了。
#************************************************************************************************************************************************
org.springframework.ui.Model(接口)
傳遞數據,Spring框架自動創建它的一個實現類,如BindingAwareModelMap
org.springframework.ui.ModelMap(類):
傳遞數據,Spring框架自動創建modelmap的實例,並作為controller方法的參數傳入,用戶無需自己創建對象。
ModelMap對象主要用於把controller方法處理的數據傳遞到jsp界面,在controller方法中把jsp界面需要的數據放到ModelMap對象中即可。
它的作用類似request對象的setAttribute方法。在視圖層通過request找到ModelMap中的數據。
Modelmap本身不能設置頁面跳轉的url,可以通過controller方法的返回值來設置跳轉的url。
org.springframework.web.servlet.ModelAndView(類):
設置轉向地址,;傳遞數據ModelAndView的實例是由用戶手動創建的,這也是和ModelMap的一個區別,
controller方法的返回值如果是ModelAndView,則其即包含模型數據信息,又包含視圖信息,
這樣SpringMVC將使用包含的視圖對模型數據進行渲染,可以簡單地將模型數據看成一個Map<String, Object>對象。
使用addObject方法即可ModelAndView addObject(String attributeName, @Nullable Object attributeValue)
使用ModelAndView傳參
使用request傳參
使用session傳參