Spring使用(一)注解

Spring使用

资源

Spring 框架内部使用 Resource 接口作为所有资源的抽象和访问接口,在上一篇文章的示例代码中的配置文件是通过ClassPathResource 进行封装的,ClassPathResource 是 Resource 的一个特定类型的实现,代表的是位于 classpath 中的资源。

对不同来源的资源文件 Spring 都提供了相应的实现:文件(FileSystemResource )、ClassPath资源(ClassPathResource)、URL资源(UrlResource)、InputStream资源(InputStreamResource)、ByteArray资源(ByteArrayResource)等。

注解

spring mvc注解.png

宏观:

①Service

@Service 注解是用来声明服务层(Service Layer)组件,通常包含业务逻辑,作为数据访问层(Data Access Layer)和控制器(Controller Layer)之间的桥梁。

  • 它是一个立即加载的注解,这意味着当 Spring 应用程序启动时,Spring 容器就会创建标注了 @Service 的类的实例。
  • @Service 是一个特殊类型的 @Component 注解。它允许自动检测通过类路径扫描,为其创建 Bean 定义并注册到 Spring 容器中。
  • 它有助于区分 Spring 组件的表示层(Controller)、服务层(Service)和数据访问层(Repository)。

②Controller

用于标记一个类作为 Spring MVC 控制器组件。控制器负责处理由 DispatcherServlet 分发的来自浏览器或其他客户端的 HTTP 请求。这个注解会将类识别为一个 Bean,并注册到 Spring 应用上下文中。

  • 通常与 @RequestMapping 或其他基于 HTTP 方法的注解(@GetMapping, @PostMapping, @PutMapping, @DeleteMapping, @PatchMapping)配合使用,以定义访问该控制器方法的 URL 模式和请求类型。
  • 允许通过依赖注入引入其他层级的服务或组件,比如服务层 (@Service) 或数据访问层 (@Repository)。
  • 与视图技术(如 Thymeleaf、JSP 等)协同工作,可以返回 String 类型的视图名,由视图解析器进一步处理。
  • 当使用 Spring Boot 和 RESTful Web 服务时,通常会结合 @RestController 注解来使用,它是 @Controller@ResponseBody 的组合体,这意味着控制器的所有响应默认都会转换成 JSON 或 XML 格式返回给客户端。
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.ui.Model;

@Controller
public class MyController {

    @RequestMapping("/greet")
    public String greet(Model model) {
        model.addAttribute("message", "Hello, World!");
        return "greeting"; // 返回视图名称,通常是一个 HTML 页面
    }
}

③Repository

用于标识持久层组件(如DAO组件)的特殊化 @Component 注解。当你在类上使用 @Repository 注解时,它会告知 Spring 容器该类是一个 Bean,并且用于封装数据访问异常,将底层数据访问技术抛出的异常转换为 Spring 的 DataAccessException

// 告诉Spring,让Spring创建一个名字叫“userDao”的UserDaoImpl实例。
@Repository(value="userDao")
public class UserDaoImpl extends BaseDaoImpl<User> {
………
}

// 注入userDao,从数据库中根据用户Id取出指定用户时需要用到
@Resource(name = "userDao")
private BaseDao<User> userDao;

④Configuration

@Configuration 是表明某个类是用来作为 bean 定义的源的。被 @Configuration 注解的类通常包含了一个或多个标记有 @Bean 注解的方法。Spring 容器会在运行时自动调用这些方法,将返回对象注册为容器中的 bean。

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

@Configuration
public class AppConfig {

    @Bean
    public MyService myService() {
        return new MyServiceImpl();
    }
}
// AppConfig 类被标注为 @Configuration,这意味着它可以包含 bean 定义。myService 方法上的 @Bean 注解告诉 Spring,这个方法将返回一个对象,该对象需要被注册为应用程序上下文中的一个 bean。默认情况下,bean 的名称与方法名相同,但可以通过 @Bean 注解的 name 属性来自定义。

