每個星期一道菜,這個星期也不例外~~~
一個軟件,一個產品,都是一點點開發並完善起來的,功能越來越多,性能越來越強,用戶體驗越來越好……這每個指標的提高都需要切切實實的做點東西出來,好比,你的這個產品做大了,用的人多了,不僅僅再是上海人用,北京人用,還有印度人用,法國人用等等,可以說這個產品已經走上了國際化的大舞台。當印度的哥們輸入url訪問產品時,界面上彈出“歡迎您,三哥”,估計哥們當場就蒙圈了。而這個時候,國際化就應運而生了。
要做國際化這道菜,真的沒有想象中的那么復雜,反而很簡單,不信你看——
1. 注入ResourceBundleMessageSource
在SpringMVC.xml添加用於國際化處理的beanResourceBundlMessageSource
<bean id="messageSource" class="org.springframework.context.support.ResourceBundleMessageSource"> <property name="basename" value="i18n"></property> </bean>
這里property中的name是與注入類中的屬性名一直的,這里的value決定了后面國際化文件的名稱,記得是i18n,馬上你就會看到它的用法。
2. 創建國際化文件
總共需要創建三個國際化屬性文件
i18n.properties-默認的國際化文件
i18n_en_US.properties-用於英文環境的國際化文件
i18n_zh_CN.properties-用於中文環境的國際化文件
注意:這里為什么文件的名稱都是i18n開頭,因為在第一點的springmvc.xml配置文件中,配置的value值就是i18n
對於i18n.properties和i18n_en_US.properties文件的內容相同,如下
i18n.username=UserName i18n.password=Password
i18n_zh_CN.properties
i18n.username=\u7528\u6237\u540D i18n.password=\u5BC6\u7801
3.新建頁面
分別新建兩個頁面,一個是i18n.jsp,顯示用戶名,同時有跳轉到i18n2.jsp的超鏈接,另一個是i18n2.jsp,顯示密碼,同時有跳轉到i18n.jsp的超鏈接。
i18n.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body> <fmt:message key="i18n.username"></fmt:message><br><br> <a href="i18n2">i18n2</a> </body> </html>
i18n2.jsp
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib prefix="fmt" uri="http://java.sun.com/jsp/jstl/fmt" %> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body> <fmt:message key="i18n.password"></fmt:message><br><br> <a href="i18n">i18n</a> </body> </html>
同時,顯然我們需要在index.jsp中添加一個入口,鏈接到i18n.jsp頁面,如下
<a href="i18n">i18n</a>
為了能夠直接點擊就鏈接過去,而不需要通過handler處理並跳轉到視圖的套路,我們需要在springmvc.xml中添加標簽
<mvc:view-controller path="/i18n" view-name="i18n"/> <mvc:view-controller path="/i18n2" view-name="i18n2"/>
這樣就能實現直接在地址欄中直接訪問到i18n.jsp和i18n2.jsp頁面了。
小坑:如果i18n.jsp和i18n2.jsp中的編碼方式采用默認“ISO-8859-1”,就會出現頁面顯示亂碼
當把編碼改為“UTF-8”后,就能夠正常顯示
以上是國際化這道菜的基礎做法,那么如果我還想做一道不用直接訪問i18n.jsp,而是經過handler處理后呈現的i18n.jsp,或者我還想做一道不用那么麻煩還要切換語言的國際化菜,有沒有可能,當然,接着看——
1. 注釋之前在springmvc.xml添加對於i18n.jsp直接訪問的標簽
<!-- <mvc:view-controller path="/i18n" view-name="i18n"/> -->
2. 在Hanlder處理類SpringMVCTest中添加處理接口
@Autowired private ResourceBundleMessageSource messageSource; @RequestMapping("/i18n") public String testI18n(Locale locale){ String val = messageSource.getMessage("i18n.username", null, locale); System.out.println(val); return "i18n"; }
注意這里注入了國際化處理類ResourceBundleMessageSource,並使用其getMessage方法獲取國際化后的屬性值。
啟動tomcat服務可以看到
那么如果根據自己的設定,在不同的語言環境中顯示相應語言的信息呢
1. 配置SessionLocaleResolver和LocaleChangeInterceptor
<!-- 配置SessionLocaleResolver --> <bean id="localeResolver" class="org.springframework.web.servlet.i18n.SessionLocaleResolver"></bean> <!-- 配置LocaleChangeInterceptor --> <mvc:interceptors> <bean class="org.springframework.web.servlet.i18n.LocaleChangeInterceptor"></bean> </mvc:interceptors>
這里的LocaleChangeInterceptor主要用於將帶有locale信息的請求解析為一個Locale對象,並得到一個LocaleResolver對象
之后這里的SessionLocalResolver就會將上面的LocalResolver對象轉化為Session的屬性,並從中取出這個屬性,也就是Locale對象,返回給應用程序。
2. 在index.jsp中添加超鏈接
<a href="i18n?locale=zh_CN">中文</a> <br><br> <a href="i18n?locale=en_US">英文</a>
這樣,我們就可以看到結果
說完國際化,再來說說SpringMVC對於json的支持。
在傳統的開發過程中,我們的handler即controller層通常遵循需要轉向一個JSP視圖的套路;但是這樣的場景並不能滿足所有的要求,比如我們很多時候只需要返回數據即可,而不是一個JSP頁面。那么這時候SPRING MVC3的@ResponseBody和@ResponseEntity就支持這樣的功能。Controller直接返回數據(這里我們說說json數據),而不是直接指向具體的視圖。這里簡單的分別舉一個上傳和下載的例子。
1. 文件上傳
1.1 使用jquery在index.jsp實現ajax請求
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> <script type="text/javascript" src="scripts/jquery-1.9.1.min.js"></script> <script> $(function(){ $("#testJson").click(function(){ var url = this.href; var args = {}; $.post(url, args, function(data){ for(var i=0; i<data.length; i++){ var id = data[i].id; var lastName = data[i].lastName; alert(id + ": " + lastName); } }) return false; }) }) </script> </head> <body> <a href="emps">list all employees</a><br/><br/> <a href="testJson" id="testJson">testJson</a> </body> </html>
這里核心的就是用jquery寫的ajax請求
請求的url就是定義的href;
data為請求響應后返回的數據;
正常情況下,我們應該請求到所有員工的信息,並且通過這里的遍歷得到每一個員工的所有信息如id、lastName等
1.2. 這里我們需要引入三個jar包
jackson-annotation-2.1.5.jar
jackso-core-2.1.5.jar
jackso-databind-2.1.5.jar
這三個主要是用於后面在返回數據的轉換中用到的。
1.3. 在handler SpringMVCTest中添加接口
@ResponseBody @RequestMapping("testJson") public Collection<Employee> testJson(){ return employeeDao.getAll(); }
這里我的個人理解,就是將通過employeeDao查詢到的所有的員工信息,作為響應返回給接口,並最終通過一系列處理得到一個json的數據形式,然后在前台頁面中遍歷解析。而完成這一切就是歸功於注解@ResponseBody。
具體來說,是有內部的一些converter來做這些轉換的,在接口方法中打斷點,進入調試
選擇DispatcherServlet,找到this->handleradapters->elementData,在這個數組中找到RequestMappingHandlerAdapter,點進去找到messageConverters,便可以看到總共有7個converters
這里第7個MappingJackson2HttpMessageConverter就是我們添加了以上三個jar包后才加載進來的converter。可以看出,這里有足夠過對於不同數據類型處理的轉換器。
1.4 在index.jsp中添加鏈接
<form action="testFileUpload" method="POST" enctype="multipart/form-data"> File: <input type="file" name="file"/> Desc: <input type="text" name="desc"/> <input type="submit" value="Submit"/> </form><br/>
最終的上傳結果如下
2. 文件下載
2.1 准備下載源
在WebContent下新建files目錄,放入aaa.txt,作為下載源
2.2 在index.jsp添加超鏈接作為下載入口
<a href="testResponseEntity" id="testJson">testResponseEntity</a><br/>
2.3 在handler SpringMVCTest中添加接口
@RequestMapping("testResponseEntity") public ResponseEntity<byte[]> testResponseEntity(HttpSession session) throws IOException{ byte[] body = null; ServletContext servletContext = session.getServletContext(); InputStream in = servletContext.getResourceAsStream("/files/aaa.txt"); body = new byte[in.available()]; in.read(body); HttpHeaders headers = new HttpHeaders(); headers.add("Content-Disposition", "attachment;filename=aaa.txt"); HttpStatus statusCode = HttpStatus.OK; ResponseEntity<byte[]> response = new ResponseEntity<>(body, headers, statusCode); return response; }
啟動tomcat,我們可以看到aaa.txt真的可以下載啦~~~
好了,到此為止,我們都干了啥
1. 支持國際化
2. 文件上傳
3. 文件下載
如果您覺得閱讀本文對您有幫助,請點一下“推薦”按鈕,您的“推薦”將是我最大的寫作動力!如果您想持續關注我的文章,請掃描二維碼,關注JackieZheng的微信公眾號,我會將我的文章推送給您,並和您一起分享我日常閱讀過的優質文章。
友情贊助
如果你覺得博主的文章對你那么一點小幫助,恰巧你又有想打賞博主的小沖動,那么事不宜遲,趕緊掃一掃,小額地贊助下,攢個奶粉錢,也是讓博主有動力繼續努力,寫出更好的文章^^。
1. 支付寶 2. 微信