教育行業(yè)A股IPO第一股(股票代碼 003032)

全國咨詢/投訴熱線:400-618-4000

Java培訓實戰(zhàn)教程之自定義spring

更新時間:2015年12月29日13時44分 來源:傳智播客Java培訓學院 瀏覽次數(shù):

1   Java培訓實戰(zhàn)教程之自定義spring

1.1   描述

       在企業(yè)級開發(fā)中,spring框架應用非常廣。為了讓已經學習過spring框架同學,可以更深入的理解和應用spring,本文將通過自定義spring,更佳系統(tǒng)的闡述spring核心:IoC、AOP。
       IoC(Inversion of Control)控制反轉:將對象的創(chuàng)建權交與spring框架,及將創(chuàng)建權反轉給spring框架。IoC主要解決計算機程序的耦合問題。
       AOP(Aspect Oriented Programming)面向切面編程:通過運行期動態(tài)代理實現(xiàn)程序功能的統(tǒng)一維護的一種技術。
 

1.2   分析

l  如果要實現(xiàn)自定義spring,可以將器拆分成多個功能實現(xiàn)。
l  階段一:編寫配置文件,服務器tomcat啟動時,加載配置文件
l  階段二:使用Jsoup解析xml,并封裝到指定的JavaBean中
l  階段三:編寫工廠類用于創(chuàng)建指定bean,并完成property注入
l  階段四:使用@Transactional進行事務管理
 

1.3   搭建環(huán)境

1.3.1   javabean

public class User {
 
   private Integer uid;
   private String username;
   private String password;
 

1.3.2    dao

public interface UserDao {
   /**
    * 保存
    * @param user
    */
   public void save(User user);
 
}
public class UserDaoImpl implements UserDao {
 
   @Override
   public void save(User user) {
      //TODO 暫時只打印
      System.out.println(user);
   }
 
}
 

1.3.3   service

public interface UserService {
   /**
    * 注冊
    * @param user
    */
   public void register(User user);
}
public class UserServiceImpl implements UserService {
 
   private UserDao userDao;
   public void setUserDao(UserDao userDao) {
      this.userDao = userDao;
   }
  
   @Override
   public void register(User user) {
      this.userDao.save(user);
   }
}
 

1.4   階段一:編寫配置文件,服務器啟動加載

1.4.1   xml配置文件

l  在src下添加“applicationContext.xml”,并將dao和service配置到xml文件中。
l  使用<bean>標簽配置一個實現(xiàn)類
       class:    配置實現(xiàn)類的全限定名稱
       id: 進行唯一命名,用于提供給程序獲得
l  使用<property>配置javabean屬性的注入
       name:   配置的service的屬性名
       ref:       配置其他bean對象的引用
      
<?xml version="1.0" encoding="UTF-8"?>
<beans>
    <!-- dao -->
    <bean id="userDaoId" class="cn.itcast.demo.dao.impl.UserDaoImpl"></bean>
    <!-- service -->
    <bean id="userServiceId" class="cn.itcast.demo.service.impl.UserServiceImpl">
        <property name="userDao" ref="userDaoId"></property>
    </bean>
</beans>
 

1.4.2   加載配置文件

l  tomcat啟動時,加載配置文件方式總結:
       1.編寫Servlet,配置servlet,并添加<load-on-startup>,在init(ServletConfig)初始化方式中加載。
       2.編寫Filter,配置filter,在init(FilterConfig)初始化方法中加載
       3.編寫Listener,實現(xiàn)接口ServletContext,配置listener,在contextInitialized(ServletContextEvent sce)方法中加載。
l  spring采用listener方案
       1.提供實現(xiàn)類ContextLoaderListener
       2.編寫全局初始化參數(shù)contextConfigLocation,用于確定xml位置
              <param-value>classpath:applicationContext.xml</param-value> 加載類路徑下的xml文件
        <param-value>applicationContext.xml</param-value> 加載WEB-INF目錄的配置文件
 
l  xml配置
<!-- 確定xml位置 -->
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:applicationContext.xml</param-value>
    </context-param>
   
    <!-- 配置監(jiān)聽器 -->
    <listener>
        <listener-class>cn.itcast.myspring.listener.ContextLoaderListener</listener-class>
    </listener>
 
l  實現(xiàn)類
      
public class ContextLoaderListener implements ServletContextListener {
 
