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 版本,导致兼容性问题,特别是在企业级应用中使用时。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/62445.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Jmeter测试工具的安装和使用,mac版本,jmeter版本5.2.1

Jmeter测试工具的安装和使用JSON格式请求 一、安装1、安装jdk包和设置java环境2、去官网下载Jmeter3、解压后&#xff0c;打开mac终端&#xff0c;进入apache-jmeter的bin文件开启jmeter 二、使用jmeter1、添加线程2、添加HTTP请求3、配置请求的协议、IP地址、端口号、请求方法…

Cookie跨域

跨域&#xff1a;跨域名&#xff08;IP&#xff09; 跨域的目的是共享Cookie。 session操作http协议&#xff0c;每次既要request&#xff0c;也要response&#xff0c;cookie在创建的时候会产生一个字符串然后随着response返回。 全网站的各个页面都会带着登陆的时候的cookie …

【设计模式】【行为型模式(Behavioral Patterns)】之策略模式(Strategy Pattern)

1. 设计模式原理说明 策略模式&#xff08;Strategy Pattern&#xff09; 是一种行为设计模式&#xff0c;它允许你定义一系列算法&#xff0c;并将每个算法封装起来&#xff0c;使它们可以互换。策略模式让算法的变化独立于使用算法的客户。通过这种方式&#xff0c;客户端可…

工程企业如何做好成本控制?该如何入手?

工程企业的成本控制是企业管理中的核心工作&#xff0c;其直接关系到项目的盈利能力和市场竞争力。以下从几个关键方向阐述如何入手做好成本控制&#xff1a; 一、明确成本控制目标 成本控制的目标不仅是减少支出&#xff0c;更重要的是保证项目质量和工期&#xff0c;避免因低…

多项式加法运算的链表实现

多项式加法运算的链表实现 主要思路&#xff1a;相同指数的项系数相加&#xff0c;其余部分进行拷贝。 两个多项式分别使用单链表实现&#xff0c;链表的每一个节点的结构为&#xff1a;系数、指数、下一个节点的地址。 链表节点按照指数递减顺序排列。 一句话&#xff1a;…

【N 卡 掉驱动 Driver 】NVML ERROR: Driver Not Loaded

问题描述 输入 nvitop 时报错 NVML ERROR: Driver Not Loaded&#xff0c;重启问题依旧存在。 问题解决-重新下载驱动 进入官网选择合适自己的驱动版本 https://www.nvidia.cn/geforce/drivers/ 根据个人情况搜索后&#xff0c;选择最新的 Driver 进行下载&#xff0c;如果希…

uniapp H5支付宝支付

1、scheme支付 uniapp scheme打开支付宝H5调起支付_支付宝scheme链接-CSDN博客 2、链接地址支付 location.href res.resultData.pay_info; 3、form表单支付 const divForm document.getElementById(divForm); if (divForm) {document.body.removeChild(divForm); } cons…

杭州网世一站式网络解决方案,助力安邦护卫网络升级改造

随着信息技术的不断进步&#xff0c;浙江台州安邦护卫有限公司现有的网络设备已无法满足其日益增长的业务需求。网络性能瓶颈、安全隐患和管理复杂性等问题逐渐凸显&#xff0c;严重影响了企业的运营效率和服务质量。为了解决这些问题&#xff0c;浙江台州安邦护卫有限公司决定…

IIC和SPI的时序图

SCL的变化快慢决定了通信速率&#xff0c;当SCL为低电平的时候&#xff0c;无论SDA是1还是0都不识别&#xff1a; ACK应答&#xff1a;当从设备为低电平的时候识别为从设备有应答&#xff1a; 谁接收&#xff0c;谁应答&#xff1a; 起始位和停止位&#xff1a; IIC的时序图&am…

C底层 函数栈帧

文章目录 一&#xff0c;什么是寄存器 二&#xff0c;栈和帧 前言 我们在学习c语言程序的时候&#xff0c;是不是有很多的疑问&#xff0c;如 1&#xff0c;为什么形参不可以改变实参 2&#xff0c;为什么我们编写程序的时候会出现烫烫烫......这个乱码 3&#xff0c;那些局…

