關(guān)于 swagger plugin 的簡(jiǎn)單介紹與運(yùn)用

之前用 swagger 都是簡(jiǎn)單的使用,沒有考慮過 swagger 具體的實(shí)現(xiàn),不過這一篇文章也并沒有介紹 swagger 的具體實(shí)現(xiàn),而是從 swagger 的 plugin 切入,研究自定義 swagger plugin 的問題
我們知道 swagger 集成進(jìn) spring 或 spring-boot 是通過@EnableSwagger2或者是在 spring-servlet.xml 中增加一個(gè)bean 的配置(具體這些都可以去網(wǎng)上隨便搜搜都有了),然后通過點(diǎn)擊進(jìn)@EnableSwagger2進(jìn)去可以看見這個(gè)注解的具體實(shí)現(xiàn)過程,這里有一篇博客,介紹的挺詳細(xì)的

http://blog.csdn.net/qq_25615395/article/details/70229139

然后這里介紹一下這篇文章里沒有涉及到的,或者說是沒有展開講的內(nèi)容,關(guān)于plugin 的使用.
swagger 通過@EnableSwagger2這個(gè)注解進(jìn)去(或者 spring-servlet.xml 中添加的 bean),會(huì)到一個(gè) SpringSwaggerConfig 的類中,這個(gè)類是 swagger 的配置類,比較重要的這里面配置了一個(gè) swaggerPluginAdapter的 bean

public class SwaggerPluginAdapter implements ApplicationListener<ContextRefreshedEvent>

這個(gè)bean 實(shí)現(xiàn)了ApplicationListener 這個(gè)接口的onApplicationEvent方法

@Override
  public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {
    if (initialized.compareAndSet(false, true)) {
      log.info("Context refreshed");
      ApplicationContext applicationContext = contextRefreshedEvent.getApplicationContext();

      Map<String, SwaggerSpringMvcPlugin> plugins = BeanFactoryUtils.beansOfTypeIncludingAncestors(
              applicationContext,
              SwaggerSpringMvcPlugin.class);

      if (plugins.isEmpty()) {
        log.info("Did not find any SwaggerSpringMvcPlugins so creating a default one");
        new SwaggerSpringMvcPlugin(springSwaggerConfig)
                .build()
                .initialize();
      } else {
        log.info("Found custom SwaggerSpringMvcPlugins");

        for (Map.Entry<String, SwaggerSpringMvcPlugin> entry : plugins.entrySet()) {
          if (entry.getValue().isEnabled()) {
            log.info("initializing plugin bean {}", entry.getKey());
            entry.getValue()
                    .build()
                    .initialize();
          } else {
            log.info("Skipping initializing disabled plugin bean {}", entry.getKey());
          }
        }
      }
    } else {
      log.info("Skipping SwaggerSpringMvcPlugin initialization already initialized!");
    }
  }

這段代碼的核心大概是在應(yīng)用的上下文中去找,有沒有滿足設(shè)定條件的 plugin,如果沒有,則 swagger 會(huì)自己建一個(gè)默認(rèn)的,這個(gè)默認(rèn)的也就是平時(shí)我們使用的,所以在這里,我們就可以做一些自定義的工作.做這些工作,需要?jiǎng)?chuàng)建一個(gè)繼承SwaggerSpringMvcPlugin的類和一個(gè)自定義的SwaggerConfig 類(這個(gè)類也就是在 spring-servlet.xml 中創(chuàng)建成 bean的那個(gè)類)
首先我們需要?jiǎng)?chuàng)建一個(gè)繼承于SwaggerSpringMvcPlugin的類,并將這個(gè)類注射成應(yīng)用上下文中的bean,在創(chuàng)建這個(gè)bean 的時(shí)候,需要注意的是得要注入SwaggerSpringConfig 這個(gè)類.當(dāng)在應(yīng)用上下文中注入了SwaggerSpringMvcPlugin這個(gè) bean 之后,在上面那段代碼中就可以監(jiān)測(cè)到上下文中有自定義的 plugin,則就會(huì)跑到下面那部分.
所以自定義的 plugin 類需要重寫父類的 build 方法和initialize方法,這兩個(gè)方法中的就是 plugin 類的核心代碼實(shí)現(xiàn)

public SwaggerSpringMvcPlugin build() {
    if (initialized.compareAndSet(false, true)) {
      configure();
      buildSwaggerGlobalSettings();
      buildApiListingReferenceScanner();
      buildSwaggerApiResourceListing();
    }
    return this;
  }

protected void initialize() {
    if (enabled) {
      this.build().swaggerApiResourceListing.initialize();
    }
  }