// 这种配置方式实现了代码的模块化,并且取代了传统的 XML 配置文件,使得配置更加清晰和类型安全

⑤Resource

  • @Resource 默认按照 by-name 自动装配,即根据 bean 的名称进行匹配;如果没有找到与名称匹配的 bean,则会退回到 by-type 自动装配(首先尝试依据属性名作为 bean 名称查找,如果失败则按类型装配)。而 @Autowired 默认按照 by-type 自动装配。
  • @Resource 所标注的自动装配过程是在 bean 属性设置完成之后、初始化方法(如 @PostConstruct 注解的方法)之前进行的,而 @Autowired 则是在构造器、字段、setter 方法或其他任意带有参数的方法上使用。
import javax.annotation.Resource;
@Component
public class MyComponent {
    // By name
    @Resource(name = "myBean")
    private MyBean myBeanByName;

    // By type (当存在多个相同类型的 bean 时,可能需要指定 name 来避免冲突)
    @Resource
    private MyBean myBeanByType;

    public void doSomething() {
        // 使用 myBeanByName 和 myBeanByType 完成一些操作...
    }
}
// @Resource 注解有两个重要的属性:name 和 type:
name: 指定要注入的 bean 的名称。
type: 指定要注入的 bean 的类型。
微观

① Async:将方法标注为异步执行

在一个普通的方法或者类中调用了由 @Async 注解标记的方法,那么当前线程会立即返回,而实际执行的过程会在另外的线程中进行,当前线程将继续执行自身的任务。

@Service
public class MyService {
  
  @Autowired
  private EmailService emailService;

  @Async
  public void sendEmails(List<String> recipients, String content) {

    for (String recipient : recipients) {
      emailService.sendEmail(recipient, content);
    }

    System.out.println("All emails sent!");
  }
}

②Autowired:自动装配依赖的 bean。

@Autowired 注解是 Spring Framework 中的一个依赖注入注解,它可以自动将容器中已经创建好的 bean 对象装配到需要他们的类或者方法中。默认先按byType进行匹配,如果发现找到多个bean,则又按照byName方式进行匹配,如果还有多个,则报出异常

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

@Component
public class MyComponent {

    private final Dependency dependency;

    // 构造器注入,如果容器中不存在Dependency类型的Bean,应用将无法启动
    @Autowired(required = true)
    public MyComponent(Dependency dependency) {
        this.dependency = dependency;
    }
    
    // 如果容器中不存在OptionalDependency类型的Bean,则dependency字段将为null,但应用依旧会启动
    @Autowired(required = false)
    private OptionalDependency optionalDependency;
}

③Bean:声明一个需要被 Spring 管理的 bean

@Configuration // 添加了 @Configuration 注解来标识这是一个配置类,因此 Spring 容器在扫描时会识别并加载该类。
public class MyConfiguration {
  @Bean
  public UserDao userDao() {
    return new JdbcUserDao(dataSource());
  }
  @@Bean(“myBean”) // 这样可以在装配其他依赖项时方便地引用该 bean 对象。
  public DataSource dataSource() {
    // 创建并配置数据源对象
    return new HikariDataSource(config);
  }
}

在上面的代码中,我们定义了一个配置类 MyConfiguration,并在其中定义了两个 bean 对象:userDao() 和 dataSource()。其中 userDao() 方法返回 JdbcUserDao 对象,而 dataSource() 方法返回一个 Hikari 数据源对象。这两个方法都标记了 @Bean 注解,这意味着它们会被 Spring 容器扫描到并将创建的对象注册到容器中。

④Cacheable:增加缓存支持

@Cacheable 是 Spring Framework 中的一个注解,用于实现方法级别的缓存。使用 @Cacheable 注解的方法在调用时,将会检查缓存中是否存在与该方法所需参数相对应的缓存项。如果存在,则直接返回缓存结果,不再执行被注解的方法;如果不存在,则执行该方法,并将方法的返回结果添加到缓存中。

