栏目分类:
子分类:
返回
文库吧用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
文库吧 > IT > 软件开发 > 后端开发 > Java

Spring-Bean的生命周期

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

Spring-Bean的生命周期

目录

一、Bean的初始化过程

Spring Bean的生命周期

总结

二、Bean的单例与例模式

如何区分单例与多例

概述

案例

多列

单例

 ​编辑

 Spring单例模式的弊端

体现单例与多例的区别

 总结:


一、Bean的初始化过程

Spring Bean的生命周期

1)通过XML、Java annotation(注解)以及Java Configuration(配置类)

等方式加载Spring Bean

2)BeanDefinitionReader:解析Bean的定义。在Spring容器启动过程中,

会将Bean解析成Spring内部的BeanDefinition结构;

理解为:将spring.xml中的标签转换成BeanDefinition结构

有点类似于XML解析

3)BeanDefinition:包含了很多属性和方法。例如:id、class(类名)、

scope、ref(依赖的bean)等等。其实就是将bean(例如)的定义信息

存储到这个对应BeanDefinition相应的属性中

例如:

 -----> BeanDefinition(id/class/scope)

4) :是Spring容器功能的扩展接口。

注意:

①.BeanFactoryPostProcessor在spring容器加载完BeanDefinition之后,

在bean实例化之前执行的

②.对bean元数据(BeanDefinition)进行加工处理,也就是BeanDefinition

属性填充、修改等操作

案例:

package com.oyang.beanLife;

public class Demo1 {
	public static void main(String[] args) {
		Person p=new Person();
		System.out.println(p.getName());
	}
}

class Person{
	private String name;
	private int age;
	private String sex;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
	public String getSex() {
		return sex;
	}
	public void setSex(String sex) {
		this.sex = sex;
	}
	public Person() {
		this.init();
		this.name="oy";
		this.age=19;
		this.sex="男";
	}
	private void init() {
		System.out.println("初始化方法");
		
	}
	public Person(String name, int age, String sex) {
		super();
		this.name = name;
		this.age = age;
		this.sex = sex;
	}
	@Override
	public String toString() {
		return "Person [name=" + name + ", age=" + age + ", sex=" + sex + "]";
	}
	
	
	
}

运行效果:

5)BeanFactory:bean工厂。它按照我们的要求生产我们需要的各种各样的bean。

例如:

BeanFactory -> List
BeanDefinition(id/class/scope/init-method)

foreach(BeanDefinition bean : List){
   //根据class属性反射机制实例化对象
   //反射赋值设置属性
}

6)Aware感知接口:在实际开发中,经常需要用到Spring容器本身的功能资源

例如:BeanNameAware、ApplicationContextAware等等

BeanDefinition 实现了 BeanNameAware、ApplicationContextAware----如图所示:

 Aware感知接口帮助你拿到Spring内置对象来使用

7)BeanPostProcessor:后置处理器。在Bean对象实例化和引入注入完毕后,

在显示调用初始化方法的前后添加自定义的逻辑。(类似于AOP的绕环通知)

前提条件:如果检测到Bean对象实现了BeanPostProcessor后置处理器才会执行

Before和After方法

修改他原有的属性值。

【 BeanPostProcessor

1)Before

2)调用初始化Bean(InitializingBean和init-method,Bean的初始化才算完成)

3)After

完成了Bean的创建工作 】

8)destory:销毁

总结

1.通过三种方式(配置文件、注解、配置内)将bean标签转成beandifinition对象
2.通过BeanFactoryPostProcessor可以在初始化之前修改属性值
3.BeanFactory进行bean实例化,就是生产Javabean
4.Aware感知接口,能够拿到Spring上下文中内部的资源对象
5.BeanPostProcessor后置处理器,相当于环绕通知(在Spring容器初始化之后,针对它再做进一步的改变)

二、Bean的单例与例模式

如何区分单例与多例

scope="singleton" --单例        如果不写,默认就是单例

scope="prototype"--原型        23种设计模式中的原型模式

概述

一个类如果在不同的地方使用了100次,如果这个项目有1W的类,那Spring上下文要创建100W个对象。那么每一个对象都会消耗我的的服务器资源的,就会显得比较臃肿,使用Spring不推荐使用多例,就会采用单例模式。

案例

