【SpringMVC】SpringMVC简介、过程分析、bean的加载和控制

1. SpringMVC简介

在这里插入图片描述
数据层框架: MyBatis
表现层框架: Servlet->SpringMVC (即SpringMVC属于一种表现层框架)
简介: SpringMVC是一种基于Java实现MVC模型的轻量级Web框架
优点:

  • 使用简单,开发便捷(相比于Servlet)
  • 灵活性强

2. SpringMVC入门案例

文件结构

在这里插入图片描述

第一步:坐标导入

pom.xml文件中添加一些依赖,这些依赖版本之间是相互联系的,如果配错或配太低都不行

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
  <modelVersion>4.0.0</modelVersion>

  <groupId>org.example</groupId>
  <artifactId>springmvc_01_test</artifactId>
  <packaging>war</packaging>
  <version>1.0-SNAPSHOT</version>

  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <maven.compiler.source>1.8</maven.compiler.source>
    <maven.compiler.target>1.8</maven.compiler.target>
  </properties>

  <dependencies>
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-webmvc</artifactId>
      <version>5.2.13.RELEASE</version>
    </dependency>
    <dependency>
      <groupId>javax.servlet</groupId>
      <artifactId>javax.servlet-api</artifactId>
      <version>3.1.0</version>
      <scope>provided</scope>
    </dependency>
    <dependency>
      <groupId>org.apache.tomcat</groupId>
      <artifactId>tomcat-api</artifactId>
      <version>10.1.7</version>
    </dependency>
  </dependencies>

  <build>
    <plugins>
      <plugin>
        <groupId>org.apache.tomcat.maven</groupId>
        <artifactId>tomcat7-maven-plugin</artifactId>
        <version>2.1</version>
        <configuration>
          <port>80</port>
          <path>/</path>
        </configuration>
      </plugin>
    </plugins>
  </build>

</project>

2023.12.20 update---------------------------------------------------------------------------
其实上面这种方法配置Tomcat感觉用起来很奇怪,因为我记得我以前是直接安装了Tomcat,在IDEA里重新配置的,这种方法到后面跳转页面时候确实出现了问题。在这里重新推荐下载安装Tomcat进行配置的方法:
第一步:安装Tomcat,可以不用看里面安装JDK的部分,毕竟都学到SpringMVC了,JDK肯定也安装了:Tomcat安装和配置,超详细(附JDK安装及配置环境变量)
第二步:在IDEA里面配置Tomcat运行环境:SpringMVC——配置tomcat
第三步:修改一下下面两个位置,使得我们通过/就可以访问,否则如果这个链接后面带了一堆乱七八糟的东西,我们访问localhost/会报404的错误:
在这里插入图片描述
注意,启动端口设置成了80,localhost等价于localhost:80
在这里插入图片描述

第二步:创建SpringMVC容器的控制器类

控制器类来指定对应的访问网址什么的,@Controller标明这是一个bean,@RequestMapping指定访问路径,@ResponseBody指定返回不做任何解析和处理

package com.demo.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@Controller
public class UserController {

    @RequestMapping("/save")
    @ResponseBody
    public String save(){
        System.out.println("User save...");
        return "{'module': 'springmvc'}";
    }
}

第三步:初始化SpringMVC环境,设定Spring加载对应的bean

类似于Spring的配置,通过@Configuration指定为配置类,通过@ComponentScan来指定扫描的控制器范围

package com.demo.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

@Configuration
@ComponentScan("com.demo.controller")
public class SpringMvcConfig {
}

第四步:初始化Servlet容器,加载SpringMVC环境,并设置SpringMVC技术请求的处理

这里相当于是一些固定的写法

package com.demo.config;

import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
import org.springframework.web.servlet.support.AbstractDispatcherServletInitializer;

public class ServletContainerInitConfig extends AbstractDispatcherServletInitializer {
//    加载SpringMVC容器配置
    @Override
    protected WebApplicationContext createServletApplicationContext() {
        AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
        ctx.register(SpringMvcConfig.class);
        return ctx;
    }
//    设置哪些请求归属SpringMVC处理
    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
//    加载Spring配置
    @Override
    protected WebApplicationContext createRootApplicationContext() {
        return null;
    }
}

测试

配置Run Configurations
在这里插入图片描述
然后启动,在浏览器里输入http://localhost/save可看到结果:
在这里插入图片描述

总结:使用到的一些注解及入门程序开发总结

@Controller:类注解,放在SpringMVC控制器上方,作用是设定SpringMVC的核心控制器bean
@RequestMapping:方法注解,放在SpringMVC控制器方法定义上方,作用是设置当前控制器方法的请求路径
@ResponseBody:方法注解, 放在SpringMVC控制器方法定义上方,作用是设置当前控制器方法相应内容为当前返回值,无需解析
在这里插入图片描述

3. 入门案例工作流程分析

启动服务器初始化过程

  1. 服务器启动,执行ServletContainerInitConfig类,初始化Web容器
  2. 执行createServletApplicationContext方法,创建了WebApplicationContext对象
  3. 加载SpringMVCConfig
  4. 执行@ComponentScan对应的bean
  5. 加载UserController(也就是加载扫描包下注释了@Controller的那些类),每个@RequestMapping的名称都对应一个具体的方法(注意:如图中/save->save()的映射实际上并不是放在每一个bean中管理的,而是由SpringMVC放在统一的地方进行管理的)
  6. 执行getServletMappings方法,定义所有的请求都通过SpringMVC
    在这里插入图片描述