@Service
public class UserService {

  private final UserMapper userMapper;

  public UserService(UserMapper userMapper) {
    this.userMapper = userMapper;
  }

  @Cacheable(value = "users", key = "#userId")
  public User getUserById(int userId) {
    return userMapper.getUserById(userId);
  }

}

⑤ Conditional

Conditional用于根据特定条件动态决定是否创建某个 Bean 实例,Spring 还提供了一些预定义的条件注解,如 @Profile、@ConditionalOnMissingClass、@ConditionalOnProperty 等,它们可以方便地满足一些常见的判断场景,减少了编写自定义条件类的工作量。

@Configuration
public class MyConfiguration {
  @Bean
  @Conditional(MyCondition.class)
  public MyBean myBean() {
    return new MyBean();
  }
}

public class MyCondition implements Condition {
  @Override
  public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
    // TODO: 自定义条件判断逻辑
    return true; // 当结果为 true 时,表示条件成立,按照要求创建 Bean;否则,则不会创建这个 Bean。
  }
}
使用 @Profile 注解可以确保只有相关的 beans 在特定的环境配置下被创建,这样可以避免在不适合的环境中运行可能引起冲突或异常的 beans,并有助于维护清晰的环境特定配置。
@Profile("development")
@Profile("production")

根据设置系统属性: -Dspring.profiles.active=development
或者使用环境变量设置 SPRING_PROFILES_ACTIVE=development等来切换profile

⑥ConstructorBinding

用于表示在使用基于构造器的依赖注入时,应该将配置属性绑定到构造器参数上

import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.context.properties.ConstructorBinding;

@ConstructorBinding
@ConfigurationProperties(prefix = "app") //指定哪个前缀的配置属性应被绑定到该类的字段上,Spring Boot 将会查找 app.name 和 app.threadPoolSize 配置项,并通过构造器自动注入相应的值。
public class AppConfigProperties {

    private final String name;
    private final int threadPoolSize;

    public AppConfigProperties(String name, int threadPoolSize) {
        this.name = name;
        this.threadPoolSize = threadPoolSize;
    }

    // Getters for the fields
    public String getName() {
        return name;
    }

    public int getThreadPoolSize() {
        return threadPoolSize;
    }
}

⑦ControllerAdvice

@ControllerAdvice 是 Spring MVC 框架中的注解,用于定义一个全局性的异常处理器类。可以对 Controller 层面和全局异常进行统一的处理。

@ControllerAdvice
public class GlobalExceptionHandler {

    @ExceptionHandler({NullPointerException.class, IllegalArgumentException.class}) //当捕获到空指针异常或者非法参数异常时,会执行这个方法,将异常信息添加到 request 中,并返回 “error” 视图。
    public String handleException(Exception e, HttpServletRequest request) {
        request.setAttribute("error", e);
        return "error";
    }
}

⑧CrossOrigin

同源策略阻止一个域下的文档或脚本与另一个域下的资源进行交互。这有助于保护用户免受恶意网站的攻击,但也限制了合法的跨源请求。

例如,如果你的页面在 https://www.example.com 上运行,而尝试通过 AJAX 请求从 https://api.another-site.com 获取数据,这将被默认视为跨域请求,并可能遭到浏览器的阻止,除非目标服务器明确允许来自原始域的请求。
@RestController
@RequestMapping("/api")
public class ApiController {

    @CrossOrigin(origins = "http://localhost:8080")
    @GetMapping("/users/{id}")
    public User getUserById(@PathVariable Long id) {
        // ...
    }
}

⑨EnableAsync

在处理一些 IO 密集型操作时,非常有用

@Configuration
@EnableAsync // @EnableAsync 注解需要与异步方法搭配使用,异步方法需要用 @Async 标记。
public class AppConfig {

