從shiro源碼角度學習建造者設計模式

緒論

今天就和你們分享一下shiro源碼裏面使用到的建造者模式。在介紹建造者模式相關知識以前,咱們先來看一段例子分析。設計模式

從一個簡單的例子提及

咱們在使用shiro獲取登陸用戶的時候,好比使用ini文件配置用戶角色權限,咱們能夠這樣寫:session

        Factory<SecurityManager> factory = new IniSecurityManagerFactory("classpath:demo01_getstarted.ini");
        SecurityManager securityManager = factory.getInstance();
        SecurityUtils.setSecurityManager(securityManager);
        Subject currentUser = SecurityUtils.getSubject();
複製代碼

SecurityUtils.getSubject()是獲取當前用戶,咱們跳進去看一下,是怎麼獲取的app

 public static Subject getSubject() {
        Subject subject = ThreadContext.getSubject();
        if (subject == null) {
            subject = (new Subject.Builder()).buildSubject();
            ThreadContext.bind(subject);
        }
        return subject;
    }
複製代碼

上面代碼,首先從ThreadContext裏面獲取Subject,獲取不到就建立,咱們看看建立過程,跳進去看看(new Subject.Builder()).buildSubject()幹了什麼。 這裏不貼過長代碼了,Builder是Subject的靜態內部類,用來完成Subject的複雜建立過程,使得調用就無需關注Subject怎麼建立的,只要知道怎麼獲取Subject就能夠了。 咱們看一下,Builder類裏面幹了什麼事情框架

 public static class Builder {

        private final SubjectContext subjectContext;

        private final SecurityManager securityManager;

        public Builder() {
            this(SecurityUtils.getSecurityManager());
        }

        public Builder(SecurityManager securityManager) {
            if (securityManager == null) {
                throw new NullPointerException("SecurityManager method argument cannot be null.");
            }
            this.securityManager = securityManager;
            this.subjectContext = newSubjectContextInstance();
            if (this.subjectContext == null) {
                throw new IllegalStateException("Subject instance returned from 'newSubjectContextInstance' " +
                        "cannot be null.");
            }
            this.subjectContext.setSecurityManager(securityManager);
        }

        protected SubjectContext newSubjectContextInstance() {
            return new DefaultSubjectContext();
        }

        protected SubjectContext getSubjectContext() {
            return this.subjectContext;
        }

        public Builder sessionId(Serializable sessionId) {
            if (sessionId != null) {
                this.subjectContext.setSessionId(sessionId);
            }
            return this;
        }

        public Builder host(String host) {
            if (StringUtils.hasText(host)) {
                this.subjectContext.setHost(host);
            }
            return this;
        }

        public Builder session(Session session) {
            if (session != null) {
                this.subjectContext.setSession(session);
            }
            return this;
        }

        public Builder principals(PrincipalCollection principals) {
            if (principals != null && !principals.isEmpty()) {
                this.subjectContext.setPrincipals(principals);
            }
            return this;
        }

        public Builder sessionCreationEnabled(boolean enabled) {
            this.subjectContext.setSessionCreationEnabled(enabled);
            return this;
        }

        public Builder authenticated(boolean authenticated) {
            this.subjectContext.setAuthenticated(authenticated);
            return this;
        }

        public Builder contextAttribute(String attributeKey, Object attributeValue) {
            if (attributeKey == null) {
                String msg = "Subject context map key cannot be null.";
                throw new IllegalArgumentException(msg);
            }
            if (attributeValue == null) {
                this.subjectContext.remove(attributeKey);
            } else {
                this.subjectContext.put(attributeKey, attributeValue);
            }
            return this;
        }
        // 建立Subject
        public Subject buildSubject() {
            return this.securityManager.createSubject(this.subjectContext);
        }
    }
複製代碼

咱們能夠看到,Builder完成了Subject一些列的複雜建立過程,包括sesson、sessionid等的設置等過程。最後會調用buildSubject()方法來建立Subject。 咱們就不繼續深刻看this.securityManager.createSubject(this.subjectContext)裏面幹了什麼事了。 回到前面,咱們能夠看到Subject對象會包含許多的當前用戶信息,可是這些信息框架底層都替咱們設置好了,咱們只要獲取Subject對象便可,調用Subject currentUser = SecurityUtils.getSubject();就能夠了。 這樣看來,是否是特別方便。這些都歸功於建造這模式的使用。學習

建造這模式基本概念

經過分析shiro源碼,咱們對建造這模式有了一個初步的感知,目前你們能夠理解成建立複雜對象由一個建造類來完成。那麼,咱們就來學習一下建造者模式的一些理論知識,相信你們結合上面的例子分析,不會感到吃力乏味的。ui

什麼是建造者模式

建造者模式又稱爲生成器模式。
建造者模式:它能夠將複雜對象的建造過程抽象出來(抽象類別),使這個抽象過程的不一樣實現方法能夠構造出不一樣表現(屬性)的對象。 參考維基百度百科:生成器模式
可能有些讀者看了這句話迷迷糊糊的,那麼這句話是什麼意思呢?結合上面的例子,咱們能夠知道Subject對象有不少屬性,經過SubjectContext對象來保存的,最後建立Subject對象的時候,傳遞SubjectContext就能夠了。這樣好比,咱們建立兩個有不一樣屬性值的Subject:this

        Subject user_1 = new Subject.Builder().authenticated(false).buildSubject();
        Subject user_2 = new Subject.Builder().authenticated(true).buildSubject();
複製代碼

咱們建立Subject的不一樣實現方法,好比authenticated()一個設置true,一個設置false,咱們就構造出來了有不一樣表現(屬性)的Subject對象了,一個是受權,一個是未受權。編碼

爲何使用建造者模式

使用建造者模式能夠屏蔽複雜的對象建立過程,對象的建立對調用者來講是透明的,這樣就使得程序耦合度下降,程序可讀性和維護性提升了。spa

結束語

對於設計模式,你們不要機械地學習,我的以爲主要理解其設計思想和好處就能夠了。在編碼過程當中你們不要刻意去使用它,而是去思考如何運用它更好地設計代碼。這樣,通過時間的推磨,你們就能夠真正作到把設計模式融會貫通,手到拈來。設計

相關文章
相關標籤/搜索