Spring添加注解读取和存储对象

news2025/1/12 2:54:31

5大注解

@Controller 控制器

@Service 服务

@Repository 仓库

@Componet 组件

@Configuration 配置

五大类注解的使用

//他们都是放在同一个目录下,不同的类中 只不过这里粘贴到一起

//控制器
@Controller
public class UserController {
    public void SayHello(){
        System.out.println("Hello");
    }
}

//服务
@Service
public class UserService {
    public void doService(){
        System.out.println("Service");
    }
}

//仓库
@Repository
public class UserRepository {
    public void DoRepository(){
        System.out.println("Repository");
    }
}

//组件
@Component
public class UserComponent {
    public void doComponent(){
        System.out.println("Component");
    }
}

//配置
@Configuration
public class UserConfiguration {
    public void doConfiguration(){
        System.out.println("userConfiguration");
    }
}

//启动类
public class App {
    public static void main(String[] args) {
        //获取spring上下文
        ApplicationContext context=new ClassPathXmlApplicationContext("Spring-config.xml");

        //获取Controller bean对象
        //使用注解默认是小驼峰
        UserController userController=context.getBean("userController",UserController.class);

        userController.SayHello();
        
        //获取Service bean对象
        UserService userService=context.getBean("userService",UserService.class);

        userService.doService();

        //获取Repository bean对象
        UserRepository userRepository=context.getBean("userRepository",UserRepository.class);

        userRepository.DoRepository();

        //获取Component bean对象
        UserComponent userComponent=context.getBean("userComponent",UserComponent.class);

        userComponent.doComponent();

        //获取Configuration bean对象
        UserConfiguration userConfiguration=context.getBean("userConfiguration",UserConfiguration.class);

        userConfiguration.doConfiguration();
    }
}

总结:Controller(控制器)、Service(服务)、Repository(仓库)、Configuration(配置)都是基于Component 他们的作用都是将Bean存储到Spring中

获取bean时,都需要将类的首字母小写

命名规则

默认情况下,使用5大类注解获取bean名称,要将首字母小写

特例:当首字母和第二个字母都是大写的时候,那么获取bean名称时,使用原名称即可

创建类时,如果要把类放入spring内管理,那么你的类名一定要遵守命名规则

1.首字母大写,后续小写

2.首字母大写,第二个字母也是大写

为什么要这样命名呢?

查看源码

ctrl+鼠标左键进入

    public static String decapitalize(String name) {
        if (name == null || name.length() == 0) {
            return name;
        }
        if (name.length() > 1 && Character.isUpperCase(name.charAt(1)) &&
                        Character.isUpperCase(name.charAt(0))){
            return name;
        }
        char chars[] = name.toCharArray();
        chars[0] = Character.toLowerCase(chars[0]);
        return new String(chars);
    }

可以看到源码是这么写的

如果为空时,返回源字符串

如果长度大于1 并且 字符串第二个字符串和首字符都是大写的时候,也返回源字符串

只有当这两个条件不成立的时候,就把字符串第一个数改为小写

测试

        //这个获取bean对象时,需要首字母小写
	   String userComponent = "UserComponent";
	   //这个首字母和第二个字母都是大写,获取bean对象时,使用源类名即可
        String sComponent = "SComponent";
        System.out.println(Introspector.decapitalize("UserComponent"));
        System.out.println(Introspector.decapitalize("SComponent"));

存储bean对象

创建一个普通类,创建一个输入数据类

这样就把User放入spring内管理,并且初始化,初始化的方法必须加Bean

为什么bean必须和五大类的其中一个配合用才可以呢?

性能原因,spring不可能把全部的类全加入到spring内管理,那太消耗性能了,所以必须先表名五大类其中一个,加了类注解,再类注解下加了bean 方法注解,才会把该类加入到spring内管理

Bean

Spring内管理的对象统称为Bean

方法的命名规则就是方法名

当多个类使用同一个方法名,并且对象都存入spring中时,可以给方法名起别名

起别名的方法有三种

 @Bean(name="UserBean")

    //取多个名称
    @Bean(name={"studentbean","stubean"})
    @Bean("StudentBean")
    //重命名方法名  获取bean对象时就之间写重命名的即可
    //当重命名后,原来的方法名就不能使用了

