成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

從模板方法模式到JDBCTemplate

番茄西紅柿 / 1009人閱讀

摘要:模板方法模式的更多應(yīng)用事實(shí)上很多有關(guān)生命周期的類都用到了模板方法模式,最典

將大象裝進(jìn)冰箱需要三步,那么老虎了?如何優(yōu)雅的將大象裝進(jìn)冰箱?

把大象裝進(jìn)冰箱
Step 大象 老虎 ...
First 打開冰箱門 打開冰箱門 打開冰箱門
Second 把大象放進(jìn)去 把老虎放進(jìn)去 ...
Third 關(guān)閉冰箱門 關(guān)閉冰箱門 關(guān)閉冰箱門

大象類

    public class Elephant {
        public void putRefrigerator() {
            openDoor();
            putElephant();
            closeDoor();
        }
        public void openDoor() {
            System.out.println("open the door");
        }
        public void putElephant() {
            System.out.println("put in the Elephant");
        }
        public void closeDoor() {
            System.out.println("close the door");
        }
    }

老虎類

    public class Tiger {
        public void putRefrigerator() {
            openDoor();
            putTiger();
            closeDoor();
        }
        public void openDoor() {
            System.out.println("open the door");
        }
        public void putTiger() {
            System.out.println("put in the Tiger");
        }
        public void closeDoor() {
            System.out.println("close the door");
        }
    }

可以看出我們將大象和老虎放進(jìn)冰箱的過(guò)程中出現(xiàn)了大量的重復(fù)代碼,這顯然不是一個(gè)好的設(shè)計(jì),如果我們?cè)谝院蟮南到y(tǒng)升級(jí)過(guò)程中需要再放入長(zhǎng)頸鹿怎么辦,我們應(yīng)該如何從我們的設(shè)計(jì)中刪除這些重復(fù)代碼?通過(guò)觀察我們發(fā)現(xiàn)放大象和放老虎之間有很多共同點(diǎn),都需要進(jìn)行開關(guān)門的操作,只是放的過(guò)程不盡相同,我們是否可以將共同點(diǎn)抽離?我們一起試試看

抽象超類

    public abstract class AbstractPutAnyAnimal {
        //這是一個(gè)模板方法,它是一個(gè)算法的模板,描述我們將動(dòng)物放進(jìn)冰箱的步驟,每一個(gè)方法代表了一個(gè)步驟
        public void putRefrigerator() {
            openDoor();
            putAnyAnimal();
            closeDoor();
        }
        //在超類中實(shí)現(xiàn)共同的方法,由超類來(lái)處理
        public void openDoor() {
            System.out.println("open the door");
        }
        public void closeDoor() {
            System.out.println("close the door");
        }
        //每個(gè)子類可能有不同的方法,我們定義成抽象方法讓子類去實(shí)現(xiàn)
        abstract void putAnyAnimal();
    }

大象類

    public class Elephant extends AbstractPutAnyAnimal {
        //子類實(shí)現(xiàn)自己的業(yè)務(wù)邏輯
        @Override
        void putAnyAnimal() {
            System.out.println("put in the Elephant");
        }
    }

老虎類

    public class Tiger extends AbstractPutAnyAnimal {
        //子類實(shí)現(xiàn)自己的業(yè)務(wù)邏輯
        @Override
        void putAnyAnimal() {
            System.out.println("put in the Tiger");
        }
    }

通過(guò)將相同的方法抽離到超類中,并定義一個(gè)抽象方法供子類提供不同的實(shí)現(xiàn),事實(shí)上我們剛剛實(shí)現(xiàn)了一個(gè)模板方法模式。

模板方法模式定義?

模板方法模式定義了一個(gè)算法的步驟,并允許子類為一個(gè)或多個(gè)步驟提供實(shí)現(xiàn),putRefrigerator 方法定義了我們將大象裝進(jìn)冰箱的步驟它就是一個(gè)模板方法。模板方法模式在一個(gè)方法中定義一個(gè)算法的骨架,而將一些步驟延遲到子類中,模板方法使得子類可以不在改變算法結(jié)構(gòu)的情況下,重新定義算法的某些步驟(子類提供自己的實(shí)現(xiàn))

