SpringMVC(1)

一)什么是SpringMVC:

Web项目:基于HTTP协议,当一个用户从浏览器上面输入URL地址之后,URL能够和我们的程序映射起来,可以让用户的请求触达到后端程序里面,并且根据程序的处理,把结果返回浏览器;

Spring MVC要进行学习的内容:

1)连接的功能:将用户浏览器和Java程序进行连接起来,也就是说访问一个url地址可以调用到我们的Spring程序,能够匹配到相应的方法,实现用户到程序的映射;

2)获取参数:用户访问的时候会带一些参数,要想办法在程序中获取到这些参数,表单文件,Header里面的参数,服务器端要获取到用户的请求参数;

3)输出数据的功能:执行了业务逻辑之后,服务器端要把程序执行的结果返回给用户前端

Spring框架是一个大的体系,SpringWebMVC只是Spring体系中的一个WEB模块,之前在去学习Spring的时候,是没有通过浏览器来访问我们的Spring程序的,而是通过Application中的main方法去执行Spring框架的,因为之前创建的是一个SpringCore项目,不是一个SpringWEBMVC项目,就不可以通过浏览器访问;

SpringWebMVC

Spring和SpringMVC诞生的历史比较久远,在他们之后才有SpringBoot

1)SpringMVC框架是基于ServletAPI和MVC设计思想构建的原始Web框架,但是从一开始就包含在Spring FrameWork框架里面,Spring框架是一个大的体系,Spring MVC只是属于Spring这个体系的Web框架模块SpringMVC其实就是Spring中的一个模块,主要处理Web开发的路径映射和视图渲染,属于Spring框架中Web层开发的一部分;

2)SpringBoot脚手架是为了快速开发Spring框架而诞生的一种框架一种技术,就是为了让大家用Spring用的比较爽的一个框架,就是一个壳,核心还是开发Spring,SpringBoot是spring提供的一个快速开发工具包,让程序员能更方便、更快速的开发spring和springmvc应用,简化了配置约定了默认配置,是最后诞生的,出生比较晚,有单独创建框架的一种模式;

3)我们既可以根据Spring创建一个Spring Core项目,也是可以根据Spring创建一个Spring Web项目;

4)他的全称为Spring Web MVC是基于Servlet API构建的原始Web框架,但是他一开始就包含在Spring框架里面,是SpringBoot的一个WEB模块,Spring MVC是一个Web框架,Spring MVC是基于Servlet API进行实现的,MVC和SpringMVC的关系就类似于IOC和DI的关系;

5)在进行创建SpringBoot项目的时候,进行勾选的Spring Web选项其实就是Spring MVC框架,我们可以看到从pom.xml里面就可以看到spring-boot-starter-web

我们加上Spring Boot DevTools是为了可以完成热部署的;

1)SpringMVC是一个基于MVC设计模式思想和ServletAPI接口实现的一个web框架,同时SpringMVC又是Spring框架中的一个web模块,是随着Spring的诞生而存在的一个Web框架

2)现在市面上面的绝大多数Spring项目都是基于SpringBoot,通过自己创建的main方法启动一个Spring框架或者是SpringCore项目,而Spring的核心就是SpringMVC,所以说SpringMVC的项目是基于SpringBoot来进行更简便的开发的,我们基于Spring Boot框架添加一个Spring Web依赖,此时的项目就变成了Spring MVC项目;

3)MVC架构分层的主要作用是解耦,采用分层架构的好处,普遍接受的是系统分层有利于系统的维护,系统的扩展,就是增强系统的可维护性和可扩展性;

MVC通常是Model View Controller,它是软件工程中的一种常见的设计模式,他把软件项目分为模型,视图和控制器三个部分

1)控制器(Controller):通常是应用程序中用来处理用户交互的部分,并向数据模型层发送数据,类似于保安,验证用户合法性和用户传递参数合法性,因为模型层把结果数据传输给Controller层之后,Controller层是不可以直接把从数据库查询出的数据传输给用户的,因为浏览器看不懂,要把数据传输给服务器端的视图层,是应用程序中处理用户交互的部分,通常控制器负责从视图读取数据,控制用户输入,并向模型发送数据;

2)模型(Model):是应用程序中用于处理应用程序数据逻辑的部分,通常模型对象负责在数据库中存储数据和拉取数据,调用数据库,是应用程序中用于处理应用程序数据逻辑的部分,通常模型对象负责在数据库中存取数据;

