Spring的创建与Bean对象的存取

news2024/9/22 2:42:05

文章目录:一.Spring项目的创建1.先创建maven项目

                                                      2.添加国内源

                                                      3.添加spring依赖

                                                      4.创建spring配置文件

                                                       5.创建启动类

                   二.Bean对象的创建和读取1.Bean对象的创建与存储方式(1)类注解

                                                                                                                      (2)方法注解

                                                                                                                         (3)<bean>格式

                                                               2.Bean对象的获取方式(1)通过id

                                                                                                       (2)通过类名

                                                                                                       (3)id+类名

                                                                                                         (4)注解@Autowired                                                                   

                                                                                                (5)@Autowired @Resource区别             

一.Spring项目的创建

1.先创建一个maven项目

2.添加国内源

查看Local reposittory目录下是否有settings这个文件

将settings.xml文件里在<mirros>里面添加阿里云镜像

 

3.添加spring-content依赖

4.添加配置文件在resources底下

 在文件里添加如下内容

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:content="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">
    <content:component-scan base-package="kale"></content:component-scan>
    <!--<bean id="beanlife" class="kale.pole.Beanlife" init-method="init" destroy-method="preDestroy"> </bean>-->

</beans>

5.添加启动类

二.Bean对象的创建与读取

1.Bean对象的创建

我们先来创立一个Student类

package spring;

public class Student {
    public String name="网红";
    public  int id=3;
    public int  score=97;
    String s=new String();
    public Student()
    {

        System.out.println("学生初始化");
    }
    public void say()
    {
        System.out.println("你好学生");
    }
}

在resources文件底下,新建一个spring-config(文件名可以随意起) xml文件

在xml文件里引入:

将bean对象存储到spring容器中

然后我们就可以在里面注册bean 对象,我们将student对象注册到spring中

<bean id="student" class="spring.Student"></bean>

 其中 class后面需要写  包名+类名

2.从spring容器中获取bean

1.因为我们是在spring中注册bean,如果我们想要获取到,我们需要先获取到spring上下文,这里我们就涉及到两个类(ApplicationContext、BeanFactory)

2.然后根据注册的id,来获取对应的bean对象

 public static void main(String[] args) {
        (1)//获取spring上下文
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
        (2)//根据id来取容器里对应的bean对象
       Student result=(Student)context.getBean("student");
        result.say();

    }
 

  3.ApplicationContext  BeanFactory有什么区别:

我们再来创建一个Teacher类,然后在spring中将实例化的对象给注册到里面去

package spring;

import com.sun.scenario.effect.impl.sw.sse.SSEBlend_SRC_OUTPeer;

public class Teacher {
    public Teacher()
    {
        System.out.println("教师初始化");
    }
    public void say()
    {
        System.out.println("你好,教师");
    }

}
<bean id="teacher" class="spring.Teacher"></bean>

我们先来用 ApplicationContext 来演示

 public static void main(String[] args) {
        
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
       

    }

 

我们可以发现当用去获取上下文时,spring中所有的类都被加载了,也就是在这时我们才真正的去创建bean对象,一开始在spring-config.xml中我们只是去注册、声明bean对象。

我们再来看一下,BeanContext

 public static void main(String[] args) {
        BeanFactory context=new XmlBeanFactory(new ClassPathResource("spring-config.xml"));
}
       

    

当我们去运行这个程序的时候,发现什么都没有

 public static void main(String[] args) {
        BeanFactory context=new XmlBeanFactory(new ClassPathResource("spring-config.xml"));
        Student result=(Student)context.getBean("student");

    }

 这时我们再调用一下getBean()方法,我们再运行程序

 由此我们可以得出当我们去使用BeanFactory的时候,我们发现只有当我们去取这个对象的时候,我们才会真正去加载并创建它

区别:(1)BeanFactory是 ApplicationContext的父类,其功能和方法相较于ApplicationContext要更少,这样效率、性能会高

        (2)ApplicationContext一次加载并实例化所有的Bean对象,而BeanFactory是能用到那个Bean对象再去加载初始化这个对象的,这样节省空间,但效率低

4.Spring存取对象的基本流程:

1.Spring读取Bean流程:先扫描xml文件配置,从配置中或注解中获取Bean对象的定义信息,然后把它注册到Spring容器中

2.Spring加载并实例化Bean:先给Bean对象分配空间,然后进行初始化,然后把它设置到Spring容器中

5.获取bean对象的几种方式

