当前位置: 首页 > news >正文

Spring MVC 深度剖析:优势与劣势全面解读

文章目录

      • Spring MVC 优势
        • 1. **松耦合**
        • 2. **易于测试**
        • 3. **灵活性**
        • 4. **强大的配置机制**
        • 5. **异常处理**
        • 6. **国际化支持**
        • 7. **数据验证**
        • 8. **安全性**
        • 9. **性能优化**
      • Spring MVC 劣势
        • 1. **学习曲线**
        • 2. **配置复杂性**
        • 3. **性能开销**
        • 4. **视图技术限制**
        • 5. **社区和支持**

Spring MVC 优势

1. 松耦合
  • 依赖注入(DI)

    • 自动装配:Spring 容器通过 @Autowired 注解自动装配依赖,减少了手动创建对象的代码量,提高了代码的可维护性和可测试性。
    • 接口编程:鼓励使用接口编程,使得组件之间的依赖关系更加灵活和可替换,增强了模块化设计。
    • 生命周期管理:Spring 容器管理 bean 的生命周期,包括初始化、销毁等,确保了资源的有效管理和释放。
  • 示例代码

    @Service
    public class UserService {@Autowiredprivate UserRepository userRepository;public User getUserById(Long id) {return userRepository.findById(id).orElse(null);}
    }@Controller
    public class UserController {@Autowiredprivate UserService userService;@GetMapping("/users/{id}")public String getUser(@PathVariable Long id, Model model) {User user = userService.getUserById(id);model.addAttribute("user", user);return "user";}
    }
    
2. 易于测试
  • 单元测试

    • MockMvc:Spring 提供了 MockMvc 类,用于模拟 HTTP 请求和响应,方便进行控制器的单元测试。
    • Spring TestContext Framework:提供了丰富的测试支持,包括事务管理、数据库初始化等,使得测试更加高效和全面。
  • 示例代码

    @RunWith(SpringRunner.class)
    @WebMvcTest(UserController.class)
    public class UserControllerTest {@Autowiredprivate MockMvc mockMvc;@MockBeanprivate UserService userService;@Testpublic void testGetUser() throws Exception {User user = new User(1L, "John Doe", "john@example.com");when(userService.getUserById(1L)).thenReturn(user);mockMvc.perform(get("/users/1")).andExpect(status().isOk()).andExpect(view().name("user")).andExpect(model().attribute("user", user));}
    }
    
3. 灵活性
  • 视图技术多样

    • JSP:传统的视图技术,适合简单的 Web 应用。
    • Thymeleaf:现代的模板引擎,支持静态原型和动态内容的混合,提供了丰富的表达式语言。
    • FreeMarker:功能强大的模板引擎,适合复杂的页面生成,提供了灵活的模板语法。
  • 配置方式多样

    • 注解配置:使用注解(如 @Controller@RequestMapping)进行配置,简洁易懂,减少了 XML 配置文件的冗余。
    • XML 配置:对于复杂的配置需求,可以使用 XML 配置文件进行详细设置,提供了更多的灵活性。
4. 强大的配置机制
  • 注解驱动

    • @Controller:标记控制器类,使其被 Spring 容器管理。
    • @RequestMapping:指定请求映射规则,支持路径、方法、参数等多种匹配方式。
    • @GetMapping@PostMapping:简化常见的 HTTP 方法映射,提高了代码的可读性。
  • XML 配置

    • 扫描组件:使用 <context:component-scan> 扫描指定包下的组件,自动注册到 Spring 容器。
    • 启用 MVC 注解驱动:使用 <mvc:annotation-driven> 启用 MVC 注解支持,简化配置。
    • 视图解析器:配置视图解析器,如 InternalResourceViewResolver,将逻辑视图名称解析为具体的视图对象。
  • 示例代码

    @Configuration
    @ComponentScan(basePackages = "com.example")
    @EnableWebMvc
    public class WebConfig implements WebMvcConfigurer {@Beanpublic InternalResourceViewResolver viewResolver() {InternalResourceViewResolver resolver = new InternalResourceViewResolver();resolver.setPrefix("/WEB-INF/views/");resolver.setSuffix(".jsp");return resolver;}
    }
    