3)视图(View):是应用程序中用于处理数据显示的部分,当Controller把数据传送到视图层的时候,这里面是服务器端的视图会将数据和服务器端的模板,叫做显示层,主要是显示数据,把浏览器能够解析的html返回给客户端,是应用程序中处理数据显示的部分,通常视图是依据模型数据创建的,是前端(客户端)的可视的页面;

服务器端的模板:模板引擎和JSP;

4)Controller在收到模型层处理完的用户数据之后,会直接将数据发送给视图层,咱们的视图层会将数据和服务器端的模板(JSP,摸板引擎,HTML)结合,会将数据加上服务器端的模板,就比如说v-for,循环渲染之后,再把最终整合成的标签,比如说三个h1返回给客户端的视图,客户端的视图在进行拼接v-for分解之后和渲染然后展示在浏览器上面,这样就可以保证用户拿到看到的数据是经过服务器端执行和渲染之后的一个结果了

视图分为两种:一种是服务器端的视图(v-for),一种是客户端的视图(浏览器直接解析)

MVC执行流程:

1)用户直接发送一个HTTP请求首先到·服务器端的Controller

2)Controller继续把请求转发给Model

3)Model会处理业务并且把数据结果提交给Controller

4)Controller把数据交给View引擎

5)View最终会把数据进行转换渲染生成最终的页面给用户(客户端)

1)MVC是一种思想,Spring MVC是一个专门的技术实现手段框架和是对MVC思想的实现,是一种MVC模式,并继承了Servlet API的WEB框架,既然是WEB框架,那么用户在浏览器上面输入了url之后,我们的SpringMVC框架就可以进行感知用户的请求;

在application.properties写上这段代码:debug=true;

2)路由映射:就是当用户访问一个url的时候,将用户的某个请求对应到程序中的某个类的某个方法的过程中就叫做路由映射;

3)现在绝大多数的项目都是基于Spring项目的,而Spring的核心就是SpringMVC,也就是说SpringMVC框架是Spring框架的核心模块,而SpringBoot又是Spring的脚手架,因此现在市面上面的JAVA项目就等同于SpringMVC项目;

4)SpringMVC项目可以是基于SpringBoot创建,也就是创建一个SpringBoot项目,勾选上SpringWEB项目,这时就从一个普通的SpringBoot项目变成了WEB项目,就是类似于maven项目变成了Servlet项目;

5)SpringMVC项目创建:基于SpringBoot框架加上一个SpringWEB也就是SpringMVC依赖,此时项目就变成了SpringMVC项目;

SpringMVC项目的优势: 现在绝大部分的JAVA项目基本上都是基于Spring或者是SpringBoot的,而Spring项目的核心就是SpringMVC,也就是说Spring MVC是Spring框架的核心模块,而SpringBoot是Spring的脚手架,但是学习Spring MVC,主要也是学习三个功能:

1)连接:将用户和JAVA程序连接起来,就是访问一个URL地址可以调用Spring程序

2)获取参数:用户访问的时候会携带一些参数数据,要能够在程序中获取到这个参数部分

3)返回数据:执行了业务逻辑以后,要把程序执行的结果返回给前端

二)实现程序到用户之间的映射: 

设置路由可以实现程序与用户之间的映射(路由映射),也就是当用户访问某一个 URL 时,会将用户的请求对应到程序中的某个类的某个方法中,实现浏览器连接程序的作用,设置路由可以使用@RequestMapping注解实现,注意要实现路由访问,必须使用 @Controller 将对象储存到 Spring 中。
 

输入一个127.0.0.1:8080/Java200就可以访问这个页面

我们在Main目录下的resorces的static里面创建了一个html文件下面我们来改一下后端代码:

在浏览器上面输入:http://localhost:8080/java200这个地址就可以访问到Text.html这个页面了

1)@RequestMappering的作用就是用来注册一个URL路由,通过这个路由,就可以进行实现前端的请求和后端程序的交互了但是@RequestMappering这个注解实际可以修饰方法也是可以修饰类的@RequestMapping和@ResponseBody注解既是可以修饰类也是可以修饰方法的

2)我们在默认情况下使用@RequestMappering会返回一个html静态界面,但是加上

@RsponseBody就会返回一个非页面的数据了,而不是返回的是一个页面地址

3)当我们使用@RequestBody来进行修饰一个类的时候,这就表示当前类中的所有方法会返回一个非页面的数据

4)同样我们还可以进行指定发送请求的类型:

@RequestMappping的参数扩展只是支持某种类型的请求方式比如说POST类型的请求方式