(1)通过id名称获取bean

 public static void main11(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
       Student result=(Student)context.getBean("student");
        result.say();

    }

(2)通过类名获取bean:

public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-config.xml");
        Student result=context.getBean(Student.class);
        result.say();
    }

 当我们通过类名的方式去获取对象的时候,我们发现这个时候,我们不用再进行强转成Student类型了

但是这种方式存在一个问题,如果我们再注册一个Student类的对象在里面

 <bean id="student" class="spring.Student"></bean>
      <bean id="stu" class="spring.Studedt"></bean>
       <bean id="teacher" class="spring.Teacher"></bean>

这时我们再去通过这种方式获取这就存在问题了。

(3)正确的方式应该是通过  id+类名的方式

 public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
        Student student1 = context.getBean("student", Student.class);
        Student student2 = context.getBean("stu", Student.class);
        student1.say();
        student2.say();
    }

 

(4)更简单的存储Bean对象的方式,添加类注解

五大类注解:1.@Controller   (控制器,用来验证前端传来的数据是否正确,如果不正确,返回false)防止别人恶意访问程序,相当于安保系统

2.@Service 服务:负责调度和提供一些方法,但是不实际执行,相当于客服中心

 3.@Repository 持久层:和数据层交互,负责实际执行

 4.@Component   组件工具类

 5.@Configuration   配置项(项目当中的一些配置)

在这里边Compoent是所有这几个类的父类,其他的四个类都继承了这个类,我们随便打开其他四个类里面的一个类看看:

通过在在类外添加注解的方式也能存储Bean对象,但是我们需要在spring-conflig.xml中增加

  <content:component-scan base-package="com.gather"></content:component-scan>

base-package表明类所在的包

package com.gather;

import org.springframework.context.annotation.Configuration;

@Configuration
public class Student{
    public void say()
    {
        System.out.println("hello student");
    }
}

像这种添加类注解的方式,我们用getBean()的方式来获取的话,那么里面的参数应该如何写呢?其实也是一种类似于Bean名称+类名的方式,如果类名第一个字母大写的话,我们只需把第一个字母变成小写

public class Text {
    public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        Student student=(Student) context.getBean("student",Student.class);
        student.say();

    }
}

在这里我们看到如果类名第一个字母大写的话,id直接将类名第一个字母小写即可,

如果我们建的类名前两个都是大写的话,这样就不可以了

public class POlice {
    public void say()
    {
        System.out.println("hello police");
    }
}
public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        POlice police=context.getBean("police",POlice.class);
        police.say();
    }

 如果id直接也是把第一个字母小写的话,我们发现会报错,

 这时候id应该直接写类名

 public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        POlice police=context.getBean("POlice",POlice.class);
        police.say();
    }

 那么为啥是这样呢?我们来看一下这里相关的源码:

Bean命名规则:首字母默认是小写的,如果首字母第一个大写 ,Bean名称需要将第一个字母小写,如果前两个字母都是大写,那么Bean名称就是类名。

在这里我们还需要注意一个问题:

如果在spring-conflig.xml里只有这一种通过添加注解的方式来存取Bean对象

<content:component-scan base-package="com.gather"></content:component-scan>

我们要存取和Bean对象与之对应的类必须在这个包底下或者在这个包的子目录里,另外如果想要访问这个包底下与之对应的所有Bean对象,这些类都要加注解,举个例子:

 因为Farmer这个类没在com.gather这个包底下,也不在这个包的子包底下,所以相当于我们没把它注册到Spring容器中,所以就取不到这个对象

那应该如何解决呢?

我们在spring-conflig.xml再加一行Bean注册内容即可

 <bean id="farmer" class="Farmer"></bean>
public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        Farmer farmer=context.getBean("farmer",Farmer.class);
        farmer.say();

    }

通过在方法上加Bean注解的方式将Bean对象存储在服务器上,但是在类上我们要加上类注解,因为这样我们是存在一些类不需要去存储在Spring中,这样再去读取扫描和配置文件的时候,可以快速的定位存储在Spring中的类,方便快速取出对象,这样可以提高性能。

@Configuration
public class Use2 {
    @Bean
    public User getuser2()
    {
        User user=new User();
        user.name="小凯";
        user.age=5;
        return user;
    }
}

但是要注意这里的方法名必须是要存储Bean对象的类名

 public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        User user=context.getBean("getuser2",User.class);
        System.out.println(user.name);
    }

这里要注意getBean()方法里的第一个参数是方法名 

 我们来看一下打印信息

