자바 의 제어 반전 과 의존 주입
15989 단어 자바
주입 과 제어 반전 에 의존 하 는 목적 은 클래스 와 클래스 간 의 결합 을 풀 고 시스템 의 확장 성과 유지 가능성 을 향상 시 키 기 위해 다음 과 같은 예 를 통 해 이 개념 을 도입 하 는 것 이다.
2. 사례
1) 일반적인 상황 에서 의 클래스 결합
Main.java
public class Main {
public static void main(String[] args) {
/******** 一般写法,Main类与Chinese类和American类之间的强耦合 ***********/
// Chinese和American,当类和方法修改时,此处的类和方法也需要修改
Chinese chinese = new Chinese();
chinese.sayHelloWorld("张三");
American american = new American();
american.sayHelloWorld("Jack");
}
}
/******************** 一般方法 ***************************/
interface Human {
public void sayHelloWorld(String name);
}
class Chinese implements Human {
public void sayHelloWorld(String name) {
String helloWorld = "你好," + name;
System.out.println(helloWorld);
}
}
class American implements Human {
public void sayHelloWorld(String name) {
String helloWorld = "Hello," + name;
System.out.println(helloWorld);
}
}
위의 코드 를 통 해 알 수 있 듯 이 Main 류 와 Chinese 류 와 American 류 사이 에 강 한 결합 이 존재 합 니 다. Chinese 와 American 클래스 와 방법 을 수정 할 때 이곳 의 클래스 와 방법 도 수정 해 야 합 니 다.확장 과 유지 보수 가 쉽 지 않 습 니 다.
2) 공장 방법 으로 결합 을 푼다
public class Main {
public static void main(String[] args) {
/******** 工厂方法, Main类与类Chinese和American不再耦合,仅仅和其接口Human耦合 ***********/
// 修改时还需要修改在Main类中修改这些字符串
// Chinese和American,当类和方法修改时,只有方法需要修改
HumanFactory humanFactory = new HumanFactory();
Human human1 = humanFactory.getHuman("chinese");
human1.sayHelloWorld("张三");
Human human2 = humanFactory.getHuman("american");
human2.sayHelloWorld("Jack");
}
}
/******************** 工厂方法 ***************************/
interface Human {
public void sayHelloWorld(String name);
}
class HumanFactory {
public Human getHuman(String type) {
if ("chinese".equals(type)) {
return new Chinese();
} else {
return new American();
}
}
}
위의 코드 를 통 해 알 수 있 듯 이 Main 류 와 클래스 Chinese 와 American 은 더 이상 결합 하지 않 고 인터페이스 Human 과 만 결합 합 니 다. 수정 할 때 Main 류 에서 이 문자열 을 수정 해 야 합 니 다. 클래스 와 방법 을 수정 할 때 방법 만 수정 해 야 합 니 다.이 는 어느 정도 에 Main 류 와 Chinese, American 류 의 결합 을 낮 추 었 다.
3) 주입 과 제어 반전 에 의존한다
public class Main {
public static void main(String[] args) {
/******************** IOC控制反转和依赖注入 ***************************/
// 利用容器,通过xml文件直接注入属性值,在Main类中只添加需要的
// Chinese和American,当类和方法修改时,代码完全不用修改,只需要修改xml文件即可,彻底实现了解耦
BeanFactory beanFactory = new BeanFactory();
beanFactory.init("/config.xml");
UserBean userBean = (UserBean) beanFactory.getBean("userBean");
System.out.println("userName=" + userBean.getUserName());
System.out.println("password=" + userBean.getPassword());
}
}
/******************** IOC控制反转和依赖注入 ***************************/
// 下面是Spring的IOC实现:Bean工厂
class BeanFactory {
private Map<String, Object> beanMap = new HashMap<String, Object>();
public void init(String fileName) {
try {
// 读取指定的配置文件
SAXReader reader = new SAXReader();
// System.out.println(xmlpath);
String realPathString = new File("").getCanonicalPath();
Document document = reader.read(new File(realPathString + "/src/com/devin/") + fileName);
Element root = document.getRootElement();
Element foo;
// 遍历bean
for (Iterator i = root.elementIterator("bean"); i.hasNext();) {
foo = (Element) i.next();
// 获取bean的属性id和class
Attribute id = foo.attribute("id");
Attribute cls = foo.attribute("class");
// 利用Java反射机制,通过class的名称获取Class对象
Class bean = Class.forName(cls.getText());
// 获取对应class的信息
java.beans.BeanInfo info = java.beans.Introspector.getBeanInfo(bean);
// 获取其属性描述
java.beans.PropertyDescriptor pd[] = info.getPropertyDescriptors();
// 设置值的方法
Method mSet = null;
// 创建一个对象
Object obj = bean.newInstance();
// 遍历该bean的property属性
for (Iterator ite = foo.elementIterator("property"); ite.hasNext();) {
Element foo2 = (Element) ite.next();
// 获取该property的name属性
Attribute name = foo2.attribute("name");
String value = null;
// 获取该property的子元素value的值
for (Iterator ite1 = foo2.elementIterator("value"); ite1.hasNext();) {
Element node = (Element) ite1.next();
value = node.getText();
break;
}
for (int k = 0; k < pd.length; k++) {
if (pd[k].getName().equalsIgnoreCase(name.getText())) {
mSet = pd[k].getWriteMethod();
mSet.invoke(obj, value);
}
}
}
// 将对象放入beanMap中,其中key为id值,value为对象
beanMap.put(id.getText(), obj);
}
} catch (Exception e) {
System.out.println(e.toString());
}
}
// 通过bean的id获取bean的对象.
public Object getBean(String beanName) {
Object obj = beanMap.get(beanName);
return obj;
}
}
UserBean.java
public class UserBean {
private String userName;
private String password;
public String getPassword() {
return password;
}
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
public void setPassword(String password) {
this.password = password;
}
}
config.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans>
<bean id="userBean" class="com.devin.UserBean">
<property name="userName">
<value>张三</value>
</property>
<property name="password">
<value>Jack</value>
</property>
</bean>
</beans>
说明:模拟了Spring中IOC的实现,虽然只是完成了Spring中依赖注入的一小部分工作,但是很好的展现了Java反射机制在Spring中的应用,能使我们能更好的从原理上了解IOC的实现。
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Is Eclipse IDE dying?In 2014 the Eclipse IDE is the leading development environment for Java with a market share of approximately 65%. but ac...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.