模板方法模式中的鉤子

我們可以在超類中定義一個(gè)空方法,我們稱這種方法為鉤子(hook)。子類可以依據(jù)情況選擇覆蓋,鉤子的存在可以讓子類有能力對(duì)算法的不同點(diǎn)進(jìn)行掛載;鉤子可以讓子類實(shí)現(xiàn)算法中的可選部分,鉤子也可以讓子類為抽象類做一些決定我們將大象裝進(jìn)冰箱后可能會(huì)想調(diào)整冰箱溫度,也可能什么都不做使用默認(rèn)溫度,我們可以通過(guò)定義一個(gè)鉤子,讓子類來(lái)選擇是否調(diào)整溫度,如下:

抽象父類

    public abstract class AbstractPutAnyAnimal {
        public void putRefrigerator() {
            openDoor();
            putAnyAnimal();
            closeDoor();
            //默認(rèn)為false,重新這個(gè)方法決定是否執(zhí)行addTemperature();方法
            if (isAdd()) {
                addTemperature();
            }
        }
        public void openDoor() {
            System.out.println("open the door");
        }
        public void closeDoor() {
            System.out.println("close the door");
        }
        abstract void putAnyAnimal();
        void addTemperature(){
            System.out.println("plus one");
        };
        //定義一個(gè)空實(shí)現(xiàn),由子類決定是否對(duì)其進(jìn)行實(shí)現(xiàn)
        boolean isAdd(){
            return false;
        }
    }

大象類

    public class Elephant extends AbstractPutAnyAnimal {
        @Override
        void putAnyAnimal() {
            System.out.println("put in the Elephant");
        }
        //子類實(shí)現(xiàn)鉤子方法
        @Override
        boolean isAdd() {
            return true;
        }
    }

我們通過(guò)定義一個(gè)鉤子方法,子類選擇是否實(shí)現(xiàn)這個(gè)鉤子方法,來(lái)決定是否調(diào)整溫度;當(dāng)然鉤子方法的用途不止如此,它還能讓子類有機(jī)會(huì)對(duì)模板中即將發(fā)生或剛剛發(fā)生的步驟做出反應(yīng),這在JDK中有很多的例子,甚至在前端開發(fā)領(lǐng)域也有很多例子,我就不具體展開代碼演示了,后面在模板方法模式的更多應(yīng)用中展開。

JDK以及Spring中使用了很多的設(shè)計(jì)模式,下面我們通過(guò)比較傳統(tǒng)JDBC編程和JDBCTemplate來(lái)看看模板方法模式是如何幫我們消除樣板代碼的

傳統(tǒng)JDBC編程

JDBC編程之新增

        String driver = "com.mysql.jdbc.Driver";
        String url = "jdbc:mysql://localhost:3306/test);;
        String username = "root";
        String password = "1234";
        Connection connection = null;
        Statement statement = null;
        try {
            Class.forName(driver);
            connection = DriverManager.getConnection(url, username, password);
            String sql = "insert into users(nickname,comment,age) values(小小譚,I love three thousand times, 21)";
            statement = connection.createStatement();
            int i = statement.executeUpdate(sql);
            return i;
        } catch (SQLException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } finally {
            try {
                if (null != statement) {
                    statement.close();
                }
                if (null != connection) {
                    connection.close();
                }
            } catch (SQLException e) {
                throw new RuntimeException(e);
            }
        }
        return 0;

JDBC編程之查詢

        String driver = "com.mysql.jdbc.Driver";
        String url = "jdbc:mysql://localhost:3306/test);;
        String username = "root";
        String password = "1234";
        Connection connection = null;
        Statement statement = null;
        try{
            Class.forName(driver);
            connection = DriverManager.getConnection(url, username, password);
            String sql = "select nickname,comment,age from users";
            statement = connection.createStatement();
            ResultSet resultSet = statement.executeQuery(sql);
            List usersList = new ArrayList<>();
            while (resultSet.next()) {
                Users users = new Users();
                users.setNickname(resultSet.getString(1));
                users.setComment(resultSet.getString(2));
                users.setAge(resultSet.getInt(3));
                usersList.add(users);
            }
            return usersList;
        } catch (SQLException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        } finally {
            try {
                if (null != statement) {
                    statement.close();
                }
                if (null != connection) {
                    connection.close();
                }
            } catch (SQLException e) {
                throw new RuntimeException(e);
            }
        }
        return null;