getBean()方法里面第一个参数也就是Bean名称是一个方法名,感觉怪怪的,其实我们是可以重新起一个名字的

@Configuration
public class User1 {
    @Bean(value = "user2")
    public User getuser2()
    {
        User user=new User();
        user.name="小凯";
        user.age=5;
        return user;
    }

}

 也就是在Bean注解之后用value重新起一个名,getBean()方法里第一个参数也就可以写这个名了

 User user=context.getBean("user2",User.class);

但是这里要注意如果重新起了名,getBean()方法里第一个参数就不能再用原来的方法名了(getuser2) 

大家在这里想一个问题,如果我们再在另一个类里面写一个和User1里面一模一样的方法,那么当我们再去取得时候

@Configuration
public class User2 {
    @Bean//(value="user7")
    public User getuser2() {

        User user = new User();
        user.name = "小中";
        user.age = 5;
        return user;


    }
}
 public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        User user=context.getBean("getuser2",User.class);
        System.out.println(user.name);
    }

这取决于谁先注入,谁先被加载,谁后注入,后注入的会覆盖先注入的,我们再来写段代码看看,

我们在两个类前面分别加上两个@Order(10) 、@Order(20)  这里的注解表示的是谁先注入,数字越小表示注入的越早,一开始是User1里面的先注入,当我们把数值改一@Order(20)@Order(10)

这时就会产生不同的结果了。

取Bean更简单的方式使用@AutowiredResource

@Autowired 注解表示将需要的对象从Spring容器里取出来,

1.属性注入

我们先建立一个Police类

@Configuration
public class Police {
    public String name;
    public int age;
    public void say()
    {
        System.out.println("hello police");
    }
}

然后再创建一个UsePolice类,我们通过属性注入的方式将Police对象注入,

@Configuration
public class UsePolice {
    @Autowired
    Police police;
    public void getpolice()
    {

        police.say();
    }

}

然后我们拿到UsePolice这个类型的对象 

 public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        UsePolice usePolice=context.getBean("usePolice",UsePolice.class);
        usePolice.getpolice();
    }

属性注入的优点:简单方便      弊端:(1)不符合单一设计的原则,我们可以将多个类型的对象共同注入(2)不能注入被final修饰的对象

因为final修饰的变量要么在后面直接赋值,要么利用构造方法赋值。

(3) 只适用于Ioc容器,不具备通用性

2.Setter注入

@Configuration
public class UsePolice2 {
     public Police police;
     @Autowired
     public void setPolice(Police police) {
        this.police = police;
    }

    public void getpolice()
    {
        police.say();

    }


}

Setter方法注入的优点:(1)符合单一设计的原则    缺点:(1)不能注入被final修饰的对象

(2)使用Setter注入的对象可能被修改   比如我们在getPolice()方法里将police置为null

@Configuration
public class UsePolice2 {
     public Police police;
     @Autowired
     public void setPolice(Police police) {
        this.police = police;
    }

    public void getpolice()
    {
        this.police=null;
        police.say();

    }
}



3.构造方法注入

@Component
public class UsePolice3 {
    public Police police;
    @Autowired
    public UsePolice3(Police police)
    {
        this.police=police;
    }
    public void getpolice()
    {
        police.say();
    }

}

构造方法注入的特点:如果当前类里面只有一个构造方法的话,那么@Autowired可以被省略

构造方法注入优点分析:1.能注入被final修饰的对象

2.构造方法只执行一次,注入对象不会再被修改

3.构造方法注入不止适用于Ioc容器,具有通用性。 

4.构造方法注入会将注入的对象完全初始化

缺点:构造方法里可以传多个参数,可能会违背单一设计原则。

与@Autowired注解作用差不多的还有@Resource注解

@Resource同样可以属性注入和Setter注入,但是如果是构造方法注入就会报错

 接下来我们来思考一个问题:如果有两个Police对象,这时注入的是谁呢?我们启动一下

@Component
public class Useuser4 {
    @Bean(value="police1")
    public Police getpolice()
    {
        Police police=new Police();
        police.name="小海";
        police.age=9;
        return police;

    }
    @Bean
    public Police getpolice2()
    {
        Police police =new Police();
        police.name="小小";
        police.age=9;
        return police;
    }
}

Component
public class Usepolice {
    @Resource
    public Police police;

    public void say()
    {
       police.say();
       System.out.println(police.name);
    }

}
 public static void main(String[] args) {
        ApplicationContext context=new ClassPathXmlApplicationContext("spring-conflig.xml");
        Usepolice useuser4=context.getBean("usepolice",Usepolice.class);
        useuser4.say();
    }

打印结果:

 我们发现会报错,这是因为Resource先根据名称去查,然后再根据类型去查,

这里的名称也就是police ,我们发现注册的两个Police实例都不是这个名称,然后再根据Police类型去找,发现两个Police实例也不是以Police为名称的

这个问题的解决办法就是在Resource后面加上具体的Bean对象的名称

 @Resource(name="police1")
    public Police police;

这时候就指定是注入police1这个对象了

 那使用@Autowired能不能指定指定的对象呢?其实也是可以的

 @Autowired
    @Qualifier(value="police1")

总结:@Resource注解和@Autowired的区别

(1)@Autowired来自Spring  @Resource来自jdk的注解

  (2)@Autowired 支持属性注入、Setter注入、和构造方法注入,@Resource不支持构造方法注入

 (3)相较于@Autowired@Resource可以设置更多的参数可以在后面指定name,依次来获取Bean

 Bean的作用域

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

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

相关文章

Java中的容器大杂烩-集合

Java中的容器大杂烩-集合 一、 集合引入二、集合框架体系三 、Collection 接口四 、List集合4.1 ArrayList类4.2 LinkedList类4.3 Vector类4.4 ArrayList 、 LinkedList 和 Vector区别 五 、Set集合5.1 HashSet类5.2 TreeSet5.3 LinkedHashSet类 六、List和Set区别七 、Map集合…

制造业巨头遭黑客勒索400万美元,企业如何防范勒索病毒攻击?

4月7日&#xff0c;台湾电脑制造商微星&#xff08;简称MSI&#xff09;发布公开声明&#xff0c;证实其部分网络信息系统遭受了勒索病毒攻击。一个名为“Money Message”的新网络黑客团伙称其从微星的网络系统中窃取了1.5TB的数据&#xff0c;其中包括微星数据库的屏幕截图、源…

YAML /Excel /CSV?自动化测试测试数据管理应用,测试老鸟总结...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 自动化测试无论是…

webhub123 前端技术社区和技术交流学习网站导航

整理了学习前端技术可以参考学习和技术交流的一些网站集合&#xff0c;全部收录到 webhub123 前端技术社区和技术交流学习网站导航http://​www.webhub123.com/#/home/detail?projectHashid30929575&ownerUserid22053727 整理后的效果如下&#xff0c;我们已经按照不同类…

React Props

state 和 props 主要的区别在于 props 是不可变的&#xff0c;而 state 可以根据与用户交互来改变。 所以&#xff0c;有些容器组件需要定义 state 来更新和修改数据。 而子组件只能通过 props 来传递数据。 props 使用 Demo.js &#xff1a; import React from reactfunct…

智能学习 | MATLAB实现ACO-BP多变量时间序列预测(蚁群算法优化BP神经网络)