DI注入

DI注入时机取决于bean的作用域,比如默认情况bean的作用域为单例模式,当需要对象时,才会注入。

DI注入都是从Spring内获取,Spring内没有的就没法注入

对象注入有三种

  1. 属性注入 (Field Injection)
  2. Setter注入 (Setter Injection)
  3. 构造方法注入 (Constructor Injection)

@Autowired

属性注入(Field Injection)

@Autowired //添加注解
    private UserService userService;//获取userService对象


    public void SayHello(){
        userService.doService();//使用userService方法
        System.out.println("Hello");
    }

启动类内不能注入,因为static执行速度是非常快的,在属性还没注入时,main函数已经执行了。所以main执行时,获取不到从spring获取的对象。

优点

现简单,使用方便,注入之后即可使用

缺点

1.不能注入被final修饰的对象

原因:final关键字在Java中表示一个字段是不可变的,它必须在声明时或在构造函数中被初始化,并且之后不能被修改。

当你使用依赖注入时,注入的值通常是在运行时由容器设置的,这意味着属性的值是在对象创建之后才确定的。但是,由于final字段必须在构造函数中被初始化,这就产生了一个矛盾:你不能在构造函数中给final字段赋值,因为那时注入的值还没有确定。

除非直接赋值或在构造方法内赋值

2.通用性问题 只适用于IoC框架(容器)
3.更容易违背单一设计原则

单一职责原则指出,一个类应该只有一个引起变化的原因,即一个类应该只负责一项任务。当使用 @Autowired 进行属性注入时,可能会导致以下问题:

  1. 职责不清晰:如果一个类中有多个 @Autowired 注解的属性,那么这个类可能承担了多个职责。这违反了单一职责原则,因为如果这些依赖中的任何一个发生变化,这个类可能需要修改。
  2. 依赖过多:一个类如果依赖了太多的其他类,那么它就变得复杂,难以理解和维护。这可能导致类之间的耦合度增加,违反了高内聚低耦合的设计原则。

Setter注入(Setter Injection)

//setter注入 (setter Injection)
    private UserService userService;

    @Autowired
    public void setUserService(UserService userService) {
        this.userService = userService;
    }

优点

符合单一设计原则 (一个setter只针对一个对象)

缺点

1.不能注入一个不可变(final)的对象 (和属性注入一样)

原因:final关键字在Java中表示一个字段是不可变的,它必须在声明时或在构造函数中被初始化,并且之后不能被修改。

当你使用依赖注入时,注入的值通常是在运行时由容器设置的,这意味着属性的值是在对象创建之后才确定的。但是,由于final字段必须在构造函数中被初始化,这就产生了一个矛盾:你不能在构造函数中给final字段赋值,因为那时注入的值还没有确定。

2.注入对象可能会被改变 (致命问题) set方法可能会被调用多次 被调用就有被修改的风险

构造方法注入(Constructor Injection)

官方推荐做法

//构造方法注入(Constructor Injection)
    private UserService userService;

    @Autowired
    public UserController(UserService userService){
        this.userService=userService;
    }

当当前类只有一个构造方法,那@Autowired可以省略

一个构造方法内可以注入多个对象

//构造方法注入(Constructor Injection)
    private UserService userService;
    private UserRepository userRepository;

    @Autowired
    public UserController(UserService userService,UserRepository userRepository){
        this.userService=userService;
        this.userRepository=userRepository;
    }

一个类可以有多个构造方法,但是只允许一个构造方法有@Autowired注解(DI注入构造只允许有一个)

并且,构造方法的参数只允许有在spring内存储的,其他参数没有办法传入构造方法,所以会报错。

优点

1.可以注入final修饰的对象

final修饰的变量必须赋初始值

赋值只有两种方法

1.创建时直接赋值

2.在构造方法中赋值

构造注入刚好满足第二种需求,所以构造注入允许注入final的对象

2.注入对象不会被修改

构造方法只能执行一次,所以赋值完就不会再赋值

3.依赖对象在使用前会被完全初始化

使用前一定会被构造方法初始化

4.通用性更好,当不使用IoC容器时,构造方法依旧奏效