上面給出了我們?cè)趥鹘y(tǒng)JDBC編程中的兩個(gè)案例,可以看到傳統(tǒng)JDBC的很多缺點(diǎn),當(dāng)然在實(shí)際項(xiàng)目中我們可能不會(huì)這么原始的進(jìn)行數(shù)據(jù)庫(kù)開發(fā),可能會(huì)對(duì)JDBC進(jìn)行一定的封裝,方便我們的使用。Spring 官方為了簡(jiǎn)化JDBC的開發(fā)也發(fā)布了JDBCTemplate,下面我們就看一下它是如何簡(jiǎn)化開發(fā)的,以及模板方法模式在其中的應(yīng)用

JDBCTemplate是個(gè)啥,它到底簡(jiǎn)化了什么?

從JDBCTemplate的名字我們就不難看出,它簡(jiǎn)化了我們JDBC的開發(fā),而且很可能大量應(yīng)用了模板方法模式,它到底為我們提供了什么?它提供了與平臺(tái)無(wú)光的異常處理機(jī)制。使用過(guò)原生JDBC開發(fā)的同學(xué)可能有經(jīng)歷,幾乎所有的操作代碼都需要我們強(qiáng)制捕獲異常,但是在出現(xiàn)異常時(shí)我們往往無(wú)法通過(guò)異常讀懂錯(cuò)誤。Spring解決了我們的問(wèn)題它提供了多個(gè)數(shù)據(jù)訪問(wèn)異常,并且分別描述了他們拋出時(shí)對(duì)應(yīng)的問(wèn)題,同時(shí)對(duì)異常進(jìn)行了包裝不強(qiáng)制要求我們進(jìn)行捕獲,同時(shí)它為我們提供了數(shù)據(jù)訪問(wèn)的模板化,從上面的傳統(tǒng)JDBC編程我們可以發(fā)現(xiàn),很多操作其實(shí)是重復(fù)的不變得比如事務(wù)控制、資源的獲取關(guān)閉以及異常處理等,同時(shí)結(jié)果集的處理實(shí)體的綁定,參數(shù)的綁定這些東西都是特有的。因此Spring將數(shù)據(jù)訪問(wèn)過(guò)程中固定部分和可變部分劃分為了兩個(gè)不同的類(Template)和回調(diào)(Callback),模板處理過(guò)程中不變得部分,回調(diào)處理自定義的訪問(wèn)代碼;下面我們具體通過(guò)源碼來(lái)學(xué)學(xué)習(xí)一下

模板方法模式在JDBCTemplate中的應(yīng)用

我所使用的版本是5.1.5.RELEASE

打開JdbcTemplate類(我這里就不截圖了,截圖可能不清晰我直接將代碼copy出來(lái)):

JdbcTemplate

    public class JdbcTemplate extends JdbcAccessor implements JdbcOperations {
        //查詢前綴
        private static final String RETURN_RESULT_SET_PREFIX = "#result-set-";
        //計(jì)數(shù)前綴
        private static final String RETURN_UPDATE_COUNT_PREFIX = "#update-count-";
        //是否跳過(guò)警告
        private boolean ignoreWarnings = true;
        //查詢大小
        private int fetchSize = -1;
        //最大行
        private int maxRows = -1;
        //查詢超時(shí)
        private int queryTimeout = -1;
        //是否跳過(guò)結(jié)果集處理
        private boolean skipResultsProcessing = false;
        //是否跳過(guò)非公共結(jié)果集處理
        private boolean skipUndeclaredResults = false;
        //map結(jié)果集是否大小寫敏感
        private boolean resultsMapCaseInsensitive = false;
    
        public JdbcTemplate() {
        }
        //調(diào)用父類方法設(shè)置數(shù)據(jù)源和其他參數(shù)
        public JdbcTemplate(DataSource dataSource) {
            this.setDataSource(dataSource);
            this.afterPropertiesSet();
        }
        //調(diào)用父類方法設(shè)置數(shù)據(jù)源,懶加載策略和其他參數(shù)
        public JdbcTemplate(DataSource dataSource, boolean lazyInit) {
            this.setDataSource(dataSource);
            this.setLazyInit(lazyInit);
            this.afterPropertiesSet();
        }
    }

