溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Spring IOC和aop的原理及實例詳解

發(fā)布時間:2020-09-29 04:28:55 來源:腳本之家 閱讀:231 作者:我看見的世界 欄目:編程語言

這篇文章主要介紹了Spring IOC和aop的原理及實例詳解,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下

Spring是一個輕量級的控制反轉(zhuǎn)(IoC)和面向切面(AOP)的容器框架。特點是面向接口編程,松耦合。

1:IOC(控制反轉(zhuǎn)) 別名(DI:依賴注入)

首先來一段ioc的實現(xiàn)原來代碼:

public class ClassPathXmlApplicationContext implements BeanFactory {
 
 private Map<String , Object> beans = new HashMap<String, Object>();
 
 //IOC Inverse of Control DI Dependency Injection
 public ClassPathXmlApplicationContext() throws Exception {
  SAXBuilder sb=new SAXBuilder();
  //解析xml配置文件
  Document doc=sb.build(this.getClass().getClassLoader().getResourceAsStream("beans.xml"));
  Element root=doc.getRootElement(); //獲取根元素
  List list=root.getChildren("bean");//根元素下的子元素
  for(int i=0;i<list.size();i++) {
   Element element=(Element)list.get(i);
   String id=element.getAttributeValue("id");
   String clazz=element.getAttributeValue("class");
   Object o = Class.forName(clazz).newInstance(); //反射獲得實例
   System.out.println(id);
   System.out.println(clazz);
   beans.put(id, o);
   //注入bean屬性
   for(Element propertyElement : (List<Element>)element.getChildren("property")) {
    String name = propertyElement.getAttributeValue("name"); //userDAO
    String bean = propertyElement.getAttributeValue("bean"); //u
    Object beanObject = beans.get(bean);//UserDAOImpl instance
    
    String methodName = "set" + name.substring(0, 1).toUpperCase() + name.substring(1);
    System.out.println("method name = " + methodName);
    
    Method m = o.getClass().getMethod(methodName, beanObject.getClass().getInterfaces()[0]);
    m.invoke(o, beanObject); 
   }
  } 
 }

 public Object getBean(String id) {
  return beans.get(id);
 }
}
//xml文件
<beans>
 <bean id="u" class="com.bjsxt.dao.impl.UserDAOImpl" />
 <bean id="userService" class="com.bjsxt.service.UserService" >
  <property name="userDAO" bean="u"/>
 </bean>
 
</beans>

以上代碼實現(xiàn)了將UserDAOImpl注入到userService.

值得注意的是:以上操作都是spring幫我們實現(xiàn)的,我們只需要理解如何配置即可。

spring還提供了bean的生存周期和范圍屬性:

scope:singleton(單例,即每次調(diào)用的對象都為同一個)。 prototype(原型,即以bean對象為原型,每次new一個新對象出來)

init-method="init" destroy-method="destroy" lazy-init="true"(延遲初始化bean,即spring啟動時,不初始化bean,當(dāng)需要使用時才實例化,作用:但spring啟動緩慢時可使用)

現(xiàn)在基本都是用注解實現(xiàn),但只要能明白spring是如何實現(xiàn)bean的注入,基本原理都是一樣的。spring在中間的作用就是幫我們實現(xiàn)元素之前的注入,誰注入到誰就需要用到不同的注解了。

AOP(切面編程)

首先還是先來一段代碼

定義一個InvocationHandler的實現(xiàn)類

public class UserServiceTest {

 @Test
 public void testAdd() throws Exception {
  BeanFactory applicationContext = new ClassPathXmlApplicationContext();

  UserService service = (UserService)applicationContext.getBean("userService");
   
  User u = new User();
  u.setUsername("zhangsan");
  u.setPassword("zhangsan");
  service.add(u);
 }
 
 @Test
 public void testProxy() {
  UserDAO userDAO = new UserDAOImpl();
  LogInterceptor li = new LogInterceptor();
  li.setTarget(userDAO);
  UserDAO userDAOProxy = (UserDAO)Proxy.newProxyInstance(userDAO.getClass().getClassLoader(), userDAO.getClass().getInterfaces(), li);
  
  userDAOProxy.delete();
  userDAOProxy.save(new User());
 }

調(diào)用測試類,service和DAO實現(xiàn)類沒有貼出來,就是打印一句話出來表現(xiàn)方法執(zhí)行了。

public class LogInterceptor implements InvocationHandler {
 private Object target;
 public Object getTarget() {
  return target;
 }
 public void setTarget(Object target) {
  this.target = target;
 }

 public void beforeMethod(Method m) {
  System.out.println(m.getName() + " start");
 }

 @Override
 public Object invoke(Object proxy, Method m, Object[] args)
   throws Throwable {
  beforeMethod(m);
  m.invoke(target, args);
  return null;
 }
}

以上代碼簡單的實現(xiàn)了一個動態(tài)代理,并執(zhí)行了自己的一段邏輯。

那么aop是如何實現(xiàn)切面編程的呢,就是通過動態(tài)代理。當(dāng)然這也是spring來實現(xiàn)的,而我們需要做的就是知道如何編寫編織語法。

  • JoinPoint:切入點
  • PointCut: 切入點的集合
  • Aspect:切面
  • Advice: 相當(dāng)于Aspect的before
  • Around:周圍,環(huán)繞(需要帶參數(shù)ProceedingJoinPoint jp)

需要注意的是:被產(chǎn)生代理的對象需要現(xiàn)實接口spring才能產(chǎn)生代理對象,默認使用java se 中的 InvocationHandler 和 proxy 兩個類產(chǎn)生代理對象,若沒有實現(xiàn)接口,則需要另外引入一個jar包(cglib-nodep-2.1_3.jar);他是通過二進流的方式產(chǎn)生代理對象。

那么spring 幫我們做了什么?

1:充分利用java se 中的反射機制幫助我們對對象的注入,即 IOC,

2: 也是java se 中的動態(tài)代理幫助我們實現(xiàn) 切面編程,當(dāng)然我們也需要熟悉一下asceptj 的切入語法。這就是aop。

所以這里可以總結(jié)一下,java se 的特點有反射,動態(tài)代理,(這里面必然會用到多態(tài),動態(tài)綁定)

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI