扫码一下
查看教程更方便
当容器调用具有多个参数的类构造函数时,就完成了基于构造函数的 DI,每个参数代表对另一个类的依赖。
以下示例显示了一个只能通过构造函数注入进行依赖注入的 TextEditor 类。
下面是 TextEditor.java 文件的内容
TextEditor.java
package com.jiyik; public class TextEditor { private SpellChecker spellChecker; public TextEditor(SpellChecker spellChecker) { System.out.println("Inside TextEditor constructor." ); this.spellChecker = spellChecker; } public void spellCheck() { spellChecker.checkSpelling(); } }
以下是另一个依赖类文件 SpellChecker.java 的内容
SpellChecker.java
package com.jiyik; public class SpellChecker { public SpellChecker(){ System.out.println("Inside SpellChecker constructor." ); } public void checkSpelling() { System.out.println("Inside checkSpelling." ); } }
以下是 MainApp.java 文件的内容
MainApp.java
package com.jiyik; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class MainApp { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("Beans.xml"); TextEditor te = (TextEditor) context.getBean("textEditor"); te.spellCheck(); } }
以下是配置文件 Beans.xml,其中包含基于构造函数的注入的配置
Beans.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-3.0.xsd"> <!-- Definition for textEditor bean --> <bean id = "textEditor" class = "com.jiyik.TextEditor"> <constructor-arg ref = "spellChecker"/> </bean> <!-- Definition for spellChecker bean --> <bean id = "spellChecker" class = "com.jiyik.SpellChecker"></bean> </beans>
创建完源代码和 bean 配置文件后,让我们运行应用程序。 如果应用程序一切正常,它将打印以下消息
Inside SpellChecker constructor.
Inside TextEditor constructor.
Inside checkSpelling.
如果有多个参数,则将参数传递给构造函数时可能会产生歧义。 为了解决这种歧义,在 bean 定义中定义构造函数参数的顺序就是将这些参数提供给适当的构造函数的顺序。 考虑下面的类
package x.y;
public class Foo {
public Foo(Bar bar, Baz baz) {
// ...
}
}
以下配置工作正常
<beans>
<bean id = "foo" class = "x.y.Foo">
<constructor-arg ref = "bar"/>
<constructor-arg ref = "baz"/>
</bean>
<bean id = "bar" class = "x.y.Bar"/>
<bean id = "baz" class = "x.y.Baz"/>
</beans>
让我们再看一种将不同类型传递给构造函数的情况。 考虑下面的类
package x.y;
public class Foo {
public Foo(int year, String name) {
// ...
}
}
如果使用 type 属性显式指定构造函数参数的类型,则容器还可以使用简单类型的类型匹配。 例如
<beans>
<bean id = "exampleBean" class = "examples.ExampleBean">
<constructor-arg type = "int" value = "2001"/>
<constructor-arg type = "java.lang.String" value = "Zara"/>
</bean>
</beans>
最后,传递构造函数参数的最佳方式是使用 index 属性显式指定构造函数参数的索引。 这里,索引是从 0 开始的。 例如
<beans>
<bean id = "exampleBean" class = "examples.ExampleBean">
<constructor-arg index = "0" value = "2001"/>
<constructor-arg index = "1" value = "Zara"/>
</bean>
</beans>
最后一点,如果你传递一个对象的引用,你需要使用 <constructor-arg> 标签的 ref 属性,如果你直接传递一个值,那么你应该使用如上所示的 value 属性。