后面部分的代碼其實(shí)沿著源碼一路點(diǎn)下去,就可以看見具體的實(shí)現(xiàn)過程,這里也大致講一下.有兩個(gè)部分,一個(gè)是 ApiResourceListing,一個(gè)是 ApiListingReferenceScanner,前面是對(duì)使用了@Api 注解的類進(jìn)行掃描,并解析,后面一個(gè)是從前面解析的類中,把使用了@ApiOperation 注解的方法找出來并解析,所以很容易想到,如果需要加入特殊的類掃描注解,可以使用自定義的ApiResourceListing,如果需要加入特殊的方法的注解,就自定義 scanner 就好了.包括swagger 中的各個(gè)注解,都是從 ApiListingReferenceScanner這個(gè)類中引申開來,這里貼一下源碼中關(guān)于這部分的代碼結(jié)構(gòu),感興趣的可以點(diǎn)進(jìn)去一個(gè)一個(gè)的看看,這些也都是可以由用戶自己自定義的

額,不會(huì)貼圖,湊合看吧,都很好找到的...

通過這一些的方法出來之后,會(huì)到ApiResourceListing這個(gè)類中

swaggerCache.addSwaggerResourceListing(swaggerGroup, resourceListing);

的這一句代碼,這里就是將解析好的resourceListing,添加到 swaggerCache 之中去,swaggerGroup 是用戶通過配置文件添加的組名,這個(gè)沒啥,關(guān)鍵是 swaggerCache,我之前也不是很懂這個(gè)到底是干嘛的,不過看源碼的時(shí)候,可以知道resourceListing里面已經(jīng)是把所有帶有注解的類和方法的有用信息都提取出來保存在一個(gè)model 中,所以估計(jì) swaggerCache 應(yīng)該是存放這些信息并在其他地方通過這個(gè) cache 獲取到這些信息并創(chuàng)建我們熟知的 swagger 頁面的,去查了一下大概,差不多也就是這個(gè)意思,這個(gè)就屬于 swagger 實(shí)現(xiàn)原理的后一半部分了,這里也就不贅述了,還是那句話,感興趣可以去 Google 一下,應(yīng)該都有的.
這里主要也就是講一個(gè)思路,所以具體代碼沒有貼,貼的也都是 swagger 自帶的實(shí)現(xiàn),我覺得如果真的要做,還是要自己動(dòng)手嘗試一下的.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市厌衔,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,496評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件绊率,死亡現(xiàn)場(chǎng)離奇詭異鉴竭,居然都是意外死亡干毅,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來榜掌,“玉大人优妙,你說我怎么就攤上這事≡髡耍” “怎么了套硼?”我有些...
    開封第一講書人閱讀 162,632評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)胞皱。 經(jīng)常有香客問我邪意,道長(zhǎng),這世上最難降的妖魔是什么反砌? 我笑而不...
    開封第一講書人閱讀 58,180評(píng)論 1 292
  • 正文 為了忘掉前任雾鬼,我火速辦了婚禮,結(jié)果婚禮上宴树,老公的妹妹穿的比我還像新娘策菜。我一直安慰自己,他們只是感情好酒贬,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評(píng)論 6 388
  • 文/花漫 我一把揭開白布又憨。 她就那樣靜靜地躺著,像睡著了一般锭吨。 火紅的嫁衣襯著肌膚如雪蠢莺。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,165評(píng)論 1 299
  • 那天零如,我揣著相機(jī)與錄音躏将,去河邊找鬼。 笑死考蕾,一個(gè)胖子當(dāng)著我的面吹牛祸憋,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播辕翰,決...
    沈念sama閱讀 40,052評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼夺衍,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了喜命?” 一聲冷哼從身側(cè)響起沟沙,我...
    開封第一講書人閱讀 38,910評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎壁榕,沒想到半個(gè)月后矛紫,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,324評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡牌里,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評(píng)論 2 332
  • 正文 我和宋清朗相戀三年颊咬,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了务甥。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,711評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡喳篇,死狀恐怖敞临,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情麸澜,我是刑警寧澤挺尿,帶...
    沈念sama閱讀 35,424評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站炊邦,受9級(jí)特大地震影響编矾,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜馁害,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評(píng)論 3 326
  • 文/蒙蒙 一窄俏、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧碘菜,春花似錦凹蜈、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至吊骤,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間静尼,已是汗流浹背白粉。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留鼠渺,地道東北人鸭巴。 一個(gè)月前我還...
    沈念sama閱讀 47,722評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像拦盹,于是被迫代替她去往敵國(guó)和親鹃祖。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理普舆,服務(wù)發(fā)現(xiàn)恬口,斷路器,智...
    卡卡羅2017閱讀 134,651評(píng)論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,804評(píng)論 6 342
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,077評(píng)論 25 707
  • 南下秦中八百里沼侣,高風(fēng)吹過劍門關(guān)祖能。 浮星慢涌出重嶺,攆月疾行照闊川蛾洛。 勿道如今巴水勁养铸,猶說自古蜀山難。 吟詩不取梓州...
    長(zhǎng)安舊人閱讀 495評(píng)論 3 15
  • 第一次聽說你鳞滨,是在小學(xué)四年級(jí)洞焙。你在一班,我在三班太援。 那年運(yùn)動(dòng)會(huì)闽晦,我們班都在議論一班轉(zhuǎn)來一位新同學(xué),跳高非常厲害...
    ssssssyyyyyylll閱讀 711評(píng)論 7 6