@Resource

也是实现对象注入

注入方式也是同样的属性注入、setter注入

但Resource不支持构造注入

方式也是一样的,只是注解不同

属性注入

//    属性注入
//读取spring中名为userService的对象,赋值给当前注入的对象
//有了这个name  所以即使下面的名称不一致,依靠上面的name依旧可以找到指定对象
    @Resource(name="userService")
    private UserService userService;

Resource可以指定注入对象的名称

Setter注入

    //setter注入
    private UserController userController;

    @Resource
    public void setUserController(UserController userController) {
        this.userController = userController;
    }

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

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

相关文章

在新页面中跳转到指定 div容器位置

要在打开新的页面时跳转到指定 div,我们需要结合 HTML、JavaScript 和后端技术来实现。以下是两种常见的方法: 使用 URL 参数传递目标 div 信息 HTML (新页面): 在新页面的链接中,添加参数来指示目标 div 的 id,例如:…

Android GPU渲染SurfaceFlinger合成RenderThread的dequeueBuffer/queueBuffer与fence机制(2)

Android GPU渲染SurfaceFlinger合成RenderThread的dequeueBuffer/queueBuffer与fence机制(2) 计算fps帧率 用 adb shell dumpsys SurfaceFlinger --list 查询当前的SurfaceView,然后有好多行,再把要查询的行内容完整的传给 ad…

im8mm 网络卡死 Rx packets:1037578 errors:66 dropped:0 overruns:66 frame:0

