基于构造函数的依赖注入
当容器调用具有多个参数的类构造函数时,就完成了基于构造函数的 DI,每个参数都表示对另一个类的依赖关系。
示例
下面的示例显示了一个类 TextEditor,它只能通过构造函数注入进行依赖注入。
让我们有一个可以工作的 Eclipse IDE,并按照以下步骤创建一个 Spring 应用程序 −
步骤 | 描述 |
---|---|
1 | 创建一个名为 SpringExample 的项目,并在创建的项目中的 src 文件夹下创建一个包 com.tutorialspoint。 |
2 | 使用 Add External JARs 选项添加所需的 Spring 库,如 Spring Hello World 示例 一章中所述。 |
3 | 在 com.tutorialspoint 包下创建 Java 类 |
4 | 在 src 文件夹下创建 Beans 配置文件 Beans.xml。 |
5 | 最后一步是创建所有 Java 文件和 Bean 配置文件的内容,然后运行应用程序,如下所述。 |
这是 TextEditor.java 文件的内容 −
package com.tutorialspoint; 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的内容
package com.tutorialspoint; public class SpellChecker { public SpellChecker(){ System.out.println("Inside SpellChecker constructor." ); } public void checkSpelling() { System.out.println("Inside checkSpelling." ); } }
以下是 MainApp.java 文件的内容
package com.tutorialspoint; 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 ,它具有基于构造函数的注入的配置 −
<?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.tutorialspoint.TextEditor"> <constructor-arg ref = "spellChecker"/> </bean> <!-- Definition for spellChecker bean --> <bean id = "spellChecker" class = "com.tutorialspoint.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 属性。