JdbcTemplate 繼承了JdbcAccessor實(shí)現(xiàn)了JdbcOperations,JdbcAccessor主要封裝了數(shù)據(jù)源的操作,JdbcOperations主要定義了一些操作接口。我們一起看一下JdbcOperations類;

    public abstract class JdbcAccessor implements InitializingBean {
        protected final Log logger = LogFactory.getLog(this.getClass());
        //數(shù)據(jù)源
        @Nullable
        private DataSource dataSource;
        //異常翻譯
        @Nullable
        private volatile SQLExceptionTranslator exceptionTranslator;
        //懶加載策略
        private boolean lazyInit = true;
        public JdbcAccessor() {
        }
        public void setDataSource(@Nullable DataSource dataSource) {
            this.dataSource = dataSource;
        }
        @Nullable
        public DataSource getDataSource() {
            return this.dataSource;
        }
        protected DataSource obtainDataSource() {
            DataSource dataSource = this.getDataSource();
            Assert.state(dataSource != null, "No DataSource set");
            return dataSource;
        }
        public void setDatabaseProductName(String dbName) {
            this.exceptionTranslator = new SQLErrorCodeSQLExceptionTranslator(dbName);
        }
        public void setExceptionTranslator(SQLExceptionTranslator exceptionTranslator) {
            this.exceptionTranslator = exceptionTranslator;
        }
    }

之所以前面提到spring讓我們更方便的處理異常就是這里他包裝了一個(gè)SQLExceptionTranslator,其他的代碼都是做數(shù)據(jù)源的檢查之類的設(shè)置數(shù)據(jù)源,我們看一下其中g(shù)etExceptionTranslator()方法

    public SQLExceptionTranslator getExceptionTranslator() {
        SQLExceptionTranslator exceptionTranslator = this.exceptionTranslator;
        if (exceptionTranslator != null) {
            return exceptionTranslator;
        } else {
            synchronized(this) {
                SQLExceptionTranslator exceptionTranslator = this.exceptionTranslator;
                if (exceptionTranslator == null) {
                    DataSource dataSource = this.getDataSource();
                    if (dataSource != null) {
                        exceptionTranslator = new SQLErrorCodeSQLExceptionTranslator(dataSource);
                    } else {
                        exceptionTranslator = new SQLStateSQLExceptionTranslator();
                    }
                    this.exceptionTranslator = (SQLExceptionTranslator)exceptionTranslator;
                }
                return (SQLExceptionTranslator)exceptionTranslator;
            }
        }
    }

這是一個(gè)標(biāo)準(zhǔn)的單例模式,我們?cè)趯W(xué)習(xí)模板方法模式的路途中有捕獲了一個(gè)野生的單例;我們繼續(xù)看JdbcOperations接口我們調(diào)其中一個(gè)接口進(jìn)行解析;

    @Nullable
     T execute(StatementCallback var1) throws DataAccessException;

StatementCallback 接口

    @FunctionalInterface
    public interface StatementCallback<T> {
        @Nullable
        T doInStatement(Statement var1) throws SQLException, DataAccessException;
    }

execute實(shí)現(xiàn)

    @Nullable
    public  T execute(StatementCallback action) throws DataAccessException {
        //參數(shù)檢查
        Assert.notNull(action, "Callback object must not be null");
        //獲取連接
        Connection con = DataSourceUtils.getConnection(this.obtainDataSource());
        Statement stmt = null;
        Object var11;
        try {
            //創(chuàng)建一個(gè)Statement
            stmt = con.createStatement();
            //設(shè)置查詢超時(shí)時(shí)間,最大行等參數(shù)(就是一開始那些成員變量)
            this.applyStatementSettings(stmt);
            //執(zhí)行回調(diào)方法獲取結(jié)果集
            T result = action.doInStatement(stmt);
            //處理警告
            this.handleWarnings(stmt);
            var11 = result;
        } catch (SQLException var9) {
            //出現(xiàn)錯(cuò)誤優(yōu)雅退出
            String sql = getSql(action);
            JdbcUtils.closeStatement(stmt);
            stmt = null;
            DataSourceUtils.releaseConnection(con, this.getDataSource());
            con = null;
            throw this.translateException("StatementCallback", sql, var9);
        } finally {
            JdbcUtils.closeStatement(stmt);
            DataSourceUtils.releaseConnection(con, this.getDataSource());
        }
        return var11;
    }