全桥LLC变换器原理及MATLAB仿真模型

“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 主电路拓扑 全桥LLC 谐振变换器主电路拓扑结构图。图中S1 &#xff5e; S4为功率开关管&#xff0c; D1 &#xff5e; D4为功率开关管的体二极管&#xff0c; C1 &#xff5e; C4 为功率开关管的寄生电容。谐振电感r…

TavilySearchResults报错

报错 pydantic_core._pydantic_core.ValidationError: 1 validation error for TavilySearchAPIWrapper Value error, Did not find tavily_api_key, please add an environment variable TAVILY_API_KEY which contains it, or pass tavily_api_key as a named parameter. …

物料理解笔记·蓝白段子线·端子线座子焊接反了怎么处理!!!

目录 蓝白端子排线 端子线座子焊接错了怎么办 端子线如何拆线 编写不易&#xff0c;请勿搬运&#xff0c;仅供学习&#xff0c;感谢理解 蓝白端子排线 蓝白端子排线&#xff0c;这种端子线常用与编码电机的接线&#xff0c;或者在板子上通过提供段子线的接口&#xff0c;通…

shell(8)until循环以及函数基本创建

声明&#xff01; 学习视频来自B站up主 泷羽sec 有兴趣的师傅可以关注一下&#xff0c;如涉及侵权马上删除文章&#xff0c;笔记只是方便各位师傅的学习和探讨&#xff0c;文章所提到的网站以及内容&#xff0c;只做学习交流&#xff0c;其他均与本人以及泷羽sec团队无关&#…

51c自动驾驶~合集35

我自己的原文哦~ https://blog.51cto.com/whaosoft/12206500 #纯视觉方案的智驾在大雾天还能用吗&#xff1f; 碰上大雾天气&#xff0c;纯视觉方案是如何识别车辆和障碍物的呢&#xff1f; 如果真的是纯纯的&#xff0c;特头铁的那种纯视觉方案的话。 可以简单粗暴的理解为…

Python中的23种设计模式:详细分类与总结

设计模式是解决特定问题的通用方法&#xff0c;分为创建型模式、结构型模式和行为型模式三大类。以下是对每种模式的详细介绍&#xff0c;包括其核心思想、应用场景和优缺点。 一、创建型模式&#xff08;Creational Patterns&#xff09; 创建型模式关注对象的创建&#xff0…

【ArcGIS Pro实操第11期】经纬度数据转化成平面坐标数据

经纬度数据转化成平面坐标数据 数据准备ArcGIS操作步骤-投影转换为 Sinusoidal1 投影2 计算几何Python 示例 另&#xff1a;Sinusoidal (World) 和 Sinusoidal (Sphere) 的主要区别参考 数据准备 数据投影&#xff1a; 目标投影&#xff1a;与MODIS数据相同&#xff08;Sinu…

Python3 pip

pip 是 Python 包管理工具&#xff0c;该工具提供了对 Python 包的查找、下载、安装、卸载的功能。 软件包也可以在 https://pypi.org/ 中找到。 目前最新的 Python 版本已经预装了 pip。 注意&#xff1a;Python 2.7.9 或 Python 3.4 以上版本都自带 pip 工具。 如果没有安…

docker搭建socks5代理

准备工作 VPS安全组/策略放行相应端口如启用了防火墙&#xff0c;放行相应端口 实际操作 我们选用“历史悠久”的Dante socks5 代理服务器&#xff0c;轻量、稳定。Github也有对dante进行进一步精简的镜像&#xff0c;更为适宜。github项目地址如下&#xff1a; https://gi…

【Leecode】Leecode刷题之路第62天之不同路径

题目出处 62-不同路径-题目出处 题目描述 个人解法 思路&#xff1a; todo代码示例&#xff1a;&#xff08;Java&#xff09; todo复杂度分析 todo官方解法 62-不同路径-官方解法 方法1&#xff1a;动态规划 思路&#xff1a; 代码示例&#xff1a;&#xff08;Java&…