查看“SpringMVC:高级应用”的源代码
←
SpringMVC:高级应用
跳到导航
跳到搜索
因为以下原因,您没有权限编辑本页:
您请求的操作仅限属于该用户组的用户执行:
用户
您可以查看和复制此页面的源代码。
[[category:SpringMVC]] == Validation == 对提交的请求数据进行检验。<br/> === 依赖包 === * hibernate-validator-4.3.0.Final.jar * jboss-logging-3.1.0.CR2.jar * validation-api-1.0.0.GA.jar === 配置validator === ==== springmvc.xml ==== {| class="wikitable" style="width: 100%;" |+ Converter配置 ! style="width:50%;"| 使用<mvc:annotation-driven> ! style="width:50%;"| 使用HandlerAdapter |- style="vertical-align:top;" | <syntaxhighlight lang="xml"> <mvc:annotation-driven validator="validator"></mvc:annotation-driven> </syntaxhighlight> | <syntaxhighlight lang="xml"> <!-- 注解适配器 --> <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"> <property name="webBindingInitializer" ref="customBinder"></property> </bean> <!-- 自定义webBinder --> <bean id="customBinder" class="org.springframework.web.bind.support.ConfigurableWebBindingInitializer"> <property name="validator" ref="validator" /> </bean> </syntaxhighlight> |- | colspan="2" | <syntaxhighlight lang="xml"> <!-- 校验器 --> <bean id="validator" class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean"> <!-- hibernate校验器--> <property name="providerClass" value="org.hibernate.validator.HibernateValidator" /> <!-- 指定校验使用的资源文件,在文件中配置校验错误信息,如果不指定则默认使用classpath下的ValidationMessages.properties --> <property name="validationMessageSource" ref="messageSource" /> </bean> <!-- 校验错误信息配置文件 --> <bean id="messageSource" class="org.springframework.context.support.ReloadableResourceBundleMessageSource"> <!-- 资源文件名--> <property name="basenames"> <list> <value>classpath:CustomValidationMessages</value> </list> </property> <!-- 资源文件编码格式 --> <!-- <property name="defaultEncoding" value="UTF-8"></property> --> <property name="fileEncodings" value="UTF-8" /> <!-- 对资源文件内容缓存时间,单位秒 --> <property name="cacheSeconds" value="120" /> </bean> </syntaxhighlight> |} ==== CustomValidationMessages.properties ==== 位置参考<syntaxhighlight lang="xml" inline><value>classpath:CustomValidationMessages</value> </syntaxhighlight>,内容如: <syntaxhighlight lang="xml"> #添加校验错误提交信息 items.name.length.error=请输入1到30个字符的商品名称 items.createtime.isNUll=请输入商品的生产日期 item.price.isNull=test,价格不能为空 items.message.test=校验信息配置测试文字 </syntaxhighlight> === 使用validator === ==== 添加验证规则 ==== 在pojo的属性之上添加验证规则:<syntaxhighlight lang="java" inline>@NotNull(message="{item.price.isNull}",groups= {ValidGroup1.class})</syntaxhighlight>,其中: # <syntaxhighlight lang="java" inline>@NotNull</syntaxhighlight>:校验名称; # <syntaxhighlight lang="java" inline>message="{item.price.isNull}"</syntaxhighlight>:错误信息(配置于“CustomValidationMessages.properties”); # <syntaxhighlight lang="java" inline>groups= {ValidGroup1.class}</syntaxhighlight>:此校验所属的分组; # <syntaxhighlight lang="java" inline>@NotNull</syntaxhighlight>:; <syntaxhighlight lang="java"> public class Items { private Integer id; //校验名称在1到30字符中间 //message是提示校验出错显示的信息 //groups:此校验属于哪个分组,groups可以定义多个分组 @Size(min=1,max=30,message="{items.message.test}",groups={ValidGroup1.class}) private String name; @NotNull(message="{item.price.isNull}",groups= {ValidGroup1.class}) private Float price; private String pic; //非空校验 @NotNull(message="{items.createtime.isNUll}",groups= {ValidGroup1.class}) private Date createtime; private String detail; //get、set、toString... } </syntaxhighlight> ==== 错误消息文件 ==== 验证规则中配置的<syntaxhighlight lang="java" inline>message="{item.price.isNull}"</syntaxhighlight>,对应于“CustomValidationMessages.properties”中的信息: <syntaxhighlight lang="java"> #添加校验错误提交信息 items.name.length.error=请输入1到30个字符的商品名称 items.createtime.isNUll=请输入商品的生产日期 item.price.isNull=test,价格不能为空 items.message.test=校验信息配置测试文字 </syntaxhighlight> * '''''如果在eclipse中编辑properties文件无法看到中文则参考“Eclipse开发环境配置-indigo.docx”添加propedit插件。'''''【???】 ==== 捕获错误 ==== 在Controller的方法形参前添加<syntaxhighlight lang="java" inline>@Validated</syntaxhighlight>来在参数绑定时进行校验,并将校验信息写入'''BindingResult'''中: # 在要校验的pojo后边添加<syntaxhighlight lang="java" inline>BingdingResult</syntaxhighlight>; # 一个<syntaxhighlight lang="java" inline>BindingResult</syntaxhighlight>对应一个pojo; <syntaxhighlight lang="java"> // 商品信息修改提交 // 在需要校验的pojo前边添加@Validated,在需要校验的pojo后边添加BindingResult // bindingResult接收校验出错信息 // 注意:@Validated和BindingResult bindingResult是配对出现,并且形参顺序是固定的(一前一后)。 // value={ValidGroup1.class}指定使用ValidGroup1分组的 校验 // @ModelAttribute可以指定pojo回显到页面在request中的key @RequestMapping("/editItemsSubmit") public String editItemsSubmit( Model model, HttpServletRequest request, Integer id, @ModelAttribute("items") @Validated(value = { ValidGroup1.class}) ItemsCustom itemsCustom, BindingResult bindingResult, MultipartFile items_pic//接收商品图片 ) throws Exception { // 获取校验错误信息 if (bindingResult.hasErrors()) { // 输出错误信息 List<ObjectError> allErrors = bindingResult.getAllErrors(); for (ObjectError objectError : allErrors) { // 输出错误信息 System.out.println(objectError.getDefaultMessage()); } // 将错误信息传到页面 model.addAttribute("allErrors", allErrors); // 可以直接使用model将提交pojo回显到页面 model.addAttribute("items", itemsCustom); // 出错重新到商品修改页面 return "items/editItems"; } //原始名称 String originalFilename = items_pic.getOriginalFilename(); //上传图片 if(items_pic!=null && originalFilename!=null && originalFilename.length()>0){ //存储图片的物理路径 String pic_path = "F:\\develop\\upload\\temp\\"; //新的图片名称 String newFileName = UUID.randomUUID() + originalFilename.substring(originalFilename.lastIndexOf(".")); //新图片 File newFile = new File(pic_path+newFileName); //将内存中的数据写入磁盘 items_pic.transferTo(newFile); //将新图片名称写到itemsCustom中 itemsCustom.setPic(newFileName); } // 调用service更新商品信息,页面需要将商品信息传到此方法 itemsService.updateItems(id, itemsCustom); return "success"; } </syntaxhighlight> ==== 错误回显 ==== 在捕获错误后,页面需要显示错误信息。<br/> 如:<br/> # 页头: #: <syntaxhighlight lang="jsp"> <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> <%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" <%@ taglib prefix="spring" uri="http://www.springframework.org/tags" %> </syntaxhighlight> # 在需要显示错误信息地方: #: <syntaxhighlight lang="jsp"> <spring:hasBindErrors name="item"> <c:forEach items="${errors.allErrors}" var="error"> ${error.defaultMessage }<br/> </c:forEach> </spring:hasBindErrors> </syntaxhighlight> #* <syntaxhighlight lang="jsp" inline><spring:hasBindErrors name="item"></syntaxhighlight>表示如果item参数绑定校验错误下边显示错误信息。 或,如:【???????????】<br/> # Controller: #: <syntaxhighlight lang="java"> if(bindingResult.hasErrors()){ model.addAttribute("errors", bindingResult); } </syntaxhighlight> # 在需要显示错误信息地方: #: <syntaxhighlight lang="jsp"> <c:forEach items="${errors.allErrors}" var="error"> ${error.defaultMessage }<br/> </c:forEach> </syntaxhighlight> === 分组校验 === 如果两处校验使用同一个Items类则可以设定校验分组,通过分组校验可以对每处的校验个性化。<br/> ==== 定义分组 ==== 分组就是一个标识,可以使用一个空接口来进行分组定义。<br/> <syntaxhighlight lang="java"> package cn.itcast.ssm.controller.validation; public interface ValidGroup1 { //接口中不需要定义任何方法,仅是对不同的校验规则进行分组 //此分组只校验商品名称长度 } </syntaxhighlight> <syntaxhighlight lang="java"> package cn.itcast.ssm.controller.validation; public interface ValidGroup2 { //接口中不需要定义任何方法,仅是对不同的校验规则进行分组 //此分组只校验xxxxxx } </syntaxhighlight> ==== 指定分组 ==== 在pojo的校验规则中使用<syntaxhighlight lang="java" inline>groups= {ValidGroup1.class}</syntaxhighlight>来指定所属的分组。<br/> <syntaxhighlight lang="java"> @Size(min=1,max=30,message="{items.message.test}",groups={ValidGroup1.class}) private String name; @NotNull(message="{item.price.isNull}",groups= {ValidGroup1.class}) private Float price; @NotNull(message="{items.createtime.isNUll}",groups= {ValidGroup1.class}) private Date createtime; </syntaxhighlight> ==== 使用分组校验 ==== 在Controller中使用<syntaxhighlight lang="java" inline>@Validated(value={ValidGroup1.class})</syntaxhighlight>来使用ValidGroup1分组的检验规则。<br/> <syntaxhighlight lang="java"> // 商品修改提交 @RequestMapping("/editItemSubmit") public String editItemSubmit( @Validated(value={ValidGroup1.class}) @ModelAttribute("item") Items items, BindingResult result, @RequestParam("pictureFile") MultipartFile[] pictureFile, Model model) throws Exception { ... } </syntaxhighlight> * 也可以使用逗号分隔来指定多个分组:<syntaxhighlight lang="java" inline>@Validated(value={ValidGroup1.class,ValidGroup2.class })</syntaxhighlight> === 校验注解 === {| class="wikitable" ! 校验 !! 说明 |- | @Null || 被注释的元素必须为 null |- | @NotNull || 被注释的元素必须不为 null |- | @AssertTrue || 被注释的元素必须为 true |- | @AssertFalse || 被注释的元素必须为 false |- | @Min(value) || 被注释的元素必须是一个数字,其值必须大于等于指定的最小值 |- | @Max(value) || 被注释的元素必须是一个数字,其值必须小于等于指定的最大值 |- | @DecimalMin(value) || 被注释的元素必须是一个数字,其值必须大于等于指定的最小值 |- | @DecimalMax(value) || 被注释的元素必须是一个数字,其值必须小于等于指定的最大值 |- | @Size(max=, min=) || 被注释的元素的大小必须在指定的范围内 |- | @Digits (integer, fraction) || 被注释的元素必须是一个数字,其值必须在可接受的范围内 |- | @Past || 被注释的元素必须是一个过去的日期 |- | @Future || 被注释的元素必须是一个将来的日期 |- | @Pattern(regex=,flag=) || 被注释的元素必须符合指定的正则表达式 |- ! colspan="2" | Hibernate Validator 附加的 constraint |- | @NotBlank(message =) || 验证字符串非null,且长度必须大于0 |- | @Email || 被注释的元素必须是电子邮箱地址 |- | @Length(min=,max=) || 被注释的字符串的大小必须在指定的范围内 |- | @NotEmpty || 被注释的字符串的必须非空 |- | @Range(min=,max=,message=) || 被注释的元素必须在合适的范围内 |} == 数据回显 == 数据回显:表单提交失败,回到表单填写页面时,将原提交的数据重新显示在页面中。<br/> === 简单类型 === 简单的数据类型的回显,使用'''Model'''将传入的参数添加到'''request域'''即可。 <syntaxhighlight lang="java"> @RequestMapping(value="/editItems",method={RequestMethod.GET}) public String editItems(Model model,Integer id)throws Exception{ //传入的id重新放到request域 model.addAttribute("id", id); ... } </syntaxhighlight> === pojo类型 === # '''默认回显''':pojo数据传入controller方法后,springmvc自动将pojo数据放到request域,; #: <syntaxhighlight lang="java"> @RequestMapping("/editItemSubmit") public String editItemSubmit(Integer id,ItemsCustom itemsCustom)throws Exception{} </syntaxhighlight> #: <syntaxhighlight lang="jsp"> <tr> <td>商品名称</td> <td><input type="text" name="name" value="${itemsCustom.name}"/></td> </tr> <tr> <td>商品价格</td> <td><input type="text" name="price" value="${itemsCustom.price}"/></td> </tr> </syntaxhighlight> #* 仅request的key等于pojo类型(首字母小写)时; #* 相当于调用下边的代码<syntaxhighlight lang="java" inline>model.addAttribute("itemsCustom", itemsCustom);</syntaxhighlight> # 使用'''@ModelAttribute'''将方法的返回值传到页面; #: <syntaxhighlight lang="java"> @RequestMapping("/editItemSubmit") public String editItemSubmit(Integer id,@ModelAttribute("item") ItemsCustom itemsCustom){} </syntaxhighlight> #: <syntaxhighlight lang="jsp"> <tr> <td>商品名称</td> <td><input type="text" name="name" value="${item.name}"/></td> </tr> <tr> <td>商品价格</td> <td><input type="text" name="price" value="${item.price}"/></td> </tr> </syntaxhighlight> # 最简单的方法是使用'''model''',而不用'''@ModelAttribute'''; #: <syntaxhighlight lang="java"> @RequestMapping("/editItemSubmit") public String editItemSubmit(Model model,ItemsCustom itemsCustom){ // 可以直接使用model将提交pojo回显到页面 model.addAttribute("items", itemsCustom); } </syntaxhighlight> #: <syntaxhighlight lang="jsp"> <tr> <td>商品名称</td> <td><input type="text" name="name" value="${item.name}"/></td> </tr> <tr> <td>商品价格</td> <td><input type="text" name="price" value="${item.price}"/></td> </tr> </syntaxhighlight> #* 通过形参中的model将model数据传到页面,相当于modelAndView.addObject方法; === 方法返回值 === 可以将方法返回值暴露为模型数据,传到视图页面。<br/> 如,将获取商品类型的方法的返回值,传递到页面: <syntaxhighlight lang="java"> // 商品分类 //itemtypes表示最终将方法返回值放在request中的key @ModelAttribute("itemtypes") public Map<String, String> getItemTypes() { Map<String, String> itemTypes = new HashMap<String, String>(); itemTypes.put("101", "数码"); itemTypes.put("102", "母婴"); return itemTypes; } </syntaxhighlight> <syntaxhighlight lang="jsp"> <select name="itemtype"> <c:forEach items="${itemtypes}" var="itemtype"> <option value="${itemtype.key}">${itemtype.value}</option> </c:forEach> </select> </syntaxhighlight> == 异常处理器 == springmvc在处理请求过程中出现异常信息交由异常处理器进行处理,自定义异常处理器可以实现一个系统的异常处理逻辑。<br/> 处理思路:<br/> * 系统的dao、service、controller出现都通过throws Exception向上抛出,最后由springmvc前端控制器交由异常处理器进行异常处理。 [[File:springMVC异常处理.png|500px]] === 自定义异常 === 为了区别不同的异常通常根据异常类型自定义异常类,这里我们创建一个自定义系统异常,如果controller、service、dao抛出此类异常说明是系统预期处理的异常信息。 <syntaxhighlight lang="java"> public class CustomException extends Exception { /** serialVersionUID*/ private static final long serialVersionUID = -5212079010855161498L; public CustomException(String message){ super(message); this.message = message; } //异常信息 private String message; public String getMessage() { return message; } public void setMessage(String message) { this.message = message; } } </syntaxhighlight> === 自定义异常处理器 === <syntaxhighlight lang="java"> public class CustomExceptionResolver implements HandlerExceptionResolver { @Override public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { ex.printStackTrace(); CustomException customException = null; //如果抛出的是系统自定义异常则直接转换 if(ex instanceof CustomException){ customException = (CustomException)ex; }else{ //如果抛出的不是系统自定义异常则重新构造一个未知错误异常。 customException = new CustomException("未知错误,请与系统管理 员联系!"); } ModelAndView modelAndView = new ModelAndView(); modelAndView.addObject("message", customException.getMessage()); modelAndView.setViewName("error"); return modelAndView; } } </syntaxhighlight> === 错误页面 === error.jsp: <syntaxhighlight lang="java"> <%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%> <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %> <%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="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>错误页面</title> </head> <body> 您的操作出现错误如下:<br/> ${message } </body> </html> </syntaxhighlight> === 配置异常处理器 === springmvc.xml中添加 <syntaxhighlight lang="xml"> <!-- 异常处理器 --> <bean id="handlerExceptionResolver" class="cn.itcast.ssm.controller.exceptionResolver.CustomExceptionResolver"/> </syntaxhighlight> === 异常测试 === 如:修改controller方法“editItem”,调用service查询商品信息,如果商品信息为空则抛出异常。 <syntaxhighlight lang="java"> // 调用service查询商品信息 Items item = itemService.findItemById(id); if(item == null){ throw new CustomException("商品信息不存在!"); } </syntaxhighlight> == 上传图片 == === 依赖包 === * commons-fileupload-1.2.2.jar * commons-io-2.4.jar === 配置上传目录 === 在服务器中(如:tomcat、nginx)上配置图片虚拟目录: # tomcat: ## 在配置文件“conf/server.xml”中添加:<syntaxhighlight lang="xml" inline><Context docBase="F:\develop\upload\temp" path="/pic" reloadable="false"/></syntaxhighlight> ## 或者,通过eclipse配置: ##: [[File:eclipse设置Tomcat.png|400px]] #: 访问http://localhost:8080/pic即可访问F:\develop\upload\temp下的图片。 === 配置解析器 === springmvc.xml中添加: <syntaxhighlight lang="java"> <!-- 文件上传 --> <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <!-- 设置上传文件的最大尺寸为5MB --> <property name="maxUploadSize"> <value>5242880</value> </property> </bean> </syntaxhighlight> === 上传测试 === # controller: #: <syntaxhighlight lang="java"> //商品修改提交 @RequestMapping("/editItemSubmit") public String editItemSubmit(Items items, MultipartFile pictureFile)throws Exception{ //原始文件名称 String pictureFile_name = pictureFile.getOriginalFilename(); if(pictureFile!=null && pictureFile_name!=null && pictureFile_name.length()>0){ //存储图片的物理路径 String pic_path = "F:\\develop\\upload\\temp\\"; //新文件名称 String newFileName = UUID.randomUUID().toString()+pictureFile_name.substring(pictureFile_name.lastIndexOf(".")); //上传图片 File uploadPic = new java.io.File(pic_path+newFileName); if(!uploadPic.exists()){ uploadPic.mkdirs(); } //向磁盘写文件 pictureFile.transferTo(uploadPic); } ... } </syntaxhighlight> # view: #: <syntaxhighlight lang="jsp"> <form id="itemForm" action="${pageContext.request.contextPath}/items/editItemsSubmit.action" method="post" enctype="multipart/form-data"> <input type="hidden" name="id" value="${items.id}"/> 修改商品信息: <table width="100%" border=1> ... <tr> <td>商品图片</td> <td> <c:if test="${items.pic!=null}"> <img src="/pic/${items.pic}" width=100 height=100/> <br/> </c:if> <input type="file" name="pictureFile"/> </td> </tr> ... </table> </form> </syntaxhighlight> #* form添加<syntaxhighlight lang="jsp" inline>enctype="multipart/form-data"</syntaxhighlight>; #* file的name与controller形参一致(为pictureFile); == json数据交互 == === 依赖包 === Springmvc默认用<syntaxhighlight lang="java" inline>MappingJacksonHttpMessageConverter</syntaxhighlight>对json数据进行转换,需要加入jackson的包: * jackson-core-asl-1.9.11.jar * jackson-mapper-asl-1.9.11.jar === 配置转换器 === 在注解适配器中加入messageConverters: <syntaxhighlight lang="xml"> <!--注解适配器 --> <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"> <property name="messageConverters"> <list> <bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean> </list> </property> </bean> </syntaxhighlight> * 使用<syntaxhighlight lang="xml" inline><mvc:annotation-driven /></syntaxhighlight>时无需配置以上内容。 === @RequestBody === <syntaxhighlight lang="xml" inline>@RequestBody</syntaxhighlight>注解用于读取http请求的内容(字符串),通过springmvc提供的<syntaxhighlight lang="xml" inline>HttpMessageConverter</syntaxhighlight>接口将读到的内容转换为json、xml等格式的数据并绑定到controller方法的参数上。 === @ResponseBody === <syntaxhighlight lang="xml" inline>@ResponseBody</syntaxhighlight>注解用于将Controller的方法返回的对象,通过<syntaxhighlight lang="xml" inline>HttpMessageConverter</syntaxhighlight>接口转换为指定格式的数据(如:json、xml等),通过Response响应给客户端。 === 请求json,响应json === 实际开发中常用,请求key/value数据,响应json结果,方便客户端对结果进行解析。<br/> controller: <syntaxhighlight lang="java"> // 商品修改提交json信息,响应json信息 @RequestMapping("/editItemSubmit_RequestJson") public @ResponseBody Items editItemSubmit_RequestJson(@RequestBody Items items) throws Exception { System.out.println(items); //itemService.saveItem(items); return items; } </syntaxhighlight> view:js方法<br/> 需引入jquery:<syntaxhighlight lang="xml" inline><script type="text/javascript" src="${pageContext.request.contextPath}/js/jquery-1.4.4.min.js"></script></syntaxhighlight> <syntaxhighlight lang="JavaScript"> //请求json响应json function request_json(){ $.ajax({ type:"post", url:"${pageContext.request.contextPath }/item/editItemSubmit_RequestJson.action", contentType:"application/json;charset=utf-8", data:'{"name":"测试商品","price":99.9}', success:function(data){ alert(data); } }); } </syntaxhighlight> 结果:可以看出请求的数据是json格式<br/> [[File:springMVC请求json响应json.png|400px]] === 请求key/value,响应json === 表单默认请求<syntaxhighlight lang="HTML" inline>application/x-www-form-urlencoded</syntaxhighlight>格式的数据即key/value,通常有post和get两种方法,响应json数据方便客户端处理。<br/> controller: <syntaxhighlight lang="java"> // 商品修改提交,提交普通form表单数据,响应json @RequestMapping("/editItemSubmit_ResponseJson") public @ResponseBody Items editItemSubmit_ResponseJson(Items items) throws Exception { System.out.println(items); //itemService.saveItem(items); return items; } </syntaxhighlight> view:js方法 <syntaxhighlight lang="JavaScript"> //请求application/x-www-form-urlencoded响应json function formsubmit(){ var user = "name=测试商品&price=99.9"; alert(user); $.ajax({ type:'post',//这里改为get也可以正常执行 url:'${pageContext.request.contextPath}/item/ editItemSubmit_RequestJson.action', //ContentType没指定将默认为:application/x-www-form-urlencoded data:user, success:function(data){ alert(data.name); } }) } </syntaxhighlight> :* 去掉ContentType定义,默认为:'''application/x-www-form-urlencoded'''格式 结果:可以看出请求的数据是标准的key/value格式<br/> [[File:springMVC请求key-value响应json.png|400px]] == RESTful支持 == == 拦截器 ==
返回至“
SpringMVC:高级应用
”。
导航菜单
个人工具
登录
命名空间
页面
讨论
大陆简体
已展开
已折叠
查看
阅读
查看源代码
查看历史
更多
已展开
已折叠
搜索
导航
首页
最近更改
随机页面
MediaWiki帮助
笔记
服务器
数据库
后端
前端
工具
《To do list》
日常
阅读
电影
摄影
其他
Software
Windows
WIKIOE
所有分类
所有页面
侧边栏
站点日志
工具
链入页面
相关更改
特殊页面
页面信息