這一個(gè)方法可謂是展現(xiàn)的淋漓盡致,這是一個(gè)典型的模板方法+回調(diào)模式,我們不需要再寫過(guò)多的重復(fù)代碼只需要實(shí)現(xiàn)自己獲取result的方法就好(StatementCallback)事實(shí)上我們自己也不需要實(shí)現(xiàn)這個(gè)方法,繼續(xù)向上看,我們是如何調(diào)用execute方法的,以查詢?yōu)槔?我們看他是如何一步步調(diào)用的:

查詢方法

    public List findAll() {
        JdbcTemplate jdbcTemplate = DataSourceConfig.getTemplate();
        String sql = "select nickname,comment,age from users";
        return jdbcTemplate.query(sql, new BeanPropertyRowMapper(Users.class));
    }

query實(shí)現(xiàn)

    public  List query(String sql, RowMapper rowMapper) throws DataAccessException {
        return (List)result(this.query((String)sql, (ResultSetExtractor)(new RowMapperResultSetExtractor(rowMapper))));
    }

這里的RowMapper是負(fù)責(zé)將結(jié)果集中一行的數(shù)據(jù)映射成實(shí)體返回,用到了反射技術(shù),這里就不展開了,有興趣的同學(xué)可以自己打開源碼閱讀,繼續(xù)向下:

query實(shí)現(xiàn)

@Nullable
    public  T query(final String sql, final ResultSetExtractor rse) throws DataAccessException {
        Assert.notNull(sql, "SQL must not be null");
        Assert.notNull(rse, "ResultSetExtractor must not be null");
        if (this.logger.isDebugEnabled()) {
            this.logger.debug("Executing SQL query [" + sql + "]");
        }
        //實(shí)現(xiàn)回調(diào)接口
        class QueryStatementCallback implements StatementCallback<T>, SqlProvider {
            QueryStatementCallback() {
            }
            @Nullable
            public T doInStatement(Statement stmt) throws SQLException {
                ResultSet rs = null;
                Object var3;
                try {
                    //這里真正的執(zhí)行我們的sql語(yǔ)句
                    rs = stmt.executeQuery(sql);
                    //處理對(duì)象映射
                    var3 = rse.extractData(rs);
                } finally {
                    JdbcUtils.closeResultSet(rs);
                }
                return var3;
            }
            public String getSql() {
                return sql;
            }
        }
        //調(diào)用execute接口
        return this.execute((StatementCallback)(new QueryStatementCallback()));
    }

看到這里相信你也不得拍手稱奇,Spring處理的非常巧妙,請(qǐng)繼續(xù)向下看:

update詳解

    protected int update(PreparedStatementCreator psc, @Nullable PreparedStatementSetter pss) throws DataAccessException {
        this.logger.debug("Executing prepared SQL update");
        return updateCount((Integer)this.execute(psc, (ps) -> {
            Integer var4;
            try {
                if (pss != null) {
                    pss.setValues(ps);
                }
                int rows = ps.executeUpdate();
                if (this.logger.isTraceEnabled()) {
                    this.logger.trace("SQL update affected " + rows + " rows");
                }
                var4 = rows;
            } finally {
                if (pss instanceof ParameterDisposer) {
                    ((ParameterDisposer)pss).cleanupParameters();
                }
            }
            return var4;
        }));
    }