单次请求过程

  1. 发送请求localhost/save
  2. web容器发现所有的请求都经过SpringMVC(上面的6,定义了所有的请求都经过SpringMVC),将请求交给SpringMVC处理
  3. 解析请求路径/save
  4. /save匹配执行对应的方法save()
  5. 执行save()
  6. 检测到有@ResponseBody,直接将save()方法的返回值作为响应请求体返回给请求方

4. bean的加载与控制

SpringMVC相关的bean是表现层的bean:

  • SpringMVC加载的bean对应的包均在com.xx.controller包内

Spring控制的bean:

  • 业务bean(Service)
  • 功能bean(DataSource等)
  • 方式一:Spring加载的bean设定扫描范围为com.xx,排除掉controller包内的bean
  • 方式二:Spring加载的bean设定扫描范围为精准范围,例如service, dao包等
  • 方式三:不区分Spring和SpringMVC的环境,加载到同一个环境中

方式一:

指定大的扫描类型,在其中通过excludeFilters排除掉一些东西,具体写法如下,我们需要再excludeFilters里边指定清楚过滤的规则

package com.demo.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.FilterType;
import org.springframework.stereotype.Controller;

@Configuration
@ComponentScan(value = "com.demo",
        excludeFilters = @ComponentScan.Filter(
            type = FilterType.ANNOTATION,
            classes = Controller.class
        )
)
public class SpringConfig {
}

执行一下测试,看能否再拿到注释了@ControllerUserController类(注意,在测试之前要注释掉SpringMvcConfig上的@Configuration):

package com.demo;

import com.demo.config.SpringConfig;
import com.demo.controller.UserController;
import com.demo.domain.User;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class App {
    public static void main(String[] args) {
        ApplicationContext ctx = new AnnotationConfigApplicationContext(SpringConfig.class);
        UserController userController = ctx.getBean(UserController.class);
        System.out.println(userController);
    }
}

执行测试代码会报错:
在这里插入图片描述
如果取消掉这个过滤规则,再执行测试代码会发现不会报错,而是会打印拿到的实例信息。通过测试说明使用excludeFilters这种方法是可以过滤包的
解释一下为什么要注释掉SpringMvcConfig上的@Configuration:因为我们在SpringConfig中扫描了com.demo下的所有包,在SpringMvcConfig上我们使用@Configuration注解了它是一个配置类,那么这个配置就会被扫描,它所包括的所有bean也会被加载进来,如果将SpringMvcConfig放到com.demo下而不注释也可以解决这个问题(老师在视频里是这么讲的,但是我自己试了一下好像把SpringMvcConfig放到com.demo下,不注释也会被加载进来,不知道是不是我的问题)

方式二:

我们使用mybatis技术来进行开发,这里的com.demo.dao是可以不写的,因为mybatis通过自动代理的技术可以扫到com.demo.dao下的bean(在Mybatis的MapperScannerConfigurer中有设置msc.setPackage("...")
但是假如我们使用JDBC的技术来进行开发,我们就必须扫描com.demo.dao下的包,否则我们就没有数据层的bean,Spring下JDBC的开发可以看一下这篇文章:Spring jdbc数据库管理,所以对于com.demo.dao的扫描还是建议写,这样通用性比较好,如果不写的话通用性就会比较差

package com.demo.config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

@Configuration
@ComponentScan({"com.demo.service", "com.demo.dao"})
public class SpringConfig {
}

将Spring下的bean也加载到Servlet容器中

修改ServletContainersInitConfig类中的内容,加上加载Spring配置的内容:

package com.demo.config;

import com.demo.controller.SpringConfig;
import com.demo.controller.SpringMvcConfig;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
import org.springframework.web.servlet.support.AbstractDispatcherServletInitializer;

public class ServletContainerInitConfig extends AbstractDispatcherServletInitializer {
//    加载SpringMVC容器配置
    @Override
    protected WebApplicationContext createServletApplicationContext() {
        AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
        ctx.register(SpringMvcConfig.class);
        return ctx;
    }
//    设置哪些请求归属SpringMVC处理
    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
//    加载Spring配置
    @Override
    protected WebApplicationContext createRootApplicationContext() {
        AnnotationConfigWebApplicationContext ctx = new AnnotationConfigWebApplicationContext();
        ctx.register(SpringConfig.class);
        return ctx;
    }
}

在Servlet容器中加载SpringMVC和Spring配置还有一个简化的类及写法,以下是以上的简化开发:

package com.demo.config;

import com.demo.controller.SpringConfig;
import com.demo.controller.SpringMvcConfig;
import org.springframework.web.context.WebApplicationContext;
import org.springframework.web.context.support.AnnotationConfigWebApplicationContext;
import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer;
import org.springframework.web.servlet.support.AbstractDispatcherServletInitializer;

public class ServletContainerInitConfig extends AbstractAnnotationConfigDispatcherServletInitializer {
    @Override
    protected Class<?>[] getRootConfigClasses() {
        return new Class[]{SpringConfig.class};
    }

    @Override
    protected Class<?>[] getServletConfigClasses() {
        return new Class[]{SpringMvcConfig.class};
    }

    @Override
    protected String[] getServletMappings() {
        return new String[]{"/"};
    }
}

小结

在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值