@RequestMapping(value="/Java200",method= RequestMethod.GET)
如果说我们在Postman上面发送了一个POST请求,就会出现405的报错信息
@Controller
public class UserController {
    @RequestMapping(value="/Java",method=RequestMethod.GET)
    @ResponseBody
    public String start(String name)
    {
     return name+"要你管";
    }
}

5)@RequestMappering在不进行指定任何参数的情况下,只进行设置路由地址,那么此方法是即支持GET请求又是支持POST请求的,但是当我们给@RequestMapping设置了Method属性之后,那么它只能支持设置的请求类型,其他类型时会报错的(405)

6)我们可以通过更详细的注解比如说@GetMapping(指定路由地址)和@PostMapping(指定路由地址)来进行决定访问那一个方法;

1)其中,访问方法所返回数据或页面的 URL 为http://IP:端口号/类上的路由地址/方法上加的路由地址,这样就能访问到每个方法所返回的页面或者数据了,也可以只在方法上设置路由地址;且当类中只有一个方法时,类上/方法上也可以不设置路由地址(挑一个设置即可),此时直接使用一级路由就能直接访问了

2)由于历史原因,方法返回默认是一个静态页面,如果要设置返回一个页面数据,需要搭配使用 @ResponseBody注解,放在类上,类里面所有的方法都会默认放回的是页面数据,而不是一个静态页面,放到方法上生效的只是当前方法

3)@RequestMapping默认情况下支持GET请求和Post请求等请求方式,而且如果想要只是支持一种请求方式,就需要做配置,需要设置 @RequestMapping 的method属性为RequestMethod.xxx,RequestMethod 其实是一个枚举,里面储存了多种的请求方式

为什么RequestMapping默认情况下会返回一个静态页面呢?

这还是历史的原因,因为SpringMVC是和Spring一起创建起来的,Spring,2002年创建的,20年前开发是前后端部分里的时间段,JAVA程序员既要写后端代码也要写前端的一个页面,模板引擎用的非常多,返回的一定是静态页面,但是在今天,格局已经变了,我们是可以返回具体的数据信息的,@ResponseBody返回的是一个非静态页面的数据了;

1)@RequestMapping:支持任意一种请求类型(get/post),既能修饰方法又是可以修饰类的

2)@GetMapping:只是支持get方式的请求(不能修饰类)

3)@PostMapping:只是支持post方式的请求(不能修饰类)

@GetMapping和@PostMapping两种注解只能修饰方法,是不可以进行修饰类的

1)SpringBoot热加载进行配置,每一次修改Idea代码的时候不需要进行重启运行,每一次我进行修改源代码的时候,都会实时编译.class文件,会自动重启SpringBoot项目:

2)可能我们修改JAVA代码的时候,可能不能实时的生成target目录,热部署会自动的获取到修改代码事件,然后热部署会自动的重启程序,我们需要做的事情只是需要等一会,还要添加框架支持,新项目直接勾选,老项目直接generate;

3)IDEA热部署也就是热加载:会自动的帮助开发者重启IEAD,重启SpringBoot项目,为了达到修改代码之后能够实时的看到新效果的目的;

1)在SpringBoot项目的POM.XML里面引入Spring dev-Tools框架,引入框架,检测当前项目有没有修改原代码

2)在Setting里面开启项目自动编译,众所周知咱们的Idea是由两个Settings,一个就是叫做Settings,一个就是叫做new ProjectSettings,一个是针对当前项目,一个是针对以后new出来的新项目

打开settings,点击Bulid里面的Compiler进行选中

3)点击选择按钮:Build project automatically,选中项目自动编译,开启项目自动编译

开启程序运行的热部署:

4)点击全局搜索,按两下shift就可以了,然后在搜索框里面进行搜索reg,得到选项Registry,然后进行双击下面这个东西,就可以了

compiler.automake.allow.when.app.running

有一个value属性进行选中

5)点击程序运行debug按钮就可以开启热部署,注意一定不要点击Restore Defaults,这是默认恢复原来的默认选项,我们在这里直接就进行点击close就可以了

三)获取用户传递过来的参数

1)传递单个参数,在SpingMVC里面可以直接通过方法中的参数来实现传参

1)我们通过前端传递的querystring参数Key必须要和后端的方法里面的参数保持一致,否则是不能正常地进行获取到参数的;

2)要尽量使用包装类来进行接收前端传递过来的参数,包装类因为是可以接受null值的,如果使用普通数据类型(int),前端没有传递这个id参数,那么后端接收id不到参数,就说明程序会出错,程序会报一个500的错误异常,但是使用包装类就可以解决这个问题Integer直接接受为空,这样可以更好地定位问题;