為什么我要把update函數(shù)拎出來(lái)講了,因?yàn)閡pdate這里使用了lambda函數(shù),回想我們StatementCallback定義只有一個(gè)方法的接口,他就是一個(gè)函數(shù)是接口,所以他是一個(gè)函數(shù)式接口,所以這里直接使用lambda語(yǔ)法,lambda函數(shù)允許你直接內(nèi)連,為函數(shù)接口的抽象方法提供實(shí)現(xiàn),并且整個(gè)表達(dá)式作為函數(shù)接口的一個(gè)實(shí)例。我們?cè)谄綍r(shí)學(xué)習(xí)中可能知道了lambda語(yǔ)法但是可能使用的較少,或者不知道如何用于實(shí)戰(zhàn),那么多閱讀源碼一定可以提升你的實(shí)戰(zhàn)能力。 我們可以看到JDBCTemplate使用了很多回調(diào)。為什么要用回調(diào)(Callback));如果父類有多個(gè)抽象方法,子類需要全部實(shí)現(xiàn)這樣特別麻煩,而有時(shí)候某個(gè)子類只需要定制父類中的某一個(gè)方法該怎么辦呢?這個(gè)時(shí)候就要用到Callback回調(diào)了就可以完美解決這個(gè)問(wèn)題,可以發(fā)現(xiàn)JDBCTemplate并沒(méi)有完全拘泥于模板方法,非常靈活。我們?cè)趯?shí)際開發(fā)中也可以借鑒這種方法。

模板方法模式的更多應(yīng)用

事實(shí)上很多有關(guān)生命周期的類都用到了模板方法模式,最典型的也是可能我們最熟悉的莫過(guò)于Servlet了,廢話不多說(shuō)上源碼

    public abstract class HttpServlet extends GenericServlet
    {
    }

HttpServlet的所有方法,我們看到HttpServlet繼承了GenericServlet,我們繼續(xù)看:

public abstract class GenericServlet 
    implements Servlet, ServletConfig, java.io.Serializable
{
    private static final String LSTRING_FILE = "javax.servlet.LocalStrings";
    private static ResourceBundle lStrings =
        ResourceBundle.getBundle(LSTRING_FILE);

    private transient ServletConfig config;
    
    public GenericServlet() { }
    
    //沒(méi)有實(shí)現(xiàn)鉤子
    public void destroy() {
    }
    
    public String getInitParameter(String name) {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getInitParameter(name);
    }
    
    public Enumeration getInitParameterNames() {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getInitParameterNames();
    }   
     
    public ServletConfig getServletConfig() {
	return config;
    }
    
    public ServletContext getServletContext() {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getServletContext();
    }

    public String getServletInfo() {
	return "";
    }

    public void init(ServletConfig config) throws ServletException {
	this.config = config;
	this.init();
    }
    
    public void init() throws ServletException {

    }
    
    public void log(String msg) {
	getServletContext().log(getServletName() + ": "+ msg);
    }
  
    public void log(String message, Throwable t) {
	getServletContext().log(getServletName() + ": " + message, t);
    }
    
    public abstract void service(ServletRequest req, ServletResponse res)
	throws ServletException, IOException;
    
    public String getServletName() {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getServletName();
    }
}

可以看到這就是個(gè)典型的模板方法類蠻,而且鉤子函數(shù)也在這里展現(xiàn)的淋漓盡致,如init、destroy方法等,JDK中很多類都是用了模板方法等著你發(fā)現(xiàn)哦。

模板方法模式在Vue.js中的應(yīng)用

模板方法模式在其他語(yǔ)言中也有實(shí)現(xiàn)比如Vue.js、React中;比如Vue生命周期肯定使用了模板方法,我就不對(duì)源碼展開分析了。

總結(jié)

設(shè)計(jì)模式在Spring中得到了大量的應(yīng)用,感興趣的同學(xué)可以看看Spring源碼加以學(xué)習(xí),如果你覺(jué)得我寫的還不錯(cuò)的話點(diǎn)個(gè)贊吧,如果你發(fā)現(xiàn)了錯(cuò)誤,或者不好的地方也可以及時(shí)告訴我加以改正,謝謝!您的贊賞和批評(píng)是進(jìn)步路上的好伙伴。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/7162.html