1:网络接收数据包异常 2:问题复现 问题在进行网络数据包同吞吐量测试的时候出现的。同时发现,在使用iperf2测试时,是不会出现网络中断卡死的情况,使用 iperf3时才会出现此问题 指令(下面的指令运行在PC2上面&#xff…

Linux 安装JDK和Idea

安装JDK 下载安装包 下载地址: Java Downloads | Oracle (1) 使用xshell 上传JDK到虚拟机 (2) 移动JDK 包到/opt/environment cd ~ cd /opt sudo mkdir environment # 在 /opt下创建一个environment文件夹 ls# 复制JDK包dao /opt/environment下 cd 下载 ls jd…

聊天室项目思路

发起群聊: 从好友表选取人发送到服务器,服务器随机生成不重复的群号,存储在数据库,同时建立中间表,处理用户与群聊的关系 申请入群: 输入群号,发消息给服务器,服务器查询是否存在…

Ftp笑脸漏洞(VSFTPD 2.3.4)复现(后门漏洞)

Ftp笑脸漏洞(VSFTPD 2.3.4)复现(后门漏洞) 一、原理二、复现准备三、漏洞复现四、Metasploit利用脚本复现 一、原理 vsftpd 是“ very secure FTP daemon ”的缩写,安全性是它的一个最大的特点。 vsftpd是一个 UNIX 类…

【智能优化算法】矮猫鼬优化算法(Dwarf Mongoose Optimization Algorithm,DMHO)

矮猫鼬优化算法(Dwarf Mongoose Optimization Algorithm,DMHO)是期刊“COMPUTER METHODS IN APPLIED MECHANICS AND ENGINEERING”(IF 7.3)的2022年智能优化算法 01.引言 矮猫鼬优化算法(Dwarf Mongoose Optimization Algorithm,DMHO)模仿矮猫鼬的觅食行…

【Linux】Linux——Centos7安装Nginx

不需要安装包 1.安装依赖 #查看 C 环境是否安装gcc -v #查看 zlib 是否安装cat /usr/lib64/pkgconfig/zlib.pc #查看 pcre 是否安装pcre-config --version 2.安装C #安装C yum install gcc-c 3.安装pcre yum install -y pcre pcre-devel 4.安装zlib #安装 yum install -y zlib…

C++对象的拷贝构造函数

如果一个构造函数的第一个参数是类本身的引用,且没有其它参数(或者其它的参数都有默认值),则该构造函数为拷贝构造函数。 拷贝(复制)构造函数:利用同类对象构造一个新的对象 ●1.函数名和类同名 (构造函数) ●2.没有返回值 (构造函数) ●3.第一个参数必…

【AIGC】重塑未来的科技巨轮

AIGC:重塑未来的科技巨轮 一、AIGC:从历史走来,向未来进发二、AIGC的三项核心技术三、AIGC的应用与未来 在当今科技飞速发展的时代,AI(人工智能)已经成为了一个无法忽视的热词。而与其紧密相连的AIGC&#…

配置OpenSSH/stelnet

其他远程连接工具:telent、realVNC、RSH、RCP等,SSH更加安全可靠 一、配置OpenSSH/stelnet 1.配置服务端 # vim /etc/ssh/sshd_config //修改ssh配置文件 Port 22 //监听端口 AddressFamily any //使用哪种地址簇,any包含v4、v6&#xff0c…

运行SpringBoot项目失败?异常显示Can‘t load IA 32-bit .dll on a AMD 64-bit platform,让我来看看~

原因是,我放入jdk的bin文件夹下的tcnative-1.dll文件是32位的,那么肯定是无法在AMD 64位平台上加载IA 32位.dll。但是网站上给出的都是32位呀,没有64位怎么办: 其实当我们把“tomcat-native-1.2.34-openssl-1.1.1o-win32-bin.zip”…

【C++】详细版 RAII技术的应用之智能指针(智能指针发展历程和简单模拟实现介绍)

目录 前言 一、智能指针有什么用? 二、什么是RAII(智能指针的底层思想)? 三、智能指针的发展历程以及模拟实现 1.auto_ptr(C98) 2.unique_ptr(C11) 3.shared_ptr(C11) 前言 C中…

Modown9.1主题无限制使用+Erphpdown17.1插件

Modown9.1主题无限制使用 1、Erphpdown17.1插件Modown9.1主题 2、送Modown主题详细教程。 1、Erphpdown插件和Modown主题无需激活 2、送的插件均无需激活 3、主题插件均不包更新 4、已亲测可以完美使用。 功能强大,适用于绝大多数虚拟资源站!物超所值&a…

智慧公厕:让厕所管理变得更智慧、高效、舒适!

公共厕所是城市的重要组成部分,但常常被忽视。它们的管理和养护往往面临着许多问题,例如卫生状况不佳、环境畏畏缩缩、设施老旧等。为了解决这些问题,智慧公厕应运而生。智慧公厕是一种全方位的应用解决方案,将科技与公共厕所管理…

RERCS系统-WDA+BOPF框架实战例子 PART 2-新建Root的子节点Node Element

1、通过事务码 BOBF进入Business Object Browser(业务对象浏览)页面; 2、输入debug 进入编辑模式; 3、双击对应的业务对象进入Business Object Detail Browser即业务对象数据浏览器 在Node Structure的Root中新建需要的SubNode子…

【C++】学习笔记——优先级队列

文章目录 十、优先级队列1. priority_queue的介绍2. 优先级队列如何使小的数据优先级高3. 仿函数介绍4. priority_queue的模拟实现 补: 反向迭代器未完待续 十、优先级队列 1. priority_queue的介绍 优先级队列 其实也不属于队列,它跟 stack 和 queue …

[Linux_IMX6ULL驱动开发]-GPIO子系统和Pinctrl子系统

目录 Pinctrl子系统的概念 GPIO子系统的概念 定义自己的GPIO节点 GPIO子系统的函数 引脚号的确定 基于GPIO子系统的驱动程序 驱动程序 设备树修改 之前我们进行驱动开发的时候,对于硬件的操作是依赖于ioremap对寄存器的物理地址进行映射,以此来达…

SpringBoot实现统一返回值+全局异常处理

在这里首先感谢的就是程序员老罗&#xff0c;从他的项目里面学到了这些东西。 首先就是去创建一个SpringBoot项目&#xff0c;这里我就不多做赘述了 封装一个统一返回对象 package com.example.demo.vo;public class ResponseVO<T> {private String status;private In…

数据驱动实战二

目标 掌握数据驱动的开发流程掌握如何读取JSON数据文件巩固PO模式 1. 案例 对TPshop网站的登录模块进行单元测试 1.1 实现步骤 编写测试用例采用PO模式的分层思想对页面进行封装编写测试脚本定义数据文件&#xff0c;实现参数化 1.2 用例设计 1.3 数据文件 {"login…