3)上面地址的URL是可以进行调换位置的,表单传参和直接输入url是一样的;

4)当有多个参数的时候,前后端进行参数匹配的时候,是以参数的名称进行匹配的,因此参数的位置是不会影响到后端获取参数的结果;

5)但是如果说我们前端传递过来的参数实在是太多了,难道我要在方法里面写成百上千个参数吗,这显然是很不现实的,工作量极大,效率极低修改成本也是特别高的,我们要使用对象的方式来进行接收,美观性和可维护性都是特别差的,但是我们使用一个类来接收,这是如果URL参数变化了,我们只需要修改类中的属性就可以了,SpringBoot不光会自动给你设置返回的字符编码,接收参数会根据参数来匹配,如果你接受的是一个对象,那么会将URl中的queryString中的值自动填充到对象里面,就是按照属性名和前端传递名字相等的进行赋值;

6)下面的路由要变成小写;

7)SpringBoot或者是SpringMVC项目会自动根据你返回的数据来决定你返回的类型是什么以及字符编码格式,ContentType和ContentEncoding是由的框架设置的;

2)根据对象来进行接收

1)我们在浏览器上面输入一个这样的地址:
localhost:8080/Java100/GetUser?username=%E6%9D%8E%E4%BD%B3%E4%BC%9F&password=12503487&ClassID=1&UserID=9
此时返回的就是一个JSON的数据(会自动设置ContentType)
会自动的给对象的属性填充数据,会自动的检查返回的数据

2)设置返回的对象的类一定要加上Getter和Setter方法
 

1.我们先进行创建一个User类,来进行接收前端的参数
注意这里面的User类一定要写上Getter和Setter方法:
@Setter
@Getter
public class User {
    public String username;
    public String password;
    public int ClassID;
    public int UserID;
}
2.我们再写出UserController里面的代码:
@Controller
@RequestMapping("/Java100")
@ResponseBody
public class UserController {
    @RequestMapping("/GetUser")
    public User start(User user)
    {
        return user;
    }
}

 注意:接受方法的参数一定要和前端传递的参数名字保持一致否则就获取不到参数的值了

三)后端参数重命名:

1)在某一些情况下,前端传递过来的参数Key和后端接收过来的Key是不一致的,比如说前端传递了一个time给后端,而后端有时使用createtime字段来进行接收的,这样就会出现参数接收不到的情况,如果出现了这种情况,我们就可以使用@RequestParam来进行重命名前后端的参数值;

2)但是如果说进行全部替换的方式有很麻烦因为这里面的后端参数createtime的参数太多了

@Controller
@ResponseBody
public class UserController {
  @RequestMapping("/Java100")
    public Object start(@RequestParam("name") String username)//接受前端的对象
    {
         return "name"+username;
    }
我们可以直接进行访问:http://localhost:8080/Java100?name=128.1

1)出现这种情况是很少的:@RequestParam("time"),这个注解的作用就是把前端为time的key赋值给后端的createtime变量;

2)除此之外,它既可以进行重命名请求,又能保证此参数是必传参数,如果说你使用了这个注解之后@RequestParam注解之后,我们在这里填的参数必须是必填的,required属性默认是true,如果前端没有传这个参数,程序就会出现报错;

@Controller
@ResponseBody
public class UserController {
  @RequestMapping("/Java100")
    public Object start(@RequestParam(value="name",required = false) String username)//接受前端的对象
    {
         return "name"+username;
    }
}

3)@RequestParam中的参数required=false,这就表示当前的参数非必传参数,如果不进行设置此属性,那么它的默认值是true,就表示此参数是必传项,属性在对象里面就不可以实现重命名了,后端在数据库里面(MYBATIS)-----也可以实现重命名

@RequestParam的注意事项:

如果说我们在参数中添加@RequestParam注解,那么前端一定要传递此参数,否则就会报错,如果说想要解决这个问题,那么就需要在@RequestParam里面添加required=false

3)获取日期参数:

可以使用@DataTimeFormat注解来完成日期参数格式转换,这个主界面有一个pattern的属性,前端传递的值就是为我们所传递的日期参数的格式

 

现在重新来实现一下博客页面的登录和注册功能

尝试把之前写的博客的登陆界面来做一个修改:

1)希望前端给后端发送一个ajax的请求,参数里面都是Json格式的数据

  <div class="row">
            <span>用户名</span><input type="text" name="username" class="username">
        </div>
        <div class="row">
            <span>密码</span><input type="password" name="password" class="password">
        </div>
        <div class="row">
            <input type="button" value="提交" id="submit">
        </div>
  <script
              src="https://code.jquery.com/jquery-3.6.0.js"
              integrity="sha256-H+K7U5CnXl1h5ywQfKtSj8PCmoN9aaq30gDh27Xc0jk="
              crossorigin="anonymous"></script>
        <script>
            let button=document.querySelector("#submit");
            button.onclick=function(){
              let username=document.querySelector(".username").value;
               let password=document.querySelector(".password").value;
               console.log(username);
               console.log(password);
            $.ajax({
                type:"POST",
                url:"login",
                data:JSON.stringify({username:username,password:password}),
                success:function(data,status)
                {
                    if(data!=null&&data.succ==200)
                    {
                        if(data.state==1)
                        {
                            alert("登陆成功");
                          location.href="blog1.html";
                        }else{
                            alert(data.message);
                        }
                    }else{
                        alert("我们此时访问后端接口失败");
                    }
                },
                error:function(data,status)
                {
                    alert("咱们的后端接口访问失败");
                }
            });
        }
        </script>

当我们进行登录成功之后,可以进行跳转:

location.href="blog1.html";

后端来进行处理我们的登录请求:

一:我们先写UserController里面的代码:

package com.example.demo.Controller;
import com.example.demo.Service.UserService;
import com.example.demo.User;
import netscape.security.UserTarget;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.annotation.Autowired;
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.RequestParam;
import org.springframework.web.bind.annotation.ResponseBody;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import java.util.HashMap;
import java.util.List;
@Controller
public class UserController {
    @Autowired
    UserService userService;
    @RequestMapping("/login")
    @ResponseBody
    public HashMap<String,Object> StartLogin(@RequestBody User user)
    {
        System.out.println(user);
        HashMap<String,Object> map=new HashMap<>();
           map.put("succ",200);
           String message="";
           int state=1;
           if(user.getUsername()==null||user.getPassword()==null||user.getUsername().equals("")||user.getPassword().equals(""))
           {
               message="您当前输入的用户名或者密码是空";
               state=-1;
           }else {
               User userDemo = userService.login(user.getUsername(), user.getPassword());
               if (userDemo == null || userDemo.equals("")) {
                   state = -1;
                   message = "您当前输入的用户名或者密码错误";
               } else {
                   state = 1;
               }
           }
           map.put("state",state);
           map.put(message,"message");
           return map;
    }
}

二:我们在写一下UserService和接口里面的代码:

package com.example.demo.Service;
import com.example.demo.Blog;
import com.example.demo.Mapper.UserMapper;
import com.example.demo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import java.util.List;
@Service
public class UserService {
        @Autowired
        private UserMapper mapper;

    public User login(String username, String password) {
        return mapper.login(username,password);
    }
}
接口里面的代码:
 User login(String username, String password);//这里面最好写成User user

三:咱们再写一下XML文件的代码:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD com.example.demo.Mapper1 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.example.demo.Mapper.UserMapper">
    <select id="login" resultType="com.example.demo.User">
        select * from user where username=#{username} and password=#{password}
    </select>
</mapper>

1)实现注册功能和上面一致,我们直接通过value字段得到确认密码和密码的值,下从而判断他们是否相等,主要是在前端页面进行校验,前端基本差不多;我们加上这一段代码:

 if(password!=password1)
               {
               alert("密码和确认密码不一致");
               return;
               }

2)现在来实现下面的注册的后端代码:希望如果登录成功在前端代码进行处理,直接跳转到登陆界面:

一:UserController里面的代码:

 @RequestMapping("/reg")
    @ResponseBody
    public HashMap<String,Object> InsertAll(@RequestBody User user)
    {
        HashMap<String,Object> map=new HashMap<>();
        int state=-1;
        String message="";
        if(user.getUsername()==null||user.getPassword()==null||user.getPassword().equals("")||user.equals(""))
        {
              state=-1;
              message="您当前输入的用户名或者密码为空";
        }else{
            int len=userService.InsertAll(user.getUsername(),user.getPassword());
            if(len==1)
            {
                state=1;
                message="注册成功";
            }else{
                message="当前我们插入失败,请检查后端业务代码";
                state=-1;
            }
        }
        map.put("succ",200);
        map.put("state",state);
        map.put("message",message);
       return map;
    }

二:我们使用的UserService里面的代码和接口里面的代码和XML文件里面的代码:

 一: public int InsertAll(String username, String password) {
        return mapper.InsertAll(username,password);
    }
 二: int InsertAll(String username, String password);
 三:  <insert id="InsertAll">
        insert into user(username,password) values(#{username},#{password})
    </insert>

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值