    @Bean
    public Executor taskExecutor() {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        executor.setCorePoolSize(10);
        executor.setMaxPoolSize(20);
        executor.setQueueCapacity(30);
        executor.initialize();
        return executor;
    }
}

⑩RequestMapping

@RequestMapping 的专门化版本提供了更简洁的语法。这些包括 @GetMapping, @PostMapping, @PutMapping, @DeleteMapping, @PatchMapping。每个都对应于一个 HTTP 方法,使得代码更加简洁易读。

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.ResponseBody;

@Controller
@RequestMapping("/api")
public class MyController {

    @GetMapping("/hello") // 等同于 @RequestMapping(value = "/hello", method = GET)
    @ResponseBody
    public String helloWorld() {
        return "Hello, World!";
    }

    @PostMapping("/submit") // 等同于 @RequestMapping(value = "/submit", method = POST)
    @ResponseBody
    public String submitForm() {
        // 方法实现
        return "Form submitted";
    }
}

其他

①lazy

@Lazy 是 Spring 框架中的注解之一,它用于控制 Bean 的实例化时间。通过在容器中标记一个 Bean 为 @Lazy,可以让 Spring 容器在第一次使用这个 Bean 时再进行实例化。相对应的,如果不加 @Lazy 注解,则默认情况下 Spring 容器会在启动时就实例化该 Bean。

@Service
@Lazy // 使用 @Lazy 注解,Spring 容器将在第一次使用 UserService 的实例时才进行实例化而不是在启动时就实例化
public class UserService {
    // ...
}

②Import

允许我们在一个配置类中引入其他配置类或普通的 Java 类

@Configuration
@Import({DataSourceConfig.class, RedisConfig.class, ServiceUtils.class})
public class AppConfig {
    // ...
}
// 在创建 Spring ApplicationContext 时,这三个配置类都会被加载并注册为 Spring 的 bean。
// 可以在 AppConfig 中使用 ServiceUtils 类中的方法或属性

③元注解

  • @Target:指定注解可以应用于 Java 的哪些元素(如类、方法、字段等)。
  • @Retention:指定注解在什么级别可用(源码、类文件或运行时)。
  • @Documented:指定注解是否应该被 javadoc 工具记录。
  • @Inherited:指定注解是否可以被子类继承。
  • @Component:Spring 特有的元注解,用于声明一个类是 Spring 组件。其它注解如 @Service, @Repository, @Controller 都是用 @Component 注解的例子。

④Valid

在控制器方法的参数列表中使用 @Valid ( @NotNull、@Min、@Max、@Size、@Email…)注解来实现请求参数的校验。

@RestController
@RequestMapping("/users")
public class UserController {
    @PostMapping
    public User createUser(@Valid @RequestBody User user) {
        // 保存用户信息到数据库
    }

    // ...
}

// 校验的规则通常定义在 User 类的属性上
public class User {
    @NotBlank
    private String username;

    @Pattern(regexp = "^[a-zA-Z0-9]+@[a-zA-Z0-9]+\\.[a-zA-Z]{2,}$", message = "Email 格式错误")
    private String email;
}
第三方
  1. @Data

    @Data注解是Lombok库的一部分,非Spring框架本身的一部分。当你在一个类上使用@Data注解,Lombok会自动生成以下代码:

    • 所有字段的getter方法
    • 所有非最终字段的setter方法
    • toString()方法
    • equals()hashCode()方法
    • 一个包含所有非静态、非瞬态字段的构造函数

    这大大减少了样板代码的数量,使得类的定义更加简洁

  2. @Mapper

    @Mapper注解是MyBatis框架的一部分,它主要用于标记一个接口作为数据库操作的映射接口。MyBatis通过这个接口关联XML配置文件或者注解中定义的数据库操作。

    @Mapper
    public interface UserMapper {
        User selectUserById(Long id);
    }
    
    // 在Spring Boot项目中,通常还会配合使用@MapperScan注解来指定需要扫描的@Mapper接口所在的包路径。
    @SpringBootApplication
    @MapperScan("com.example.project.mapper")
    public class Application {
        public static void main(String[] args) {
            SpringApplication.run(Application.class, args);
        }
    }
    

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