5. 异常处理
  • 全局异常处理

    • @ControllerAdvice:定义全局异常处理器,集中处理多个控制器的异常。
    • @ExceptionHandler:处理特定类型的异常,提供统一的错误响应。
  • 示例代码

    @ControllerAdvice
    public class GlobalExceptionHandler {@ExceptionHandler(ResourceNotFoundException.class)@ResponseStatus(HttpStatus.NOT_FOUND)public ModelAndView handleResourceNotFoundException(ResourceNotFoundException ex) {ModelAndView modelAndView = new ModelAndView("error");modelAndView.addObject("errorMessage", ex.getMessage());return modelAndView;}
    }
    
6. 国际化支持
  • 多语言支持

    • MessageSource:用于加载多语言资源文件,支持动态获取国际化消息。
    • LocaleResolver:解析用户的语言偏好,支持多种语言环境的切换。
    • LocaleChangeInterceptor:拦截请求,动态切换语言,提供个性化的用户体验。
  • 示例代码

    @Bean
    public MessageSource messageSource() {ResourceBundleMessageSource messageSource = new ResourceBundleMessageSource();messageSource.setBasename("messages");messageSource.setDefaultEncoding("UTF-8");return messageSource;
    }@Bean
    public LocaleResolver localeResolver() {CookieLocaleResolver resolver = new CookieLocaleResolver();resolver.setDefaultLocale(Locale.US);resolver.setCookieName("locale");resolver.setCookieMaxAge(3600);return resolver;
    }@Bean
    public LocaleChangeInterceptor localeChangeInterceptor() {LocaleChangeInterceptor interceptor = new LocaleChangeInterceptor();interceptor.setParamName("lang");return interceptor;
    }@Override
    public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(localeChangeInterceptor());
    }
    
7. 数据验证
  • JSR-303 标准

    • Hibernate Validator:常用的验证框架,支持多种验证注解,如 @NotNull@Size@Email 等。
    • @Valid@Validated:用于触发验证,确保输入数据的有效性。
    • BindingResult:捕获验证结果,提供详细的错误信息。
  • 示例代码

    @PostMapping("/users")
    public String createUser(@Valid @ModelAttribute User user, BindingResult result, Model model) {if (result.hasErrors()) {return "userForm";}userService.saveUser(user);model.addAttribute("message", "User created successfully");return "success";
    }
    
8. 安全性
  • Spring Security 集成

    • 认证:用户身份验证,支持多种认证机制,如表单登录、HTTP 基本认证等。
    • 授权:访问控制,支持基于角色的访问控制(RBAC)、方法级别的安全控制等。
    • CSRF 保护:防止跨站请求伪造攻击,保护用户免受恶意攻击。
  • 示例代码

    @Configuration
    @EnableWebSecurity
    public class SecurityConfig extends WebSecurityConfigurerAdapter {@Overrideprotected void configure(HttpSecurity http) throws Exception {http.authorizeRequests().antMatchers("/", "/home").permitAll().anyRequest().authenticated().and().formLogin().loginPage("/login").permitAll().and().logout().permitAll();}@Overrideprotected void configure(AuthenticationManagerBuilder auth) throws Exception {auth.inMemoryAuthentication().withUser("user").password("{noop}password").roles("USER").and().withUser("admin").password("{noop}admin").roles("ADMIN");}
    }
    