智能学习 | MATLAB实现ACO-BP多变量时间序列预测(蚁群算法优化BP神经网络) 目录 智能学习 | MATLAB实现ACO-BP多变量时间序列预测(蚁群算法优化BP神经网络)预测效果基本介绍程序设计参考资料预测效果 基本介绍 MATLAB实现ACO-BP多变量时间序列预测(蚁群算法优化BP神经网络…

数据集合注入

集合注入 前面我们已经能完成引入数据类型和简单数据类型的注入&#xff0c;但是还有一种数据类型集合&#xff0c;集合中既可 以装简单数据类型也可以装引用数据类型&#xff0c;对于集合&#xff0c;在Spring中该如何注入呢? 先来回顾下&#xff0c;常见的集合类型有哪些…

Vue电商项目--应用开发详解

vue-cli脚手架初始化项目 首先&#xff0c;页面上新建一个文件夹。然后打开命令端口 vue create app 选择Default ([Vue 2] babel, eslint) 然后把项目拖拽到vscode中。项目目录看一下 脚手架项目的目录 node_modules:放置项目依赖的地方 public:一般放置一些共用的静态资源&a…

数据采集方式有哪些,都有什么特点?

随着中国社会的进一步发展&#xff0c;各行各业都得到了一定程度的进步。进入21世纪以来&#xff0c;大数据、人工智能等行业的飞速发展&#xff0c;极大的带动全社会进步。但是&#xff0c;在一些传统行业内部&#xff0c;还存在这落后的东西&#xff0c;例如数据采集还是沿用…

【机器学习】P24 随机森林算法(1) 实现 “鸢尾花” 预测

随机森林算法 Random Forest Algorithm 随机森林算法随机森林算法实现分类鸢尾花 随机森林算法 随机森林&#xff08;Random Forest&#xff09;算法 是一种 集成学习&#xff08;Ensemble Learning&#xff09;方法&#xff0c;它由多个决策树组成&#xff0c;是一种分类、回…

OS实战笔记(8)-- 设置基本OS基本工作环境

本笔记会搭建OS实战所需的虚拟机环境&#xff0c;主要是创建好虚拟机&#xff0c;设置虚拟机启动硬盘&#xff0c;在启动盘上安装Grub。 由于本专题是个人在业余时间除了Unity学习之外做的&#xff0c;没有时间和精力去解答具体的问题。本笔记中的实验个人在做的过程中除了遇到…

17.集合

集合 集合类是Java数据结构的实现。Java的集合类是java.util包中的重要内容&#xff0c;它允许以各种方式将元素分组&#xff0c;并定义了各种使这些元素更容易操作的方法。Java集合类是Java将一些基本的和使用频率极高的基础类进行封装和增强后再以一个类的形式提供。集合类是…

整理现有的wiki私服项目

文章目录 核心功能现有项目wikijsBookStackmediawikiTiddlyWikigollumdokuwikixwiki 总结参考 核心功能 查找编辑 在线/离线内链【核心】代码高亮图表、表达式生成多媒体&#xff08;图片、音频、视频&#xff09;管理 协作&#xff08;用户管理模式/github模式&#xff09; 修…

JVM 关键点详解

一&#xff0c;JVM 的主要组成部分及其作用 JVM包含两个子系统和两个组件&#xff0c;两个子系统为Class loader(类装载)、Execution engine(执行引擎); 两个组件为Runtime data area(运行时数据区)、Native Interface(本地接口)。 Class loader(类装载): 根据给定的全限定名类…

【Linux网络】部署YUM仓库及NFS服务

部署YUM仓库及NSF服务 一、YUM仓库1.1、YUM仓库概述1.2准备安装来源1.3在软件仓库加载非官方RPM包组1.4yum与apt 二、配置yam源与制作索引表2.1配置FTP源2.2配置国内在线yum源2.3在线源与本地源同时使用2.4建立软件包索引关系表的三种方法 三、nfs共享存储服务3.1安装软件&…

LVS负载均衡群集——NAT模式实操

1.1 群集的的定义及意义 群集的定义 Cluster&#xff0c;集群&#xff08;也称群集&#xff09;由多台主机构成&#xff0c;但对外只表现为一一个整体&#xff0c;只提供一-个访问入口(域名或IP地址)&#xff0c; 相当于一台大型计算机。 群集的作用 对于企业服务的的性能提升…

数学知识四

容斥原理 S表示面积&#xff0c;下面公式可求出不相交的面积 2个圆的公式是这样 4个圆的面积是 总面积-所有俩俩相交的面积所有三三相交的面积-四四相交的面积&#xff0c;公式里加和减互相出现。 从n个集合里面挑一个一直到从n个集合里面挑n个 1-10中&#xff0c;能被2&#x…

【KingSCADA】如何创建新应用

大家好&#xff0c;我是雷工! 今天学习使用KingSCADA3.8创建一个新的应用&#xff0c;以下为学习过程和操作笔记。 一、前言 KingSCADA3.8集成开发环境是基于工程的应用管理模式&#xff0c;实现了对多个应用的集中开发和管理的功能&#xff0c;一个工程可以同时管理多个应用…

【WinForm】Android手机群控工具-桌面程序开发实现

如何将手下多个Android手机统一管理起来呢&#xff0c;这里是用通过终端输入adb命令来实现控制多个手机的&#xff0c;具体怎么做&#xff0c;接下来给讲一讲。 使用adb工具包 首先&#xff0c;需要准备一套工具&#xff0c;以下是adb工具套件&#xff0c;是在Android SDK开发…

lanuage-driven semantic segmentation

CLIP 改进工作串讲&#xff08;上&#xff09;【论文精读42】_哔哩哔哩_bilibili更多论文&#xff1a;https://github.com/mli/paper-reading, 视频播放量 64310、弹幕量 274、点赞数 1939、投硬币枚数 1332、收藏人数 821、转发人数 438, 视频作者 跟李沐学AI, 作者简介 &…