方法注入不同于set注入和构造注入.
主要是使用场景不同:一个singleton的Bean需要引用一个prototype的Bean; 一个无状态的Bean需要引用一个有状态的Bean; ... ; 等等情景下.
试想,我们的容器创建之后就加载了所有的Bean. 而BeanA中需要引用的BeanB是状态个不确定的Bean. 那么我们需要在每次需要BeanB的时候都重新让容器加载一次吗? 真是个笨拙的想法.
好在我们有cglib.
cglib为我们动态的构造BeanB的子类, 当我们的BeanA需要BeanB的时候, cglib把这个子类对象给BeanA.
看代码说话:
/**
* 我们把这个类作为BeanB.
* (Hp就是血量... 游戏中某个英雄的血量总在变化)
* (比如说我们希望通过英雄名找到的英雄的血是实时变化的. )
*/
public class HpDaoImpl implements HpDao
{
@Override
public int getHp()
{
// 返回0~100之间的随即数
return (int)(Math.random() * 100);
}
}
这是beanB的定义. 注意有个scope="prototype". 再请注意是全部小写的prototype.
<bean id="hpDao" class="org.mycompany.spring.aop.dao.impl.HpDaoImpl" scope="prototype" />
// -----------------------------------------------------------------------------
接下来是BeanA.
/**
* 我们把这个类作为BeanA.
* 这是一个抽象类. 为什么要抽象? 因为有个抽象方法..
* 既然抽象了, 就不能final... 切记切记
*/
public abstract class HpServiceImpl implements HpService
{
// protected修饰
protected HpDao hpDao = getHpDao();
/*
* 这个方法是抽象的.
* 返回的是cglib构造的BeanB的子类.
*/
public abstract HpDao getHpDao();
@Override
public int getHp()
{
// 调用原型BeanB的方法
return hpDao.getHp();
}
}
再看看BeanA的定义:
<bean id="hpService" class="org.mycompany.spring.aop.service.impl.HpServiceImpl">
<lookup-method name="getHpDao" bean="hpDao" />
</bean>
解释:
lookup-method中name是BeanA中的抽象方法的名字.这里这个方法的用途是得到由cglib构造的BeanB的动态子类.
lookup-mentod中bean是BeanB的类型.也就是cglib构造的BeanB的动态子类的父类型...目的当然是父类的句柄可以引用子类:用这个父类型引用构造出来的动态子类.
// -----------------------------------------------------------------------------
讲解完毕.
实验一下:
public static void main(String[] args)
{
ApplicationContext context = new ClassPathXmlApplicationContext("classpath*:spring/application-*.xml");
// print: false
if(context.getBean("hpDao") == context.getBean("hpDao"))
{
System.out.println(true);
}else
{
System.out.println(false);
}
// print: true
if(context.getBean("hpService") == context.getBean("hpService"))
{
System.out.println(true);
}else
{
System.out.println(false);
}
}
// -----------------------------------------------------------------------------
额外注意一下:
如果BeanB没有写:scope="prototype" ...
那么会出异常并提示你说:
No Scope registered for scope '
Prototype'
如果你添加scope='Prototype'...那还见鬼咧.还是这个错.
所以我说了嘛...一定是
小写的prototype
spring出的这个异常太蛊祸人了. 在这里我提出批评.
分享到:
相关推荐
RevitLookup-2021.0.0.13,供各位做REVIT二次开发的同仁们下载、学习、研究。
revit开发必备工具 本包中是已经编译过的文件,可以直接使用 ...编辑其中RevitLookup.dll 路径为编译路径 C:\RevitLookup-2016.0.0.6\CS\bin\Debug\RevitLookup.dll 重启revit: 即可在菜单栏看到工具
RevitLookup-2017.0.0.5RevitLookup-2017.0.0.5RevitLookup-2017.0.0.5RevitLookup-2017.0.0.5RevitLookup-2017.0.0.5
“Lookup方法”可以使Spring替换一个bean原有的,获取其它对象具体的方法,并自动返回在容器中的查找结果。下面这篇文章主要给大家介绍了关于Spring中Lookup(方法注入)的相关资料,需要的朋友可以参考下
RevitLookup - 2016 的 源码,需要自己编译 用法:将RevitLookup.addin和RevitLookup.dll放到C:\ProgramData\Autodesk\Revit\Addins\2016下即可
RevitLookup - 2017 的 源码,需要自己编译 用法:将RevitLookup.addin和RevitLookup.dll放到C:\ProgramData\Autodesk\Revit\Addins\2017下即可
前端开源库-nsq-lookup-jcnsq lookup jc,通过nsqlookupd查找nsqd实例
资源分类:Python库 所属语言:Python 使用前提:需要解压 资源全名:iso3_lookup-0.3.0-py3-none-any.whl 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059
RevitLookup-2020.0.0.4 官方公布源代码需要自己编译,方便学习REVIT内核构架和开发
lookup-dns- dns.lookup替换dns.lookup标准方法的DNS缓存超级简单易用const request = require ( 'request' ) ;const { lookup } = require ( 'lookup-dns-cache' ) ;// With "request" modulerequest ( { url : '...
RevitLookup - 2018 的 源码,需要自己编译 用法:将RevitLookup.addin和RevitLookup.dll放到C:\ProgramData\Autodesk\Revit\Addins\2018下即可
python库,解压后可用。 资源全名:databricks_feature_lookup-0.0.5-py3-none-any.whl
资源来自pypi官网。 资源全名:databricks_feature_lookup-0.0.5-py3-none-any.whl
资源来自pypi官网。 资源全名:digdeo-syspass-ansible-lookup-0.2.4.tar.gz
资源来自pypi官网。 资源全名:drf-multi-lookup-0.0.14.tar.gz
python库,解压后可用。 资源全名:spacy_lookup-0.0.5-py2.py3-none-any.whl
资源来自pypi官网。 资源全名:spacy_lookup-0.0.5-py2.py3-none-any.whl
资源来自pypi官网。 资源全名:drf-multi-lookup-0.0.19.tar.gz
资源分类:Python库 所属语言:Python 资源全名:digdeo-syspass-ansible-lookup-0.1.1.tar.gz 资源来源:官方 安装方法:https://lanzao.blog.csdn.net/article/details/101784059