技术标签: spring 【Spring-IOC】 注入参数
在Spring配置文件中,不但可以将String、int等字面值注入bean中,还可以将集合、map等类型的数据注入Bean中, 此外还可以注入配置文件中其他定义的Bean.
所谓字面值一般指的是可以用字符串表示的值,这些值可以通过<value>
元素标签进行注入。
在默认情况下,基本数据类型及其封装类、String等类型都可以采用字面值注入的方式。
比如前几篇博文中经常使用的案例:
<bean id="plane" class="com.xgj.ioc.inject.set.Plane">
<property name="brand">
<value><![CDATA[Airbus&A380]]></value>
<!-- 或者使用转义序列
<value>Airbus&A380</value>
-->
</property>
<property name="color">
<value>red</value>
</property>
<property name="speed">
<value>700</value>
</property>
</bean>
XML共有5个特殊符号 & < > " '
<![CDATA[ ]]>
特殊标签,阻止XML解析,让其当做普通的文本处理特殊符号 | 转移序列 |
---|---|
< | < |
> | > |
& | & |
“ | " |
‘ | ' |
一般情况下,xml解析器会忽略元素标签内部字符串的前后空格,但是Spring不会忽略空格。
举个例子:
<property name="brand">
<value> Airbus&A380 </value>
</property>
运行结果如下:
Spring IoC容器中定义的Bean可以相互引用,IoC容器充当“红娘”的角色。
警察类中有枪类型的属性
POJO类
package com.xgj.ioc.inject.construct.ref;
public class Police {
private Gun gun;
public void setGun(Gun gun) {
this.gun = gun;
}
public void userGun() {
gun.fire();
}
}
POJO类
package com.xgj.ioc.inject.construct.ref;
public class Gun {
public void fire() {
System.out.println("Gun Fires");
}
}
配置文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="police" class="com.xgj.ioc.inject.construct.ref.Police">
<property name="gun">
<!-- 通过ref应用容器中的gun ,建立police和gun的依赖关系 -->
<ref bean="gun" />
</property>
</bean>
<bean id="gun" class="com.xgj.ioc.inject.construct.ref.Gun" />
</beans>
测试类
package com.xgj.ioc.inject.construct.ref;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class RefTest {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("classpath:com/xgj/ioc/inject/construct/ref/police-gun.xml");
Police police = ctx.getBean("police", Police.class);
police.userGun();
}
}
运行结果
bean:通过该属性可以应用同一容器或者父容器中的bean,这是最常见的形式
local:通过该属性只能引用同一个配置文件中定义的Bean,它可以利用XML解析器自动检查引用的合法性,以便开发人员在编写配置文件时及时发现错误
parent:引用父容器中的Bean,如<ref parent="gun">
的配置说明gun的Bean是父容器中的Bean
演示子容器对父容器Bean的引用。
其中 beans_father.xml被父容器加载,beans_son.xml被子容器加载。
POJO类
package com.xgj.ioc.inject.construct.refParentBean;
public class Gun {
private String brand;
private int bulletNum;
private double price;
public void setBrand(String brand) {
this.brand = brand;
}
public void setBulletNum(int bulletNum) {
this.bulletNum = bulletNum;
}
public void setPrice(double price) {
this.price = price;
}
public void gunInfo() {
System.out.println("Gun Information brand:" + brand + ",bulletNum:"
+ bulletNum + ",price:" + price);
}
}
POJO类
package com.xgj.ioc.inject.construct.refParentBean;
public class Police {
private Gun gun;
public void setGun(Gun gun) {
this.gun = gun;
}
public void introduceGun() {
gun.gunInfo();
}
}
beans_father.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="gun" class="com.xgj.ioc.inject.construct.refParentBean.Gun">
<property name="brand" value="父容器--伯莱塔92F" />
<property name="bulletNum" value="11" />
<property name="price" value="2888" />
</bean>
</beans>
beans_son.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="gun" class="com.xgj.ioc.inject.construct.refParentBean.Gun">
<property name="brand" value="柯尔特M2000" />
<property name="bulletNum" value="7" />
<property name="price" value="999" />
</bean>
<bean id="police" class="com.xgj.ioc.inject.construct.refParentBean.Police">
<property name="gun">
<!-- 通过parent属性,引用父容器中的gun,如果定义为bean="gun"则引用本容器中的gun -->
<ref parent="gun"/>
</property>
</bean>
</beans>
测试类
package com.xgj.ioc.inject.construct.refParentBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class RefParentBeanTest {
public static void main(String[] args) {
// 父容器
ApplicationContext pFather = new ClassPathXmlApplicationContext(
"classpath:com/xgj/ioc/inject/construct/refParentBean/beans_father.xml");
// 指定pFather为该容器的父容器
ApplicationContext ctx = new ClassPathXmlApplicationContext(
new String[] { "com/xgj/ioc/inject/construct/refParentBean/beans_son.xml" },
pFather);
Police police = ctx.getBean("police", Police.class);
// 观察是否输出为父容器设置的属性
police.introduceGun();
}
}
运行结果
如果某个bean只会被其中一个引用,而不会被容器中的其它Bean引用,可以以内部Bean的方式注入。
内部bean和Java匿名内部类相似,既没有名字,也不会被前台bean引用,只能在声明处为外部Bean提供实例注入。
内部Bean即使提供了id、name、scope等属性,也会被忽略,scope默认prototype类型
举例:
其他类同上,区别仅在于beans的配置文件。
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="police" class="com.xgj.ioc.inject.construct.innerBean.Police">
<property name="gun">
<!-- 如果gun bean只会被police引用,而不会被容器中的其它Bean引用,可以以内部Bean的方式注入 -->
<bean class="com.xgj.ioc.inject.construct.innerBean.Gun">
<property name="brand" value="柯尔特M2000"/>
<property name="bulletNum" value="7"/>
<property name="price" value="788"/>
</bean>
</property>
</bean>
</beans>
运行结果
如果希望往一个属性中注入一个null值? 使用专用的<null/>
元素标签
POJO类
package com.xgj.ioc.inject.construct.nullValue;
public class Pilot {
private Plane plane;
public void setPlane(Plane plane) {
this.plane = plane;
}
public void drivePlane() {
System.out.println("Pilot is driving plane....");
plane.info();
}
}
POJO类
package com.xgj.ioc.inject.construct.nullValue;
public class Plane {
private String brand;
public void setBrand(String brand) {
this.brand = brand;
}
public void info() {
System.out.println("Plane brand:" + brand);
}
}
配置文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="plane" class="com.xgj.ioc.inject.construct.nullValue.Plane">
<!-- 设置brand为 null -->
<property name="brand" value="null" />
<!-- 或者 <property name="brand"> <null/> </property> -->
<!-- 错误的写法,这是spring会将 <value></value>解析为一个空字符串
<property name="brand">
<value></value>
</property>-->
</bean>
<bean id="pilot" class="com.xgj.ioc.inject.construct.nullValue.Pilot">
<property name="plane">
<ref bean="plane" />
</property>
</bean>
</beans>
测试类
package com.xgj.ioc.inject.construct.nullValue;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class NullValueTest {
public static void main(String[] args) {
// TODO Auto-generated method stub
ApplicationContext ctx = new ClassPathXmlApplicationContext(
"classpath:com/xgj/ioc/inject/construct/nullValue/beans.xml");
Pilot pilot = ctx.getBean("pilot", Pilot.class);
pilot.drivePlane();
}
}
运行结果:
Spring支持级联属性的配置,假设我们希望在定义Pilot时,直接为Plane的属性提供注入值,可以采取如下方式:
<bean id="pilot" class="com.xgj.ioc.inject.construct.cascadeProperty.Pilot">
<!-- 以圆点的方式定义级别属性 -->
<property name="plane.brand" value="A380"/>
</bean>
上述配置的含义:Spring将调用Pilot.getPlane().setBrand(“A380”)方法进行属性的注入操作。
此时,必须对Pilot的类进行改造,为Plane属性声明一个初始化对象。
实例:
POJO类
package com.xgj.ioc.inject.construct.cascadeProperty;
public class Pilot {
// 声明初始化对象
private Plane plane = new Plane();
// 获取实例
public Plane getPlane() {
return plane;
}
public void setPlane(Plane plane) {
this.plane = plane;
}
public void introduce() {
plane.info();
}
}
POJO类
package com.xgj.ioc.inject.construct.cascadeProperty;
public class Plane {
private String brand;
public void setBrand(String brand) {
this.brand = brand;
}
public void info() {
System.out.println("【Plane brand:】" + brand);
}
}
配置文件
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="pilot" class="com.xgj.ioc.inject.construct.cascadeProperty.Pilot">
<!-- 以圆点的方式定义级别属性 -->
<property name="plane.brand" value="A380"/>
</bean>
</beans>
测试类
package com.xgj.ioc.inject.construct.cascadeProperty;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class CascadePropertyTest {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext(
"classpath:com/xgj/ioc/inject/construct/cascadeProperty/beans.xml");
Pilot pilot = ctx.getBean("pilot", Pilot.class);
pilot.introduce();
}
}
运行结果
如果没有未Plane属性提供Plan对象,Spring在设置级联属性时将抛出NullValueInNestedPathException异常。
我们在Plane类中 ,不对Plane进行new实例化操作,仅仅声明,再次运行NullValueInNestedPathException异常。
Caused by: org.springframework.beans.NullValueInNestedPathException: Invalid property 'plane' of bean class [com.xgj.ioc.inject.construct.cascadeProperty.Pilot]: Value of nested property 'plane' is null
Spring对级联属性的层级没有限制。
文章浏览阅读2w次,点赞7次,收藏51次。四个步骤1.创建C++ Win32项目动态库dll 2.在Win32项目动态库中添加 外部依赖项 lib头文件和lib库3.导出C接口4.c#调用c++动态库开始你的表演...①创建一个空白的解决方案,在解决方案中添加 Visual C++ , Win32 项目空白解决方案的创建:添加Visual C++ , Win32 项目这......_c#调用lib
文章浏览阅读4.6k次。苹方字体是苹果系统上的黑体,挺好看的。注重颜值的网站都会使用,例如知乎:font-family: -apple-system, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Microsoft YaHei, Source Han Sans SC, Noto Sans CJK SC, W..._ubuntu pingfang
文章浏览阅读159次。表单表单概述表单标签表单域按钮控件demo表单标签表单标签基本语法结构<form action="处理数据程序的url地址“ method=”get|post“ name="表单名称”></form><!--action,当提交表单时,向何处发送表单中的数据,地址可以是相对地址也可以是绝对地址--><!--method将表单中的数据传送给服务器处理,get方式直接显示在url地址中,数据可以被缓存,且长度有限制;而post方式数据隐藏传输,_html表单的处理程序有那些
文章浏览阅读1.2k次。使用说明:开启Google的登陆二步验证(即Google Authenticator服务)后用户登陆时需要输入额外由手机客户端生成的一次性密码。实现Google Authenticator功能需要服务器端和客户端的支持。服务器端负责密钥的生成、验证一次性密码是否正确。客户端记录密钥后生成一次性密码。下载谷歌验证类库文件放到项目合适位置(我这边放在项目Vender下面)https://github.com/PHPGangsta/GoogleAuthenticatorPHP代码示例://引入谷_php otp 验证器
文章浏览阅读4.3k次,点赞5次,收藏11次。matplotlib.plot画图横坐标混乱及间隔处理_matplotlib更改横轴间距
文章浏览阅读2.2k次。①Storage driver 处理各镜像层及容器层的处理细节,实现了多层数据的堆叠,为用户 提供了多层数据合并后的统一视图②所有 Storage driver 都使用可堆叠图像层和写时复制(CoW)策略③docker info 命令可查看当系统上的 storage driver主要用于测试目的,不建议用于生成环境。_docker 保存容器
文章浏览阅读834次,点赞27次,收藏13次。网络拓扑结构是指计算机网络中各组件(如计算机、服务器、打印机、路由器、交换机等设备)及其连接线路在物理布局或逻辑构型上的排列形式。这种布局不仅描述了设备间的实际物理连接方式,也决定了数据在网络中流动的路径和方式。不同的网络拓扑结构影响着网络的性能、可靠性、可扩展性及管理维护的难易程度。_网络拓扑csdn
文章浏览阅读1.8k次,点赞5次,收藏8次。IOS系统Date的坑要创建一个指定时间的new Date对象时,通常的做法是:new Date("2020-09-21 11:11:00")这行代码在 PC 端和安卓端都是正常的,而在 iOS 端则会提示 Invalid Date 无效日期。在IOS年月日中间的横岗许换成斜杠,也就是new Date("2020/09/21 11:11:00")通常为了兼容IOS的这个坑,需要做一些额外的特殊处理,笔者在开发的时候经常会忘了兼容IOS系统。所以就想试着重写Date函数,一劳永逸,避免每次ne_date.prototype 将所有 ios
文章浏览阅读5.3k次。方法一:用PLSQL Developer工具。 1 在PLSQL Developer的sql window里输入select * from test for update; 2 按F8执行 3 打开锁, 再按一下加号. 鼠标点到第一列的列头,使全列成选中状态,然后粘贴,最后commit提交即可。(前提..._excel导入pl/sql
文章浏览阅读83次。Git常用命令速查手册1、初始化仓库git init2、将文件添加到仓库git add 文件名 # 将工作区的某个文件添加到暂存区 git add -u # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,不处理untracked的文件git add -A # 添加所有被tracked文件中被修改或删除的文件信息到暂存区,包括untracked的文件...
文章浏览阅读202次。分享119个ASP.NET源码总有一个是你想要的_千博二手车源码v2023 build 1120
文章浏览阅读1.8k次。版权声明:转载请注明出处 http://blog.csdn.net/irean_lau。目录(?)[+]1、缺省构造函数。2、缺省拷贝构造函数。3、 缺省析构函数。4、缺省赋值运算符。5、缺省取址运算符。6、 缺省取址运算符 const。[cpp] view plain copy_空类默认产生哪些类成员函数