    @Override
    public void contextInitialized(ServletContextEvent sce) {
        // 0 獲得ServletContext對象應用
        ServletContext context = sce.getServletContext();
        // 1 加載配置
        String config = context.getInitParameter("contextConfigLocation");
        if(config == null){ //默認配置文件位置
            config= "applicationContext.xml";
        }
       
        InputStream xmlIs = null;
        // 2  處理路徑不同情況
        // * classpath:applicationContext.xml --> 表示 src/applicationContext.xml
        // * applicationContext.xml --> 表示 /WEB-INF/applicationContext.xml
        if (config.startsWith("classpath:")) { // 2.1 加載 類路徑 (classpath、src)下的xml
            xmlIs = ContextLoaderListener.class.getClassLoader().getResourceAsStream(config.substring("classpath:".length()));
        } else { //2.2 加載/WEB-INF/目錄下的資源
            xmlIs = context.getResourceAsStream("/WEB-INF/" + config);
        }
        //2.3 配置文件必須存在,否則拋異常
        if(xmlIs == null){
            throw new RuntimeException("資源文件["+config+"]沒有找到");
        }
       
        //TODO 3 解析配置
        if (xmlIs != null) {
            System.out.println(xmlIs);
        }
    }
    @Override
    public void contextDestroyed(ServletContextEvent sce) {
    }
}
 

1.5   階段二:解析xml,并封裝到指定javabean中

1.提供Property類,用于封裝<property name=" " ref=" "></property>
2.提供Bean類,用于封裝<bean id=" " class=" ">
       一個<bean> 標簽體中可以配置多個<property>需要一個容器存放,沒有順序要求,且不能重復,選擇Set
3.提供BeanFactory類,并在類同提供容器存放多個Bean 類,為了方便獲取使用Map。

1.5.1    property javabean

/**
 * 用于封裝 <property name="userDao" ref="userDaoId"></property>
 */
public class Property {
  
   //屬性名稱
   private String name;
   //另一個bean引用名
   private String ref;
  
   public Property(String name, String ref) {
      super();
      this.name = name;
      this.ref = ref;
   }
 

1.5.2    bean  javabean

public class Bean {
 
   //bean名稱
   private String beanId;
   //bean的實現(xiàn)類
   private String beanClass;
   //取值:singleton 單例,prototype 原型(多例)【擴展】
   private String beanType;   
  
   //所有的property
   private Set<Property> propSet = new HashSet<Property>();
  
   public Bean(String beanId, String beanClass) {
      super();
      this.beanId = beanId;
      this.beanClass = beanClass;
   }
 

1.5.3    BeanFactory 工廠模式類

public class BeanFactory {
   
    //////////////////工廠模式////////////////////////
    private static BeanFactory factory = new BeanFactory();
    private BeanFactory(){
    }
 
    /**
     * 獲得工廠實例
     * @author lt
     * @return
     */
    public static BeanFactory getInstance() {
        return factory;
    }
 

1.5.4    BeanFactory 提供Map 緩存

    //////////////////緩存所有的Bean/////////////////////////////////
    //bean數(shù)據(jù)緩存集合 ,key:bean名稱 ,value:bean封裝對象
    private static Map<String, Bean> beanData;// = new HashMap<String, String>();
    static{
        // 從配置文件中獲得相應的數(shù)據(jù) 1.properties 2.xml
        //beanData.put("userDao", "com.itheima.ebs.service.impl.BusinessServiceImpl");
    }
   