多列
package com.oyang.beanLife;
public class Demo1 {
	public static void main(String[] args) {
		Person p1=new Person();
		Person p2=new Person();
		Person p3=new Person();
		Person p4=new Person();
		System.out.println(p1);
		System.out.println(p2);
		System.out.println(p3);
		System.out.println(p4);
	}
}

class Person{
	private String name;
	private int age;
	private String sex;
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}
	public String getSex() {
		return sex;
	}
	public void setSex(String sex) {
		this.sex = sex;
	}
	
public Person() {
	// TODO Auto-generated constructor stub
}	
}

单例
package com.oyang.beanLife;

public class Demo1 {
	public static void main(String[] args) {
		Person p1=Person.newInstance();
		Person p2=Person.newInstance();
		Person p3=Person.newInstance();
		Person p4=Person.newInstance();
		System.out.println(p1);
		System.out.println(p2);
		System.out.println(p3);
		System.out.println(p4);
	}
}

class Person{
	private Person() {
		
	}
	
	private final static Person p=new Person();
	
	public static Person newInstance() {
		return p;
	}
}

 

 Spring单例模式的弊端

体现单例与多例的区别

 InstanceFactory

package com.oyang.beanLife;

public class InstanceFactory {
	public void init() {
		System.out.println("初始化方法");
	}

	public void destroy() {
		System.out.println("销毁方法");
	}

	public void service() {
		System.out.println("业务方法");
	}
}

ParamAction 

package com.oyang.beanLife;

import java.util.List;


public class ParamAction {
	private int age;
	private String name;
	private List hobby;
	private int num = 1;
	// private UserBiz userBiz = new UserBizImpl1();

	public ParamAction() {
		super();
	} 

	public ParamAction(int age, String name, List hobby) {
		super();
		this.age = age;
		this.name = name;
		this.hobby = hobby;
	}

	public void execute() {
		// userBiz.upload();
		// userBiz = new UserBizImpl2();
		System.out.println("this.num=" + this.num++);
		System.out.println(this.name);
		System.out.println(this.age);
		System.out.println(this.hobby);
	}
}

spring-context.xml


		
		
		
			
				抽烟
				烫头
				大保健
			
		
	

	

 Demo2

package com.oyang.beanLife;

import org.junit.Test;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;


public class Demo2 {
	// 体现单例与多例的区别
	@Test
	public void test1() {
		ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
//		ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
		ParamAction p1 = (ParamAction) applicationContext.getBean("paramAction");
		ParamAction p2 = (ParamAction) applicationContext.getBean("paramAction");
		// System.out.println(p1==p2);
		p1.execute();
		p2.execute();
		
//		单例时,容器销毁instanceFactory对象也销毁;多例时,容器销毁对象不一定销毁;
		applicationContext.close();
	}

	// 体现单例与多例的初始化的时间点 instanceFactory
	@Test
	public void test2() {
		ApplicationContext applicationContext = new ClassPathXmlApplicationContext("/spring-context.xml");
	}

	// BeanFactory会初始化bean对象,但会根据不同的实现子类采取不同的初始化方式
	// 默认情况下bean的初始化,单例模式立马会执行,但是此时XmlBeanFactory作为子类,单例模式下容器创建,bean依赖没有初始化,只有要获取使用bean对象才进行初始化
	@Test
	public void test3() {
		// ClassPathXmlApplicationContext applicationContext = new
		// ClassPathXmlApplicationContext("/spring-context.xml");

		Resource resource = new ClassPathResource("/spring-context.xml");
		BeanFactory beanFactory = new XmlBeanFactory(resource);
//		InstanceFactory i1 = (InstanceFactory) beanFactory.getBean("instanceFactory");
		
	}

}

test1

容器默认是单例的

 修改一下配置,改成多例模式

单例时,容器销毁instanceFactory对象也销毁;多例时,容器销毁对象不一定销毁;

test3

 总结:

单例模式下Javabean的生命周期
    容器生对象生,容器死对象死
多例模式下Javabean的生命周期
    使用时对象生,死亡跟着jvm垃圾回收机制走


 OK,今日的学习就到此结束啦,如果对个位看官有帮助的话可以留下免费的赞哦(收藏或关注也行),如果文章中有什么问题或不足以及需要改正的地方可以私信博主,博主会做出改正的。个位看官,小陽在此跟大家说拜拜啦  

转载请注明:文章转载自 www.wk8.com.cn
本文地址:https://www.wk8.com.cn/it/1039058.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 wk8.com.cn

ICP备案号:晋ICP备2021003244-6号