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