相關(guān)文章

  • 模板方法模式JDBCTemplate

    摘要:模板方法模式的更多應(yīng)用事實(shí)上很多有關(guān)生命周期的類都用到了模板方法模式,最典 將大象裝進(jìn)冰箱需要三步,那么老虎了?如何優(yōu)雅的將大象裝進(jìn)冰箱? 把大象裝進(jìn)冰箱 Step 大象 老虎 ... First 打開冰箱門 打開冰箱門 打開冰箱門 Second 把大象放進(jìn)去 把老虎放進(jìn)去 ... Third 關(guān)閉冰箱門 關(guān)閉冰箱門 關(guān)閉冰箱門 大象類 public ...

    番茄西紅柿 評(píng)論0 收藏0
  • 模板方法模式JDBCTemplate

    摘要:模板方法模式的更多應(yīng)用事實(shí)上很多有關(guān)生命周期的類都用到了模板方法模式,最典 將大象裝進(jìn)冰箱需要三步,那么老虎了?如何優(yōu)雅的將大象裝進(jìn)冰箱? 把大象裝進(jìn)冰箱 Step 大象 老虎 ... First 打開冰箱門 打開冰箱門 打開冰箱門 Second 把大象放進(jìn)去 把老虎放進(jìn)去 ... Third 關(guān)閉冰箱門 關(guān)閉冰箱門 關(guān)閉冰箱門 大象類 public ...

    MoAir 評(píng)論0 收藏0
  • 動(dòng)手搭建后端框架-Velocity模板引擎的應(yīng)用

    摘要:目錄建造者模式應(yīng)用。其實(shí)不用也可以,因?yàn)椴皇呛軓?fù)雜,只是為了復(fù)習(xí)一下所學(xué)過(guò)的設(shè)計(jì)模式知識(shí)目錄工廠模式應(yīng)用。 為了提高開發(fā)效率,通常會(huì)想辦法把一些模式固定的重復(fù)性的勞動(dòng)抽取出來(lái),以后再使用的時(shí)候,拿來(lái)主義就可以了。這樣既可以提高開發(fā)效率,又降低了出錯(cuò)的風(fēng)險(xiǎn)。 這一思想在我們的日常工作中可以說(shuō)隨處可見,我們完成一項(xiàng)復(fù)雜的工程,并不需要面面俱到什么都自己寫,我們完全可以利用第三方的jar包讓...

    villainhr 評(píng)論0 收藏0
  • 詳解Spring中的9種設(shè)計(jì)模式「記得收藏」

    摘要:簡(jiǎn)單工廠模式的實(shí)質(zhì)是由一個(gè)工廠類根據(jù)傳入的參數(shù),動(dòng)態(tài)決定應(yīng)該創(chuàng)建哪一個(gè)產(chǎn)品類。中的就是簡(jiǎn)單工廠模式的體現(xiàn),根據(jù)傳入一個(gè)唯一的標(biāo)識(shí)來(lái)獲得對(duì)象,但是否是在傳入?yún)?shù)后創(chuàng)建還是傳入?yún)?shù)前創(chuàng)建這個(gè)要根據(jù)具體情況來(lái)定。中的就是典型的工廠方法模式。 showImg(https://segmentfault.com/img/bVbwbd9?w=640&h=492); 一. 簡(jiǎn)單工廠又叫做靜態(tài)工廠方法(...

    Dean 評(píng)論0 收藏0
  • 這些Spring中的設(shè)計(jì)模式,你都知道嗎?

    摘要:簡(jiǎn)單工廠模式的實(shí)質(zhì)是由一個(gè)工廠類根據(jù)傳入的參數(shù),動(dòng)態(tài)決定應(yīng)該創(chuàng)建哪一個(gè)產(chǎn)品類。中的就是簡(jiǎn)單工廠模式的體現(xiàn),根據(jù)傳入一個(gè)唯一的標(biāo)識(shí)來(lái)獲得對(duì)象,但是否是在傳入?yún)?shù)后創(chuàng)建還是傳入?yún)?shù)前創(chuàng)建這個(gè)要根據(jù)具體情況來(lái)定。 設(shè)計(jì)模式作為工作學(xué)習(xí)中的枕邊書,卻時(shí)常處于勤說(shuō)不用的尷尬境地,也不是我們時(shí)常忘記,只是一直沒(méi)有記憶。 Spring作為業(yè)界的經(jīng)典框架,無(wú)論是在架構(gòu)設(shè)計(jì)方面,還是在代碼編寫方面,都堪...

    LeviDing 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<