vue3+Pinia的使用 - 封装

目录&#xff1a; persist.ts 可存储到本地 import { PersistedStateOptions } from "pinia-plugin-persistedstate";/*** description pinia 持久化参数配置* param {String} key 存储到持久化的 name* param {Array} paths 需要持久化的 state name* return per…

基于Transformer的医学图像分类研究

医学图像分类目前面临的挑战 医学图像分类需要研究人员同时具备医学图像分析和数字图像的知识背景。由于图像尺度、数据格式和数据类别分布的影响&#xff0c;现有的模型方法&#xff0c;如传统的机器学习的识别方法和基于深度卷积神经网络的方法&#xff0c;取得的识别准确度…

2024第六届环境科学与可再生能源国际会议能源 (ESRE 2024) 即将召开!

2024第六届环境科学与可再生能源国际会议 能源 &#xff08;ESRE 2024&#xff09; 即将举行 2024 年 6 月 28 日至 30 日在德国法兰克福举行。ESRE 2024 年 旨在为研究人员、从业人员和专业人士提供一个论坛 从工业界、学术界和政府到研究和 发展&#xff0c;环境科学领域的专…

Kubernetes 知识体系 系列一

多年前&#xff0c;大多数软件应用程序都是大型的单体&#xff0c;要么作为单个进程运行&#xff0c;要么作为少数服务器上的少量进程运行。这种过时的系统一直延续很久。 它们的发布周期较慢&#xff0c;更新相对较少。 在每个发布周期结束时&#xff0c;开发人员将整个系统…

算法第三十四天-有效数独

有效数独 题目要求 解题思路 一个简单的方法是&#xff0c;遍历9*9书读三次&#xff0c;以确保&#xff1a; 行中没有重复的数字列中没有重复的数字3*3子数独中没有重复的数字 但是&#xff0c;实际上&#xff0c;所有的一切都以可以在一次迭代中完成 可以使用box_index (r…

网约车APP小程序源码代驾顺风拼车货运司乘端安卓苹果源码可二开

内容目录 一、详细介绍二、效果展示1.部分代码2.效果图展示 一、详细介绍 系统是基于Thinkphpuniapp开发的&#xff0c;全开源未加密&#xff0c;这套源码可以拿回去自己做二开 后台用户端司机端 功能详情介绍&#xff1a; 车主实名认证&#xff0c;驾驶证认证&#xff0c;车…

Python列表、元组、字典及集合

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 一、列表定义方式&#xff1a; 二、元组1、定义方式&#xff1a;2、元组中的物理存储地址不可修改,如果修改则会报错&#xff0c;但是元组中的列表、字典项等却可以…

代码随想录算法训练营第三十七天 | 738. 单调递增的数字、968. 监控二叉树

代码随想录算法训练营第三十七天 | 738. 单调递增的数字、968. 监控二叉树&#xff08;待刷&#xff09; 738. 单调递增的数字题目解法 968. 监控二叉树&#xff08;待刷&#xff09; 738. 单调递增的数字 题目 解法 class Solution { public:int monotoneIncreasingDigits(in…

File和IO流

1. File类常用方法 1.1 获取基本属性 • public String getName() &#xff1a;获取名称 • public String getPath() &#xff1a;获取路径 • public String getAbsolutePath()&#xff1a;获取绝对路径 • public File getAbsoluteFile()&#xff1a;获取绝对路径表示…

嵌入式培训3-28

编写一条学生链表&#xff0c;写一些能够像链表里边添加数据的函数 实现&#xff1a;将链表中的所有内容保存到文件中去 以及 读取文件中的所有内容&#xff0c;加载到链表里面 #include <stdio.h> #include <string.h> #include <stdlib.h> #include <ma…

AD学习笔记