9. 性能优化
  • 缓存支持

    • Spring Cache:支持多种缓存提供者,如 EhCache、Redis,提供了声明式的缓存管理。
    • @Cacheable@CachePut@CacheEvict:用于控制缓存操作,提高数据访问速度。
  • 异步处理

    • @Async:标记异步方法,支持后台任务的执行。
    • CompletableFuture:用于异步编程,提供了丰富的异步操作方法。
  • 示例代码

    @Configuration
    @EnableCaching
    public class CacheConfig {@Beanpublic CacheManager cacheManager() {return new ConcurrentMapCacheManager("users");}
    }@Service
    public class UserService {@Cacheable("users")public User getUserById(Long id) {// 模拟耗时操作Thread.sleep(1000);return userRepository.findById(id).orElse(null);}
    }@Service
    public class AsyncService {@Asyncpublic CompletableFuture<User> getUserAsync(Long id) {return CompletableFuture.supplyAsync(() -> userService.getUserById(id));}
    }
    

Spring MVC 劣势

1. 学习曲线
  • 初学者友好度
    • 概念复杂:Spring MVC 涉及的概念和技术较多,初学者需要时间来理解和掌握,特别是依赖注入、AOP、事务管理等高级特性。
    • 文档复杂:官方文档内容丰富但较为复杂,新手可能需要较长时间来消化和理解,特别是对于没有 Java Web 开发经验的开发者。
2. 配置复杂性
  • 配置繁琐
    • 大量配置:对于大型项目,需要编写大量的配置代码,尤其是在使用 XML 配置时,配置文件可能变得庞大且难以维护。
    • 依赖管理:需要手动管理多个依赖库,增加了项目的复杂度,尤其是在没有使用构建工具(如 Maven 或 Gradle)的情况下。
3. 性能开销
  • 初始化开销

    • 启动时间:Spring MVC 在启动时会初始化大量的组件和服务,可能会导致应用启动时间较长,尤其是在配置复杂的项目中。
    • 内存占用:大量的依赖注入和代理机制可能会增加内存占用,影响应用的性能。
  • 运行时开销

    • 反射和代理:Spring 框架广泛使用反射和代理机制,虽然提供了强大的功能,但也可能引入一定的运行时开销,特别是在高并发场景下。
4. 视图技术限制
  • 视图技术选择
    • 技术多样性:虽然支持多种视图技术,但在某些情况下,特定的视图技术可能不够成熟或存在性能问题,例如某些模板引擎的渲染速度较慢。
    • 模板引擎兼容性:不同模板引擎之间可能存在兼容性问题,需要额外的工作来解决,尤其是在多团队协作的情况下。
5. 社区和支持
  • 社区活跃度
    • 问题解答:虽然 Spring 社区非常活跃,但某些特定问题可能需要较长时间才能得到解答,特别是在处理边缘情况或高级特性时。
    • 第三方库支持:一些第三方库可能没有及时更新以支持最新的 Spring 版本,导致兼容性问题,特别是在企业级应用中使用时。

相关文章:

Spring MVC 深度剖析:优势与劣势全面解读

文章目录 Spring MVC 优势1. **松耦合**2. **易于测试**3. **灵活性**4. **强大的配置机制**5. **异常处理**6. **国际化支持**7. **数据验证**8. **安全性**9. **性能优化** Spring MVC 劣势1. **学习曲线**2. **配置复杂性**3. **性能开销**4. **视图技术限制**5. **社区和支…...

力扣hot100-->前缀和/前缀书/LRU缓存

前缀和 1. 560. 和为 K 的子数组 中等 给你一个整数数组 nums 和一个整数 k &#xff0c;请你统计并返回 该数组中和为 k 的子数组的个数 。 子数组是数组中元素的连续非空序列。 示例 1&#xff1a; 输入&#xff1a;nums [1,1,1], k 2 输出&#xff1a;2示例 2&#…...

Three.js CSS2D/CSS3D渲染器

在Three.js开发过程中&#xff0c;有时需要将 HTML 元素与 Three.js 渲染的 3D 场景相结合&#xff0c;这就需要用到 CSS2DRenderer 和 CSS3DRenderer。本文将详细介绍这两种渲染器的原理及其应用 一、CSS2DRenderer 渲染器 概述 CSS2DRenderer 渲染器用于在 3D 场景中渲染纯…...

mongodb文档字符串批量替换

【mongodb文档字符串批量替换脚本语句】 前言&#xff1a; 1、本方式对于数据量大的情况不适用&#xff0c;执行可能比较慢&#xff1b; 2、数据量大的情况&#xff0c;个人推荐代码层面解决&#xff0c;多线程替换更快&#xff1a; &#xff08;1&#xff09;写实体类的方式…...

前端安全和解决方案

提到这个我可能想到的就是不要暴露太多的账号密码信息。一些页面的请求和操作要加上权限。 然后下面就详细的介绍前端可能遇到的安全问题以及解决方法。 首先比较常见的前端的安全性问题就是跨站脚本攻击&#xff08;XSS&#xff09;。跨站请求伪造&#xff08;csrf&#xff…...

Tlias智能辅助学习系统-部门管理

包括查询、新增、删除、修改功能 控制层 package com.itheima.controller;import com.itheima.pojo.Dept; import com.itheima.pojo.Result; import com.itheima.service.DeptService; import lombok.extern.slf4j.Slf4j; import lombok.extern.slf4j.XSlf4j; import org.spr…...

React第十节组件之间传值之context

1、Context 使用creatContext() 和 useContext() Hook 实现多层级传值 概述&#xff1a; 在我们想要每个层级都需要某一属性&#xff0c;或者祖孙之间需要传值时&#xff0c;我们可以使用 props 一层一层的向下传递&#xff0c;或者我们使用更便捷的方案&#xff0c;用 creatC…...

flink中barrier不对齐的原因和影响

Barrier 不对齐&#xff08;Barrier Misalignment&#xff09;可能导致一些性能和一致性相关的问题&#xff0c;但 Flink 提供了机制来确保即使在不对齐的情况下&#xff0c;也可以保证数据的一致性。 1. 什么是 Barrier 不对齐&#xff1f; Barrier 不对齐是指在分布式数据流…...

软银集团孙正义再度加码OpenAI,近屿智能专注AI人才培养

11月28日凌晨&#xff0c;全球最大财经CNBC报道&#xff0c;软银集团创始人兼CEO孙正义再次向人工智能领域的领军企业OpenAI投资了15亿美元。软银对OpenAI的投资已不是首次。就在上个月&#xff0c;软银已在OpenAI的上一轮融资中注入了5亿美元的资金。但他一直寻求获得OpenAI更…...

麒麟系统x86安装达梦数据库

一、安装准备前工作 操作系统&#xff1a;银河麒麟V10&#xff0c;CPU&#xff1a; x86_64 架构 下载地址&#xff0c;麒麟官网&#xff1a;https://www.kylinos.cn/ 数据库&#xff1a;dm8_20220915_x86_kylin10_64 下载地址&#xff0c;达梦数据库官网&#xff1a;https://…...

Java中的“多态“详解

多态&#xff08;Polymorphism&#xff09;是面向对象编程&#xff08;OOP&#xff09;中的一个核心概念&#xff0c;它允许同一个接口或方法在不同对象上具有不同的实现方式。多态性使得程序在运行时可以根据对象的实际类型来决定调用哪个方法&#xff0c;从而提高代码的灵活性…...

buuctf-[SUCTF 2019]EasySQL 1解题记录

把你的旗帜给我&#xff0c;我会告诉你这面旗帜是对的。 堆叠注入查询数据库 1; show databases; ​ 查询表名 1; show tables; 获取flag 1;set sql_modepipes_as_concat;select 1...

ASP.NET Core 入门

使用 .NET CLI 创建并运行 ASP.NET Core Web 应用。 文章目录 一、先决条件二、创建Web应用项目三、运行应用四、编辑Razor页面 一、先决条件 .NET 8.0 SDK 二、创建Web应用项目 打开命令行界面&#xff0c;然后输入以下命令&#xff1a; dotnet new webapp --output aspne…...

php反序列化1_常见php序列化的CTF考题

声明&#xff1a; 以下多内容来自暗月师傅我是通过他的教程来学习记录的&#xff0c;如有侵权联系删除。 一道反序列化的CTF题分享_ctf反序列化题目_Mr.95的博客-CSDN博客 一些其他大佬的wp参考&#xff1a;php_反序列化_1 | dayu’s blog (killdayu.com) 序列化一个对象将…...

题目 1013: [编程入门]Sn的公式求和

题目 1013: [编程入门]Sn的公式求和 [编程入门]Sn的公式求和 求Snaaaaaa…aa…aaa&#xff08;有n个a&#xff09;之值&#xff0c;其中a是一个数字&#xff0c;为2。 例如&#xff0c;n5时222222222222222&#xff0c;n由键盘输入。 #include<stdio.h> int A(int n)…...

算法——赎金信(leetcode383)

题目&#xff1a; 给你两个字符串&#xff1a;ransomNote 和 magazine &#xff0c;判断 ransomNote 能不能由 magazine 里面的字符构成。 如果可以&#xff0c;返回 true &#xff1b;否则返回 false 。 magazine 中的每个字符只能在 ransomNote 中使用一次。 示例 1&#…...

transformers训练(NLP)阅读理解(多项选择)

简介 在阅读理解任务中&#xff0c;有一种通过多项选择其中一个答案来训练机器的阅读理解。比如&#xff1a;给定一个或多个文档h,以及一个问题S和对应的多个答案候选&#xff0c;输出问题S的答案E&#xff0c;E是答案候选中的某一个选项。 这样的目的就是通过文档&#xff0c…...

微软企业邮箱:安全可靠的企业级邮件服务!

微软企业邮箱的设置步骤&#xff1f;如何注册使用烽火域名邮箱&#xff1f; 微软企业邮箱作为一款专为企业设计的邮件服务&#xff0c;不仅提供了高效便捷的通信工具&#xff0c;更在安全性、可靠性和功能性方面树立了行业标杆。烽火将深入探讨微软企业邮箱的多重优势。 微软…...

什么是分布式锁

定义 分布式锁是控制分布式系统或集群中不同节点对共享资源访问的一种机制。在分布式环境下&#xff0c;多个节点&#xff08;如多个服务器或多个进程&#xff09;可能会同时访问诸如数据库中的某条记录、一个共享文件或者一个全局计数器等共享资源。分布式锁的目的是确保在同一…...

【含开题报告+文档+PPT+源码】基于SpringBoot的艺术培训学校管理系统的设计与实现

开题报告 艺术培训学校管理在现代教育行业中发挥着至关重要的作用&#xff0c;旨在为学员提供及时、专业且高效的课程服务&#xff0c;同时也激励培训机构不断提升教学质量与管理水平。然而&#xff0c;传统的艺术培训学校管理模式常面临一系列挑战&#xff0c;如课程报名程序…...

应用升级/灾备测试时使用guarantee 闪回点迅速回退

1.场景 应用要升级,当升级失败时,数据库回退到升级前. 要测试系统,测试完成后,数据库要回退到测试前。 相对于RMAN恢复需要很长时间&#xff0c; 数据库闪回只需要几分钟。 2.技术实现 数据库设置 2个db_recovery参数 创建guarantee闪回点&#xff0c;不需要开启数据库闪回。…...

ardupilot 开发环境eclipse 中import 缺少C++

目录 文章目录 目录摘要1.修复过程摘要 本节主要解决ardupilot 开发环境eclipse 中import 缺少C++,无法导入ardupilot代码,会引起查看不方便的问题。如下图所示 1.修复过程 0.安装ubuntu 软件中自带的eclipse 1.打开eclipse—Help—install new software 2.在 Work with中…...

C# SqlSugar:依赖注入与仓储模式实践

C# SqlSugar&#xff1a;依赖注入与仓储模式实践 在 C# 的应用开发中&#xff0c;数据库操作是必不可少的环节。为了让数据访问层更加简洁、高效且易于维护&#xff0c;许多开发者会选择成熟的 ORM&#xff08;对象关系映射&#xff09;框架&#xff0c;SqlSugar 就是其中备受…...

【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)

骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术&#xff0c;它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton)&#xff1a;由层级结构的骨头组成&#xff0c;类似于人体骨骼蒙皮 (Mesh Skinning)&#xff1a;将模型网格顶点绑定到骨骼上&#xff0c;使骨骼移动…...

k8s业务程序联调工具-KtConnect

概述 原理 工具作用是建立了一个从本地到集群的单向VPN&#xff0c;根据VPN原理&#xff0c;打通两个内网必然需要借助一个公共中继节点&#xff0c;ktconnect工具巧妙的利用k8s原生的portforward能力&#xff0c;简化了建立连接的过程&#xff0c;apiserver间接起到了中继节…...

网络编程(UDP编程)

思维导图 UDP基础编程&#xff08;单播&#xff09; 1.流程图 服务器&#xff1a;短信的接收方 创建套接字 (socket)-----------------------------------------》有手机指定网络信息-----------------------------------------------》有号码绑定套接字 (bind)--------------…...

Linux离线(zip方式)安装docker

目录 基础信息操作系统信息docker信息 安装实例安装步骤示例 遇到的问题问题1&#xff1a;修改默认工作路径启动失败问题2 找不到对应组 基础信息 操作系统信息 OS版本&#xff1a;CentOS 7 64位 内核版本&#xff1a;3.10.0 相关命令&#xff1a; uname -rcat /etc/os-rele…...

【Go语言基础【12】】指针:声明、取地址、解引用

文章目录 零、概述&#xff1a;指针 vs. 引用&#xff08;类比其他语言&#xff09;一、指针基础概念二、指针声明与初始化三、指针操作符1. &&#xff1a;取地址&#xff08;拿到内存地址&#xff09;2. *&#xff1a;解引用&#xff08;拿到值&#xff09; 四、空指针&am…...

uniapp手机号一键登录保姆级教程(包含前端和后端)

目录 前置条件创建uniapp项目并关联uniClound云空间开启一键登录模块并开通一键登录服务编写云函数并上传部署获取手机号流程(第一种) 前端直接调用云函数获取手机号&#xff08;第三种&#xff09;后台调用云函数获取手机号 错误码常见问题 前置条件 手机安装有sim卡手机开启…...

Android写一个捕获全局异常的工具类

项目开发和实际运行过程中难免会遇到异常发生&#xff0c;系统提供了一个可以捕获全局异常的工具Uncaughtexceptionhandler&#xff0c;它是Thread的子类&#xff08;就是package java.lang;里线程的Thread&#xff09;。本文将利用它将设备信息、报错信息以及错误的发生时间都…...