👑 博主简介:
🥇 Java领域新星创作者
🥇 阿里云开发者社区专家博主、星级博主、技术博主
🤝 交流社区:BoBooY(优质编程学习笔记社区)
文章目录
Bean的基础配置
1、id 与 class属性
<beanid=""class=""/><!--举例--><beanid="bookDao"class="com.bby.dao.bookDaoImpl"/><beanid="bookDao"class="com.bby.dao.bookDaoImpl"></bean>
- id: bean的id,使用容器可以通过id值获取对应的bean,在一个容器中id值唯一
- class: bean的类型,即配置的bean的全路径类名
注意:
- class属性不能写接口如
BookDao
的类全名(因为接口是没办法创建对象的) - id必须唯一,如果因为命名习惯产生分歧解决办法是设置name属性,也就是给Bean取别名
2、name属性
定义bean的别名,可定义多个,使用**逗号(,) 分号(😉 空格()**分隔
<beanid="bookDao"name="dao bookDaoImpl"class="com.bby.dao.bookDaoImpl"/><beanid="bookDao"name="dao,bookDaoImpl"class="com.bby.dao.bookDaoImpl"/><beanid="bookDao"name="dao;bookDaoImpl"class="com.bby.dao.bookDaoImpl"/>
- 测试
- 这里我设置了两个别名
dao
和bookDaoImpl
- 获取这个Bean并调用它的方法
/**
* @author BoBooY
*/publicclassApp{publicstaticvoidmain(String[] args){ApplicationContext applicationContext =newClassPathXmlApplicationContext("classpath:applicationContext.xml");BookDao dao =(BookDao)applicationContext.getBean("dao");BookDao bookDaoImpl =(BookDao)applicationContext.getBean("bookDaoImpl");
dao.save();
bookDaoImpl.save();}}
- 查看控制台结果
可以看到通过这个两个别名都可以获取到这个Bean
注意:
- bean依赖注入的 ref 属性指定bean,必须在容器中存在,ref的属性值,也可也是另一个bean的name属性值,不过还是建议使用其id来进行注入
- 如果id不存在,在获取时会获取不到,然后就会报错:NoSuchBeanDefinitionException
3、scope属性
- singleton:单例(默认)
- prototype:非单例
<beanid="bookDao"class="com.bby.dao.bookDaoImpl"scope="prototype"/>
分别获取单例模式和多例模式下的bean
- 单例模式如下(singleton):
- 非单例模式如下(prototype):
思考:
- 为什么bean默认为单例?- bean为单例的意思是在Spring的IOC容器中只会有该类的一个对象- bean对象只有一个就避免了对象的频繁创建与销毁,达到了bean对象的复用,性能高
- bean在容器中是单例的,会不会产生线程安全问题?- 如果对象是有状态对象,即该对象有成员变量可以用来存储数据的,- 因为所有请求线程共用一个bean对象,所以会存在线程安全问题。- 如果对象是无状态对象,即该对象没有成员变量没有进行数据存储的,- 因方法中的局部变量在方法调用完成后会被销毁,所以不会存在线程安全问题。
- 哪些bean对象适合交给容器进行管理?- 表现层对象- 业务层对象- 数据层对象- 工具对象
- 哪些bean对象不适合交给容器进行管理?- 封装实例的域对象,因为会引发线程安全问题,所以不适合。
Bean的实例化
bean本质上就是对象,对象在new的时候会使用构造方法完成,那创建bean也是使用构造方法完成的。
1、构造方法实例化
这里先说结论:Spring底层通过反射访问的类的无参构造方法来实例化Bean
验证:
- 为
bookDaoImpl
添加一个无参构造方法,并打印一句话,方便观察结果。
/**
* @author BoBooY
*/publicclassBookDaoImplimplementsBookDao{publicBookDaoImpl(){System.out.println("book dao constructor is running ....");}publicvoidsave(){System.out.println("book dao save ...");}}
查看控制台结果:
从这里可以看出:Spring容器在创建对象的时候也走的是构造函数
- 将构造函数改成private测试
/**
* @author BoBooY
*/publicclassBookDaoImplimplementsBookDao{privateBookDaoImpl(){System.out.println("book dao constructor is running ....");}publicvoidsave(){System.out.println("book dao save ...");}}
查看控制台结果:
从这里可以看出:Spring内部走的依然是构造函数,能访问到类中的私有构造方法,显而易见Spring底层用的是反射
- 使用带参构造方法测试
/**
* @author BoBooY
*/publicclassBookDaoImplimplementsBookDao{privateBookDaoImpl(int a){System.out.println("book dao constructor is running ....");}publicvoidsave(){System.out.println("book dao save ...");}}
查看控制台结果:
程序会报错,说明Spring底层使用的是类的无参构造方法。
Caused by: org.springframework.beans.BeanInstantiationException: Failed to instantiate [com.bby.dao.BookDaoImpl]: No default constructor found; nested exception is java.lang.NoSuchMethodException: com.bby.dao.BookDaoImpl.()
译:
<init>
()指定是类的构造方法,即该类的无参构造方法
- Caused by:
引发
- BeanInstantiationException:翻译为
bean实例化异常
- No default constructor found:没有一个默认的构造函数被发现
2、静态工厂实例化
- 准备一个OrderDao和OrderDaoImpl类
publicinterfaceOrderDao{publicvoidsave();}``````publicclassOrderDaoImplimplementsOrderDao{publicvoidsave(){System.out.println("order dao save ...");}}
- 创建一个工厂类OrderDaoFactory并提供一个静态方法
//工厂类publicclassOrderDaoFactory{publicstaticOrderDaogetOrderDao(){returnnewOrderDaoImpl();}}
- 在spring的配置文件applicationContext.xml中添加以下内容
<bean id="orderDao"class="com.bby.factory.OrderDaoFactory" factory-method="getOrderDao"/>
- class:工厂类的类全名- factory-mehod:具体工厂类中创建对象的方法名 - 获取Bean测试
publicclassApp{publicstaticvoidmain(String[] args){ApplicationContext ctx =newClassPathXmlApplicationContext("applicationContext.xml");OrderDao orderDao =(OrderDao) ctx.getBean("orderDao"); orderDao.save();}}
3、实例工厂实例化
- 创建一个工厂类OrderDaoFactory2并提供一个普通方法,注意此处和静态工厂的工厂类不一样的地方是方法不是静态方法
//实例工厂publicclassOrderDaoFactory2{publicOrderDaogetOrderDao(){returnnewOrderDaoImplava}}
- 在spring的配置文件中添加以下内容
<beanid="orderDaoFactory2"class="com.bby.factory.OrderDaoFactory2"/><beanid="orderDao"factory-method="getOrderDao"factory-bean="orderDaoFactory2"/>
- factory-bean:工厂的实例对象- factory-method:工厂对象中的具体创建对象的方法名 - 获取Bean进行测试
/** * @author BoBooY */publicclassApp{publicstaticvoidmain(String[] args){ApplicationContext ctx =newClassPathXmlApplicationContext("applicationContext.xml");OrderDao orderDao =(OrderDao) ctx.getBean("orderDao2"); orderDao.save();}}
实例化工厂运行的顺序是:
- 创建实例化工厂对象,对应的是第一行配置
- 调用对象中的方法来创建bean
🚩 FactoryBean的使用
实例工厂实例化的方式配置的过程还是比较复杂,所以Spring为了简化这种配置方式就提供了一种叫
FactoryBean
的方式来简化开发。
- 创建一个OrderDaoFactoryBean的类,实现FactoryBean接口,重写接口的方法
/** * @author BoBooY */publicclassOrderDaoFactoryBeanimplementsFactoryBean<OrderDao>{//代替原始实例工厂中创建对象的方法publicOrderDaogetObject()throwsException{returnnewOrderDaoImpl();}//返回所创建类的Class对象publicClass<?>getObjectType(){returnOrderDao.class;}}
- 在Spring的配置文件中进行配置
<beanid="orderDao"class="com.bby.factory.OrderDaoFactoryBean"/>
- 获取Bean进行测试
查看源码会发现,FactoryBean接口其实会有三个方法,分别是:
TgetObject()throwsException;Class<?>getObjectType();defaultbooleanisSingleton(){returntrue;}
- **getObject()**:被重写后,在方法中进行对象的创建并返回
- **getObjectType()**:被重写后,主要返回的是被创建类的Class对象
- isSingleton:没有被重写,因为它已经给了默认值,从方法名中可以看出其作用是设置对象是否为单例,默认true
尾言:创作不易,如果本文的内容对您有帮助,还望客官可以支持一下博主,👍(点赞)+✏️(评论)+⭐️(收藏)是我创作的巨大动力!
版权归原作者 -BoBooY- 所有, 如有侵权,请联系我们删除。