控制器方法的返回值
package com.gzy.web.controller;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.itheima.domain.User;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Date;
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping("/test1")
public ModelAndView test1(ModelAndView modelAndView){
String s = new Date().toString();
modelAndView.addObject("content",s);
modelAndView.setViewName("user/test");
return modelAndView;
}
/**
* 可以采用原始的方案来做 傻子才做呢
* @param request
* @param response
* @throws ServletException
* @throws IOException
*/
@RequestMapping("/test2")
public void test2(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//完全按照以前的写法
String s = new Date().toString();
request.setAttribute("content",s);
request.getRequestDispatcher("/WEB-INF/pages/user/test.jsp").forward(request,response);
}
/**
* 玩重定向原始方案
* @param request
* @param response
* @throws ServletException
* @throws IOException
*/
@RequestMapping("/test3")
public void test3(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//完全按照以前的写法
response.sendRedirect("http://www.baidu.com");
}
/**
* 玩重定向 springmvc方式
* @throws ServletException
* @throws IOException
*/
@RequestMapping("/test4")
public ModelAndView test4(ModelAndView modelAndView) throws ServletException, IOException {
modelAndView.setViewName("redirect:http://www.baidu.com");
//return "redirect:http://www.baidu.com";
return modelAndView;
}
/**
* 玩重定向 springmvc方式 这个好
*redirect相当于“response.sendRedirect(url)”。需要注意的
是, 如果是重定向到 jsp 页面,则 jsp 页面不
能写在 WEB-INF 目录中,否则无法找到。
* @throws ServletException
* @throws IOException
*/
@RequestMapping("/test5")
public String test5() throws ServletException, IOException {
return "redirect:http://www.baidu.com";
}
/**
* springmvc方 没有必须非要返回 modelandview 完全可以返回字符串
*
* 如果字符串 以redirect起头 重定向了 如果不是 返回的是一个视图
* springmvc 自动包装成一个 modelAndView
*
* //先开始springmvc代码
*
* //找到你的方法
*
* //帮你创建 model对象
* var model =new model();
*
* var obj=userController.test(model);
*
* if(obj instanceof modelandview){
* //之前的正常逻辑
* }else if(obj instanceof modelandview){
* //检查是否已redirect开头
* //是否 直接 resonse.sendRedirect(...)
* //不是以它开头
* new modelandview().setViewName(obj);
* modelandview.setmodel(model);
*
* }
* @throws ServletException
* @throws IOException
*/
@RequestMapping("/test6")
public String test6(Model model) {
String s = new Date().toString();
model.addAttribute("content",s);
return "user/test";
}
交互 json 数据
ResponseBody
作用:
该注解用于将 Controller 的方法返回的对象,通过 HttpMessageConverter 接口转换为指定格式的
数据如:json,xml 等,通过 Response 响应给客户端。
用法
1.配置springmvc.xml文件(配json转换器)
<mvc:annotation-driven>
<mvc:message-converters>
<bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"/>
</mvc:message-converters>
</mvc:annotation-driven>
2.使用注解
@RequestMapping("test5")
//告诉程序以json字符串返回给客户端
@ResponseBody
public User test5() {
User user = new User(1, "张撒", "44");
return user;
}
RequestBody 使用说明
作用:
用于获取请求体内容。直接使用得到是 key=value&key=value...结构的数据。
get 请求方式不适用。
属性:
required:是否必须有请求体。默认值是:true。当取值为 true 时,get 请求方式会报错。如果取值为 false,get 请求得到是 null。
使用
举个例子
1.一个html页面ajax发送数据
注:后台得到前端返回的数据后要有请求头:
contentType:"application/json;charset=utf-8"
否则后台无法转成json数据。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<script src="/static/js/jquery.min.js"></script>
<script>
function f() {
$.ajax({
url:"/user/test6",
type:"post",
contentType:"application/json;charset=utf-8",
data:"{\"name\":\"xiaoming\",\"age\":\"18\",\"id\":2}",
success:function(data){
alert(data)
}
})
}
</script>
</head>
<body>
<h3>演示ajax发送请求 并且发送的数据 不再是传统键值对了 而是json格式的数据</h3>
<input type="button" value="点我发请求" onclick="f()">
</body>
</html>
前端效果:
2.后台代码
@RequestMapping("test6")
@ResponseBody
public String test6(@RequestBody User user) {
System.out.println(user);
return "success";
}
后台效果:
Restful 风格的 URL
restful风格的url设计
传统风格设计:
商品的增删改查
1.查询某个商品
/product/findById
参数 id=xxxx
2.添加一个商品
/product/save
参数 id=xxx&name=xx&price=xxx
3.更新一个商品
/product/update
参数 id=xxx&name=xx&price=xxx
4.删除一个商品
/product/delById
参数 id=xxx
5.查询所有商品
/product/findAll
restful风格:
商品的增删改查
1.查询某个商品
路径 请求方式
/product/{id} GET
2.添加一个商品
路径 请求方式
/product PUT
参数 id=xxx&name=xx&price=xxx
3.更新一个商品
路径 请求方式
/product POST
参数 id=xxx&name=xx&price=xxx
4.删除一个商品
路径 请求方式
/product/{id} DELETE
5.查询所有商品
路径 请求方式
/products GET
总结:
就将放在服务器上任何当做一个资源
操作 增删改查 不在路径写动作名词 而是采用请求方式 表明动作
优点:
路径看起来简洁
缺点:
表现力还不够
注意在本身设计想法中 只要出现名词 代表的是另外一个资源
商品{
名字:
价格:
id:
产地:{
xx省
xx市
}
}
/product/1/address GET
分页查询
/product/_page
根据商品名字
/product/_query
使用
举个例子
前端
注意
规定用<input type="hidden" name="_method" value="xxx">
来模拟特殊请求。(name必须叫"_method" ),框架底层会自动识别到value中的值及请求方式。
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<h3>模拟restful风格请求方式</h3>
<fieldset>
<lengend>模拟post请求</lengend>
<form action="/product" method="post">
<input type="hidden" name="id" value="2">
名字:<input type="text" name="name"><br>
价格:<input type="text" name="price"><br>
<input type="submit" value="点我提交更新">
</form>
</fieldset>
<fieldset>
<lengend>模拟put请求 添加商品</lengend>
<form action="/product" method="post">
<input type="hidden" name="_method" value="PUT">
名字:<input type="text" name="name"><br>
价格:<input type="text" name="price"><br>
<input type="submit" value="点我提">
</form>
</fieldset>
<fieldset>
<lengend>模拟delete 添加商品</lengend>
<form action="/product" method="post">
<input type="hidden" name="_method" value="delete">
<input type="text" name="id">
<input type="submit" value="点我提交">
</form>
</fieldset>
</body>
</html>
web.xml配置
配置org.springframework.web.filter.HiddenHttpMethodFilte过滤器,对模拟的请求方式睁一只眼闭一只眼。
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
version="3.1">
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:dispatcher-servlet.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
<!--
这个地方你可以 *.do *.action / 千万不要写/*
/* 属于目录匹配的
/属于最后那个特殊
实际含义 / 和/* 表示匹配所有
因为 *.jsp的存在
精确匹配 目录匹配 后缀名匹配 /
第二个方案 写 *.do 你在访问的时候 必须写xxx.do
-->
</servlet-mapping>
<!--
继续将静态资源访问 交给默认servlet来处理
-->
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.css</url-pattern>
<url-pattern>*.png</url-pattern>
<url-pattern>*.html</url-pattern>
<url-pattern>*.htm</url-pattern>
<url-pattern>*.txt</url-pattern>
<url-pattern>*.jpg</url-pattern>
<url-pattern>*.js</url-pattern>
</servlet-mapping>
<filter>
<filter-name>encodingFilter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>utf-8</param-value>
</init-param>
<!--<init-param>
<param-name>forceResponseEncoding</param-name>
<param-value>true</param-value>
</init-param>-->
</filter>
<filter-mapping>
<filter-name>encodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<filter>
<filter-name>hiddenMethodFilter</filter-name>
<filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>hiddenMethodFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
</web-app>
后台
@Controller
public class ProductController {
@RequestMapping(value = "/product/{id}",method = RequestMethod.GET)
@ResponseBody
public Product findById(@PathVariable("id") int id){
//mock呗
Product product = new Product();
product.setId(id);
product.setName("虾米手机");
product.setPrice(2999);
return product;
}
@RequestMapping(value = "/product",method = RequestMethod.POST)
@ResponseBody
public String update(Product product){
//mock呗
System.out.println(product);
return "success";
}
@RequestMapping(value = "/product",method = RequestMethod.PUT)
@ResponseBody
public String save(Product product){
//mock呗
System.out.println(product);
return "success";
}
@RequestMapping(value = "/product",method = RequestMethod.DELETE)
@ResponseBody
public String delete(int id){
//mock呗
System.out.println(id);
return "success";
}
@RequestMapping(value = "/product",method = RequestMethod.GET)
@ResponseBody
public List findAll(){
//mock呗
ArrayList<Object> list = new ArrayList<>();
Product product1 = new Product();
product1.setId(1);
product1.setName("虾米手机");
product1.setPrice(2999);
list.add(product1);
Product product2 = new Product();
product2.setId(2);
product2.setName("华为手机");
product2.setPrice(4999);
list.add(product2);
return list;
}
}
springMVC方式文件上传
1.添加Maven依赖
<!-- https://mvnrepository.com/artifact/commons-fileupload/commons-fileupload -->
<!--apache提供了 一个工具类 来帮助解析这种格式-->
<dependency>
<groupId>commons-fileupload</groupId>
<artifactId>commons-fileupload</artifactId>
<version>1.4</version>
</dependency>
<!-- https://mvnrepository.com/artifact/commons-io/commons-io -->
<!---操作流的工具包-->
<dependency>
<groupId>commons-io</groupId>
<artifactId>commons-io</artifactId>
<version>2.6</version>
</dependency>
2.注入文件解析对象
<!--这里id必须叫multipartResolver,与底层有关-->
<bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
<property name="maxUploadSize" value="100000000"/>
</bean>
3.前端开发
注意:
springmvc对于文件上传的支持
完成文件上传本质就是流的传输
1.必须有一个表单
2.表单提交方式必须post
3.表单必须存在file框
4.表单属性 必须改 enctype="multipart/form-data",默认提交的数据是键值对形式的,设置此属性表示,有什么数据就提交什么数据。
代码:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>上传</title>
</head>
<body>
<form enctype="multipart/form-data" method="post" action="/upload">
上传头像:<input type="file" name="avatar">
<br><input type="submit" value="上传">
</form>
</body>
</html>
后端:
小问题:
1.文件名重复问题
拿到文件 就名字改了 保证唯一了 廉价 uuid
2.某个目录文件过多问题
分散目录
package com.gzy.controller;
import org.apache.commons.io.IOUtils;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.multipart.MultipartFile;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.Random;
import java.util.UUID;
@Controller
public class UploadController {
@RequestMapping("upload")
@ResponseBody
//注意:这里avatar必须与前端的<input type="file"name="avatar">
//的name 一致。
public String upload(MultipartFile avatar) throws IOException {
//获取上传文件名
String filename = avatar.getOriginalFilename();
System.out.println(filename);
//防止文件名重复利用uuid 获取随机名称
String uuidName = getUUIDName(filename);
System.out.println(uuidName);
//得到上传文件流对象
InputStream inputStream = avatar.getInputStream();
//获取随机路径
String secondDir = randDir();
String path="E:\\"+secondDir;
mustExist(path);
FileOutputStream fileOutputStream = new FileOutputStream(path+uuidName);
IOUtils.copy(inputStream,fileOutputStream);
inputStream.close();
fileOutputStream.close();
return "success";
}
/**
* 获取随机名称
* @param realName 真实名称
* @return uuid 随机名称
*/
public String getUUIDName(String realName){
//realname 可能是 1.jpg 也可能是 1
//获取后缀名
int index = realName.lastIndexOf(".");
if(index==-1){
return UUID.randomUUID().toString().replace("-", "").toUpperCase();
}else{
return UUID.randomUUID().toString().replace("-", "").toUpperCase()+realName.substring(index);
}
}
/**
* 获取文件目录,可以获取256个随机目录
* @return 随机目录 /a/4/ /b/c/
*/
public String randDir(){
String s="0123456789ABCDEF";
Random r = new Random();
return "/"+s.charAt(r.nextInt(16))+"/"+s.charAt(r.nextInt(16))+"/";
}
//判断文件夹是否存在,否则创建
public static void mustExist(String path){
File file = new File(path);
if (!file.exists()){
file.mkdirs();
}
}
}
SpringMVC全局异常处理
自定义异常处理器
package com.gzy.excition;
import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class MyExceptinHander implements HandlerExceptionResolver {
@Override
public ModelAndView resolveException(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) {
ModelAndView modelAndView = new ModelAndView();
if(e instanceof ArithmeticException){
modelAndView.addObject("code","1234");
}
modelAndView.setViewName("error/beautiful");
return modelAndView;
}
}
springmvc.xml配置异常处理器
<bean id="error" class="com.gzy.excition.MyExceptinHander"/>
错误演示
代码:
@RequestMapping("test7")
@ResponseBody
public String test7() {
int i=8/0;
return "success";
}
异常跳转页面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h3>sorry!!你访问页面去浪了.........</h3>
<hr>
不好意思,发生错误:${code}
<br>
如有疑问,请拨打:1231992213
</body>
</html>
效果