1. EOS 选择中心点 2. PDL 标记距离 3. 选择线框&#xff0c;按DSD重新定义板框 4. su框选&#xff0c;sl线选 5. mx x/y轴移动 6. dk 打开层叠管理器 7. shifts单层显示 8. 单端引脚不报错&#xff0c;CO工程选项下关闭报错即可。 9. PCB界面中寻找元件&#xff1a;…

蓝桥杯 - 小明的背包3(多重背包)

解题思路&#xff1a; 动态规划 多重背包问题需要在01背包问题&#xff08;不重复&#xff09;的基础上多加一层循环进行遍历&#xff0c;并且dp[ j ]的式子也需要修改 import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scan …

【JavaSE】网络编程

定义&#xff1a;在网络通信协议下&#xff0c;不同计算机上运行的程序&#xff0c;进行的数据传输 常见的软件架构&#xff1a; C/S&#xff1a;Client/Server&#xff08;客户端/服务器&#xff09; 在用户本地需要下载并安装客户端程序 画面可以很精美&#xff0c;用户体…

【面试经典 | 150】单词拆分

文章目录 Tag题目来源解题思路方法一&#xff1a;动态规划 写在最后 Tag 【动态规划】【字符串】 题目来源 139. 单词拆分 解题思路 方法一&#xff1a;动态规划 定义状态 定义 dp[i] 表示字符串 s 前 i 个字符组成的字符串&#xff08;s[0, ..., i-1]&#xff09;是否能被…

7.JDK下载和安装

文章目录 一、下载二、安装三、JDK的安装目录介绍 写JAVA代码不是随随便便能写的&#xff0c;我们得先做一点准备工作。例如&#xff0c;我们平时想要玩一把游戏&#xff0c;就需要先下载、安装才能玩游戏。JAVA也是一样的&#xff0c;也是需要下载并安装相关的软件&#xff0c…

2010-2021年银行网点及员工信息数据

2010-2021年银行网点及员工信息数据 1、时间&#xff1a;2010-2021年 2、来源&#xff1a;整理自csmar 3、指标&#xff1a;银行代码、股票代码、银行中文简称、统计截止日期、分行数量、机构网点数量、其中&#xff1a;境内网点数量、其中&#xff1a;境外网点数量、在职员…

[疑难杂症2024-002]一个“显而易见“的问题,是如何进入生产环境的?

本文由Markdown语法编辑器编辑完成。 1. 前言 最近在处理一个在医院上线的系统的问题。这个问题&#xff0c;由于关联的模块比较多&#xff0c;至少涉及到3个模块之间的功能调用。因此&#xff0c;协调大家都有时间来排查问题不是很方便。这个问题就拖了有一周左右。医院那边…

2024-03-28 Quest3 开发环境配置教程

文章目录 准备条件1 登录 Meta 账号2 Oculus 软件下载与配置3 下载 Quest3 开发包4 Unity 环境配置环境测试 准备条件 Quest3 头显一个。一根 USB 3.0 数据线。魔法。 ​ 有关 quest3 激活与配置可参考 B 站 UP &#xff1a;“南七月nqy_”。跳转链接&#xff1a;https://spa…

Exception in thread “main“ com.fasterxml.jackson.databind.JsonMappingException:

问题&#xff1a;jaskson反序列化超出最大长度 Caused by: com.fasterxml.jackson.core.exc.StreamConstraintsException: String length (5043456) exceeds the maximum length (5000000) 场景&#xff1a;前端传递过大base64 原因&#xff1a; jaskon默认已经限制了最大长…

货币系统(闫氏DP分析法)

题目描述&#xff1a; 给定 V 种货币&#xff08;单位&#xff1a;元&#xff09;&#xff0c;每种货币使用的次数不限。 不同种类的货币&#xff0c;面值可能是相同的。 现在&#xff0c;要你用这 V 种货币凑出 N 元钱&#xff0c;请问共有多少种不同的凑法。 输入格式&am…
最新文章