    public static void setBeanData(Map<String, Bean> beanData) {
        BeanFactory.beanData = beanData;
    }
 

1.5.5    修改Listener解析xml

l  使用Jsoup解析,導入jar包
      
 
l  修改contextInitialized 方法
//TODO 3 解析配置
      if (xmlIs != null) {
         //3.1解析
         Map<String, Bean> data = parserBeanXml(xmlIs);
         //3.2 將解析結果放置到工廠中
         BeanFactory.setBeanData(data);
      }
 
l  解析方法parserBeanXml(InputStream)
/**
    * 將數(shù)據(jù)解析成bean
    * @param xmlIs
    * @return
    */
   public static Map<String, Bean> parserBeanXml(InputStream xmlIs) {
      try {
         //0提供緩沖區(qū)域
         Map<String, Bean> data = new HashMap<String, Bean>();
        
        
         //1解析文件,并獲得Document
         Document document = Jsoup.parse(xmlIs, "UTF-8", "");
        
         //2 獲得所有的bean元素
         Elements allBeanElement = document.getElementsByTag("bean");
        
         //3遍歷
         for (Element beanElement : allBeanElement) {
            //5 將解析的結果封裝到bean中
            // 5.1 bean名稱
            String beanId = beanElement.attr("id");
            // 5.2 bean實現(xiàn)類
            String beanClass = beanElement.attr("class");
           
            // 5.3 封裝到Bean對象
            Bean bean = new Bean(beanId,beanClass);
           
            // 6 獲得所有的子元素 property
            Elements allPropertyElement = beanElement.children();
            for (Element propertyElement : allPropertyElement) {
                String propName = propertyElement.attr("name");
                String propRef = propertyElement.attr("ref");
                Property property = new Property(propName, propRef);
               
               // 6.1 將屬性追加到bean中
                bean.getPropSet().add(property);
            }
           
            data.put(beanId, bean);
         }
         return data;
      } catch (Exception e) {
         throw new RuntimeException(e);
      }
   }
 

1.6   階段三:完善BeanFactory獲得實例

l  使用 BeanUtils.setProperty 設置數(shù)據(jù),需要導入jar包
      
l  BeanFactory 提供 getBean方法
////////////////獲得Bean實例///////////////////////////////////
    public Object getBean(String beanId) {
       
        try {
            // 通過bean 的名稱獲得具體實現(xiàn)類
            Bean bean = beanData.get(beanId);
            if(bean ==null) return null;
            String beanClass = bean.getBeanClass();
            Class clazz = Class.forName(beanClass);
            Object beanObj = clazz.newInstance();
            //DI 依賴注入,將在配置文件中設置的內容,通過bean的set方法設置到bean實例中
            Set<Property> props = bean.getPropSet();
            for (Property property : props) {
                String propName = property.getName();
                String propRef = property.getRef(); //另一個javabean,需要重容器中獲得
                Object propRefObj = getBean(propRef);
                //PropertyDescriptor pd = new PropertyDescriptor(propName, clazz);
                //pd.getWriteMethod().invoke(bean, propRefObj);
                BeanUtils.setProperty(beanObj, propName, propRefObj);
            }
           
            return beanObj;
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
 
l  測試
//測試
UserService userService = (UserService) BeanFactory.getInstance().getBean("userServiceId");
userService.register(null);
 
 

1.7   階段四:spring事務管理

 

1.7.1   修改自定義spring

l  提供JdbcUtils工具類,用于在當前線程中共享Connection
l  提供@Transaction 用于標記那些類需要進行事務管理
 
 

1.7.1.1         JdbcUtils工具類

l  使用ThreadLocal 保存Connection,在當前線程中共享Connection
l  并提供提交和回滾方法,自動關閉連接
public class JdbcUtils {
   
    private static ThreadLocal<Connection> local = new ThreadLocal<Connection>();
   
    static{
        try {
            //注冊驅動
            Class.forName("com.mysql.jdbc.Driver");
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
    /**
     * 獲得連接
     * @return
     */
    public static Connection getConnection(){
        try {
            Connection conn = local.get();
            if (conn == null) {
                conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/test2", "root", "1234");
                local.set(conn);
            }
            return conn;
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
    /**
     * 提交事務
     */
    public static void commit() {
        Connection conn = getConnection();
        DbUtils.commitAndCloseQuietly(conn);
    }
 
    /**
     * 回滾事務
     */
    public static void rollback() {
        Connection conn = getConnection();
        DbUtils.rollbackAndCloseQuietly(conn);
    }
 
}
 

1.7.1.2         @Transactional

 
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Transactional {
 
}
 

1.7.1.3         修改BeanFactory

l  通過getBean 獲得對象實例時,如果對象有@Transactional注解,將返回代理對象,對目標對象上所有的方法都進行事務管理。
l  如果沒有異常提交事務,并關閉連接
l  如果有異?;貪L事務,并關閉連接
 
public Object getBean(String beanId) {
       
        try {
            // 通過bean 的名稱獲得具體實現(xiàn)類
            Bean bean = beanData.get(beanId);
            if(bean ==null) return null;
            String beanClass = bean.getBeanClass();
            Class<?> clazz = Class.forName(beanClass);
            Object beanObj = clazz.newInstance();
            //DI 依賴注入,將在配置文件中設置的內容,通過bean的set方法設置到bean實例中
            Set<Property> props = bean.getPropSet();
            for (Property property : props) {
                String propName = property.getName();
                String propRef = property.getRef(); //另一個javabean,需要重容器中獲得
                Object propRefObj = getBean(propRef);
                //PropertyDescriptor pd = new PropertyDescriptor(propName, clazz);
                //pd.getWriteMethod().invoke(bean, propRefObj);
                BeanUtils.setProperty(beanObj, propName, propRefObj);
            }
           
            //如果類上有注解返回代理對象
            if(clazz.isAnnotationPresent(Transactional.class)){
                final Object _beanObj = beanObj;
                return Proxy.newProxyInstance(
                        clazz.getClassLoader(),
                        clazz.getInterfaces(),
                        new InvocationHandler() {
                            @Override
                            public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                                try {
                                    //開啟事務
                                    JdbcUtils.getConnection().setAutoCommit(false);
                                   
                                    //執(zhí)行目標方法
                                    Object obj = method.invoke(_beanObj, args);
                                   
                                    //提交事務
                                    JdbcUtils.commit();
                                   
                                    return obj;
                                   
                                } catch (Exception e) {
                                    //回顧事務
                                    JdbcUtils.rollback();
                                    throw new RuntimeException(e);
                                }
                            }
                        });
            }
           
            return beanObj;
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
 
 

1.7.2   初始化數(shù)據(jù)庫

create table account(
  id int primary key auto_increment,
  username varchar(50),
  money int
);
insert into account(username,money) values('jack','10000');
insert into account(username,money) values('rose','10000');
 

1.7.3   dao層

l  必須使用自定義spring提供的JdbcUtils獲得連接,從而保證當前線程使用的是同一個線程。
l  通過xml配置文件創(chuàng)建QueryRunner實例,并注入給dao
l  擴展:如果將QueryRunner和JdbcUtils都省略,需要自己實現(xiàn)JdbcTemplate完成。
 
public class AccountDaoImpl implements AccountDao {
   
    private QueryRunner runner;
    public void setRunner(QueryRunner runner) {
        this.runner = runner;
    }
 
    @Override
    public void out(String outer, Integer money) {
        try {
            Connection conn = JdbcUtils.getConnection();
            runner.update(conn, "update account set money = money - ? where username = ?", money, outer);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
 
    @Override
    public void in(String inner, Integer money) {
        try {
            Connection conn = JdbcUtils.getConnection();
            runner.update(conn, "update account set money = money + ? where username = ?", money,inner);
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
 
}
 

1.7.4   service層

l  在實現(xiàn)類上添加注解
 
@Transactional
public class AccountServiceImpl implements AccountService {
 
    private AccountDao accountDao;
    public void setAccountDao(AccountDao accountDao) {
        this.accountDao = accountDao;
    }
    @Override
    public void transfer(String outer, String inner, Integer money) {
        accountDao.out(outer, money);
        //斷電
//      int i = 1/0;
        accountDao.in(inner, money);
    }
 
}
 

1.7.5   編寫spring配置

    <!-- 創(chuàng)建queryRunner -->
    <bean id="runner" class="org.apache.commons.dbutils.QueryRunner"></bean>
   
    <bean id="accountDao" class="cn.itcast.dao.impl.AccountDaoImpl">
        <property name="runner" ref="runner"></property>
    </bean>
    <!-- service -->
    <bean id="accountService" class="cn.itcast.service.impl.AccountServiceImpl">
        <property name="accountDao" ref="accountDao"></property>
    </bean>
 

1.7.6   編寫servlet測試

l  通過請求servlet,進行轉賬,此處使用固定值進行測試
public class AccountServlet extends HttpServlet {
 
    public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        AccountService accountService = (AccountService) BeanFactory.getInstance().getBean("accountService");
        accountService.transfer("jack", "rose", 100);
    }
 
 
本文版權歸傳智播客Java培訓學院所有,歡迎轉載,轉載請注明作者出處。謝謝!
作者:傳智播客Java培訓學院
首發(fā):http://8y3kgpwe.cn/javaee 
 
 
 
 
 

0 分享到:
和我們在線交談!