黄色网页视频 I 影音先锋日日狠狠久久 I 秋霞午夜毛片 I 秋霞一二三区 I 国产成人片无码视频 I 国产 精品 自在自线 I av免费观看网站 I 日本精品久久久久中文字幕5 I 91看视频 I 看全色黄大色黄女片18 I 精品不卡一区 I 亚洲最新精品 I 欧美 激情 在线 I 人妻少妇精品久久 I 国产99视频精品免费专区 I 欧美影院 I 欧美精品在欧美一区二区少妇 I av大片网站 I 国产精品黄色片 I 888久久 I 狠狠干最新 I 看看黄色一级片 I 黄色精品久久 I 三级av在线 I 69色综合 I 国产日韩欧美91 I 亚洲精品偷拍 I 激情小说亚洲图片 I 久久国产视频精品 I 国产综合精品一区二区三区 I 色婷婷国产 I 最新成人av在线 I 国产私拍精品 I 日韩成人影音 I 日日夜夜天天综合

FindBugs檢測器實現(xiàn)(2)

系統(tǒng) 2581 0

FindBugs 檢測器實現(xiàn)(1) 中提到,F(xiàn)indBugs 主要有5類檢測器,這篇日志介紹下FindBugs在類、方法、字段結(jié)構(gòu)上的檢測器實現(xiàn)。前面提到基于棧和簡單的字節(jié)碼模式要繼承OpcodeStackDetector類,并實現(xiàn)sawOpcode方法用來檢測每一個字節(jié)碼。

一般在寫檢測器之前,我們應(yīng)該有一個自己想要檢測的代碼模式,但作為學習,這里使用一些簡單的模式作為例子:

  1. 重寫了equals函數(shù),卻沒有重寫hashCode函數(shù)。
  2. 經(jīng)hashCode函數(shù)拼寫為hashcode。

FindBugs在類、方法、字段層面的檢測器實現(xiàn)要繼承PreorderDetector類,簡單介紹些該類:

?

PreorderVisitor

PreorderVisitor 繼承了 BetterVisitor,并在 BetterVisitor的基礎(chǔ)之上,添加了大量的方法用來從當前訪問的類、方法或字段中獲取信息,另外該類還提供了一些visitAfter方法,用來實現(xiàn)訪問上的次序,在下面的例子中,我們會看到它的作用。像下面的方法 PreorderVisitor 中存在太多,這里就不一一列出,可能有些方法不能直接明了的知道其作用,大多可能是我們還不懂java的一些功能的緣故:

      
        public
      
      
         FieldDescriptor getFieldDescriptor()


      
      
        public
      
      
         MethodDescriptor getMethodDescriptor()


      
      
        public
      
      
         ClassDescriptor getClassDescriptor()


      
      
        public
      
      
         @SlashedClassName  String getClassName()


      
      
        public
      
      
        void
      
      
         visitAfter(Code obj)


      
      
        public
      
      
        void
      
       visitAfter(JavaClass obj)
    

?

來個例子
        
          public
        
        
          class
        
         DemoPlugin 
        
          extends
        
        
           PreorderDetector {

    
        
        
          private
        
        
           BugReporter bugReporter;

    

    
        
        
          public
        
        
           DemoPlugin(BugReporter reporter){

        bugReporter 
        
        =
        
           reporter;

    }



    
        
        
          public
        
        
          boolean
        
        
           hasHashcode;

    
        
        
          public
        
        
          boolean
        
        
           hasEqual;

    
        
        
          public
        
        
           MethodAnnotation equals;

    

    @Override

    
        
        
          public
        
        
          void
        
        
           visit(JavaClass obj){

        System.out.println(
        
        "visit class!!!!!!!!!" +
        
           obj.getClassName());

        hasHashcode 
        
        = 
        
          false
        
        
          ;

        hasEqual 
        
        = 
        
          false
        
        
          ;

        equals 
        
        = 
        
          null
        
        
          ;

    }

    

    @Override

    
        
        
          public
        
        
          void
        
        
           visit(Method me){

        
        
        
          if
        
        (!
        
          me.isPublic())

            
        
        
          return
        
        
          ;

        
        
        
          if
        
        ("equals".equals(me.getName()) && "(Ljava/lang/Object;)Z"
        
          .equals(me.getSignature())){

            hasEqual 
        
        = 
        
          true
        
        
          ;

            System.out.println(
        
        "find equals() method"
        
          );

            equals 
        
        = MethodAnnotation.fromVisitedMethod(
        
          this
        
        
          );

        }

        

        
        
        
          if
        
        ("hashCode".equals(me.getName()) && "()I"
        
          .equals(me.getSignature())){

            hasHashcode 
        
        = 
        
          true
        
        
          ;

            System.out.println(
        
        "find hashCode() method"
        
          );

        }

        

        
        
        
          if
        
        ("hashcode".equals(me.getName()) && "()I"
        
          .equals(me.getSignature())){

            bugReporter.reportBug(
        
        
          new
        
         BugInstance(
        
          this
        
        , "HASHCODE WRONG SPELL"
        
          , NORMAL_PRIORITY).

                    addClass(getDottedClassName()).addMethod(MethodAnnotation.fromVisitedMethod(
        
        
          this
        
        
          )));

        }

    }

    

    @Override

    
        
        
          public
        
        
          void
        
        
           visitAfter(JavaClass obj){

        
        
        
          if
        
        (hasEqual && !
        
          hasHashcode){

            BugInstance instance 
        
        = 
        
          new
        
         BugInstance(
        
          this
        
        , "EQUALS WITHOUT HASHCODE"
        
          , NORMAL_PRIORITY).

                    addClass(getDottedClassName()).addMethod(equals);

            bugReporter.reportBug(instance);

        }

    }

}
        
      

這里我們繼承了PreorderDetector,而PreorderDetector繼承了PreorderVisitor,是因為BugInstance需要使用Detector進行實例,而PreorderDetector實現(xiàn)了Detector接口。

上面方法我們使用了上個變量hasHashcode,hasEquals和equals,前兩個變量用來保存當前訪問的類是否存在hashCode和equals方法,第三個變量存放的是方法注釋(要在報告bug中,這里先不管)。

上述三個變量每次都在visit(JavaClass obj)方法中進行狀態(tài)重置,這樣才能對所有的類都進行檢測。visit(Method me)方法中用來判斷是否是equals、hashCode、hashcode方法,并修改相應(yīng)的狀態(tài)。如果是hashcode方法,則直接將該bug報告。

visitAfter(JavaClass obj)方法執(zhí)行時,已經(jīng)對當前類檢測完畢,這樣可以判斷是否出現(xiàn)equals方法,而沒有出現(xiàn)hashCode方法的情況。

將fandbugs.xml和massage.xml加到項目中打包,便能對我們定義的模式進行檢測。

以上僅是一個自定義在類、方法、字段層面上的檢測器的一般步驟,例子僅是為了說明過程,只要我們熟悉我們想要檢測的模式,就能通過上述方法實現(xiàn)。

FindBugs檢測器實現(xiàn)(2)


更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主

微信掃碼或搜索:z360901061

微信掃一掃加我為好友

QQ號聯(lián)系: 360901061

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。

【本文對您有幫助就好】

您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描上面二維碼支持博主2元、5元、10元、自定義金額等您想捐的金額吧,站長會非常 感謝您的哦!!!

發(fā)表我的評論
最新評論 總共0條評論