<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    和風細雨

    世上本無難事,心以為難,斯乃真難。茍不存一難之見于心,則運用之術自出。

    使用正則表達式解析SQL語句

    本文詳細代碼請見:
    http://m.tkk7.com/sitinspring/archive/2008/03/14/186372.html

     

    問題:將左邊的SQL語句解析成右邊的形式

    Select c1,c2,c3 From t1,t2,t3 Where condi1=5 and condi6=6 or condi7=7 Group  by g1,g2,g3 order  by g2,g3

    select
         c1,
        c2,
        c3
    from
         t1,
        t2,
        t3
    where
         condi1=5 and
         condi6=6 or
         condi7=7
    group by
         g1,
        g2,
        g3
    order by
         g2,
        g3

    按關鍵字找出SQL語句中各部分

    我們閱讀SQL語句會把整句分來成列,表,條件,分組字段,排序字段來理解,解析SQL的目的也是這樣.
    分解SQL語句有規律可循,以列為例,它必定包含在select和from之間,我們只要能找到SQL語句中的關鍵字select和from,就能找到查詢的列.
    怎么找到select和from之間的文字呢?其實一個正則表達式就能解決:(select)(.+)(from),其中第二組(.+)代表的文字就是select和from之間的文字.
    程序見右邊.

    /**
     * 從文本text中找到regex首次匹配的字符串,不區分大小寫
     * @param regex: 正則表達式
     * @param text:欲查找的字符串
     * @return regex首次匹配的字符串,如未匹配返回空
     */
    private static String getMatchedString(String regex,String text){
      Pattern pattern=Pattern.compile(regex,Pattern.CASE_INSENSITIVE);
       
        Matcher matcher=pattern.matcher(text);

        while(matcher.find()){
          return matcher.group(2);
        }
       
        return null;
    }

    解析函數分析

    private static String getMatchedString(String regex,String text){
      Pattern pattern=Pattern.compile(regex,Pattern.CASE_INSENSITIVE);
       
        Matcher matcher=pattern.matcher(text);

        while(matcher.find()){
          return matcher.group(2);
        }
       
        return null;
    }

    左邊的這個函數,第一個參數是擬定的正則表達式,第二個是整個SQL語句.
    當正則表達式為(select)(.+)(from)時,程序將在SQL中查找第一次匹配的地方(有Pattern.CASE_INSENSITIVE的設置,查找不區分大小寫),如果找到了則返回模式中的第二組代表的文字.
    如果sql是select a,b from tc,則返回的文字是a,b.

    選擇的表對應的查找正則表達式

    選擇的表比較特殊,它不想選擇的列一樣固定處于select和from之間,當沒有查找條件存在時,它處于from和結束之間;當有查找條件存在時,它處于from和where之間.
    因此查詢函數寫為右邊的形式:

    /**
     * 解析選擇的表
     *
     */
    private void parseTables(){
        String regex="";  
       
        if(isContains(sql,"\\s+where\\s+")){
          regex="(from)(.+)(where)";  
        }
        else{
          regex="(from)(.+)($)";  
        }
       
        tables=getMatchedString(regex,sql);
    }


    isContains函數

    isContains函數用于在lineText中查找word,其中不區分大小些,只要找到了即返回真.

    /**
     * 看word是否在lineText中存在,支持正則表達式
     * @param lineText
     * @param word
     * @return
     */
    private static boolean isContains(String lineText,String word){
      Pattern pattern=Pattern.compile(word,Pattern.CASE_INSENSITIVE);
      Matcher matcher=pattern.matcher(lineText);
      return matcher.find();
    }

    解析查找條件的函數

    private void parseConditions(){
        String regex="";  
       
        if(isContains(sql,"\\s+where\\s+")){
          // 包括Where,有條件
         
          if(isContains(sql,"group\\s+by")){
            // 條件在where和group by之間
            regex="(where)(.+)(group\\s+by)"; 
          }
          else if(isContains(sql,"order\\s+by")){
            // 條件在where和order by之間
            regex="(where)(.+)(order\\s+by)"; 
          }
          else{
            // 條件在where到字符串末尾
            regex="(where)(.+)($)"; 
          }       
        }
        else{
          // 不包括where則條件無從談起,返回即可
          return;
        }
       
        conditions=getMatchedString(regex,sql);
    }

    解析GroupBy的字段

    private void parseGroupCols(){
        String regex="";  
       
        if(isContains(sql,"group\\s+by")){
          // 包括GroupBy,有分組字段

          if(isContains(sql,"order\\s+by")){
            // group by 后有order by
            regex="(group\\s+by)(.+)(order\\s+by)"; 
          }
          else{
            // group by 后無order by
            regex="(group\\s+by)(.+)($)"; 
          }     
        }
        else{
          // 不包括GroupBy則分組字段無從談起,返回即可
          return;
        }
       
        groupCols=getMatchedString(regex,sql);
    }


    解析OrderBy的字段

    private void parseOrderCols(){
        String regex="";  
       
        if(isContains(sql,"order\\s+by")){
          // 包括order by,有分組字段
          regex="(order\\s+by)(.+)($)";                 
        }
        else{
          // 不包括GroupBy則分組字段無從談起,返回即可
          return;
        }
         
        orderCols=getMatchedString(regex,sql);
    }

    得到解析后的各部分

    按以上解析方法獲得了列,表,條件,分組條件,排序條件各部分之后,它們會存儲到各個成員變量中.
    注意這些成員變量的原值都是null,如果在SQL語句中能夠找到對應的部分的話它們將借助getMatchedString獲得值,否則還是null.我們通過判斷這些成員變量是否為空就能知道它對應的部分是否被解析出來.

     /**
       * 待解析的SQL語句
       */
      private String sql;
     
      /**
       * SQL中選擇的列
       */
      private String cols;
     
      /**
       * SQL中查找的表
       */
      private String tables;
     
      /**
       * 查找條件
       */
      private String conditions;
     
      /**
       * Group By的字段
       */
      private String groupCols;
     
      /**
       * Order by的字段
       */
      private String orderCols;

    取得不需要單行顯示時的SQL語句

    進展到這一步,SQL語句中列,表,條件,分組條件,排序條件各部分都被獲取了出來,這時把它們重新組合一下就能得到整理后的SQL語句.
    如下面的SQL語句將變成右邊的部分(先使靜態成員isSingleLine=false):
    Select c1,c2,c3 From t1,t2,t3 Where condi1=5 and condi6=6 or condi7=7 Group  by g1,g2,g3 order  by g2,g3

    select
         c1,c2,c3
    from
         t1,t2,t3
    where
         condi1=5 and condi6=6 or condi7=7
    group by
         g1,g2,g3
    order by
         g2,g3


    進一步解析

    有時我們需要把列,表,條件,分組條件,排序條件單行顯示以方便查看或加上注釋,這就要求我們對列,表,條件,分組條件,排序條件等進行進一步解析.
    初看解析很方便,以固定的分隔符劈分即可,但需要注意的是查詢條件中分隔符有and和or兩種,如果貿然分隔會使重新組合時使SQL失真.
    推薦一種做法,我們可以在分隔符后加上一個標志如空行,然后再以這個標志來劈分.這樣就不會使SQL失真了.
    請見下頁的getSplitedParagraph函數.

    getSplitedParagraph函數

    private static List<String> getSplitedParagraph(String paragraph,String splitStr){
      List<String> ls=new ArrayList<String>();   
     
      // 先在分隔符后加空格
      Pattern p = Pattern.compile(splitStr,Pattern.CASE_INSENSITIVE);

      Matcher m = p.matcher(paragraph);
      StringBuffer sb = new StringBuffer();

      boolean result = m.find();
      while (result) {
        m.appendReplacement(sb, m.group(0) + Crlf);
        result = m.find();
      }
      m.appendTail(sb);
     
      // 再按空格斷行
      String[] arr=sb.toString().split("[\n]+");
      for(String temp:arr){
        ls.add(FourSpace+temp+Crlf);
      }
     
      return ls;
    }

    處理結果

    把靜態成員變量isSingleLine=true后我們來看看執行結果:
    select
         c1,
        c2,
        c3
    from
         t1,
        t2,
        t3
    where
         condi1=5 and
         condi6=6 or
         condi7=7
    group by
         g1,
        g2,
        g3
    order by
         g2,
        g3

    小結

    從這個例子中我們體會了分治的思想:分治是把一個大問題分解成小問題,然后分別解決小問題,再組合起來大問題的解決方法就差不多了.這種思想在工程領域解決問題時很普遍,我們要學會使用這種思想來看待,分析和解決問題,不要貪多求大,結果導致在大問題面前一籌莫展.
    其次我們可以從這個例子中學習找規律,然后借助規律的過程,現實世界千變萬化,但都有規律可循,只要我們找到了規律,就等于找到了事物之門的鑰匙.
    接下了我們復習了正則表達式用于查找的方法,以前的正則表達式學習多用于驗證匹配,其實這只是正則表達式的一部分功能.
    最后從解析條件成單行的過程中,我們可以學習到一種解決問題的技巧,即當現實中的規律存在變數時加入人為設置的規律,這有時能使我們更好更快的解決問題.

    posted on 2008-03-19 22:00 和風細雨 閱讀(9660) 評論(4)  編輯  收藏 所屬分類: 正則表達式

    評論

    # re: 使用正則表達式解析SQL語句 2008-04-22 02:44 java 開發

    謝謝,很有幫助  回復  更多評論   

    # re: 使用正則表達式解析SQL語句 2008-11-25 22:46 bza

    謝謝  回復  更多評論   

    # re: 使用正則表達式解析SQL語句 2011-06-22 17:12 青青園中葵

    復合的sql就不能處理了,這樣太簡單  回復  更多評論   

    # re: 使用正則表達式解析SQL語句 2012-10-08 19:44 郭濤

    的確不錯,很有幫助。
    @青青園中葵  回復  更多評論   


    只有注冊用戶登錄后才能發表評論。


    網站導航:
     
    主站蜘蛛池模板: 久久国内免费视频| 国内精品免费视频精选在线观看| 最近2019中文字幕免费直播| 亚洲av丰满熟妇在线播放| 在线看片免费人成视频福利| 亚洲国产综合无码一区| 免费一级不卡毛片| 亚洲AV无码久久精品色欲| 8x8×在线永久免费视频| 亚洲色图视频在线观看| 在线观看免费人成视频色| 亚洲人成人无码.www石榴| 日韩免费福利视频| 免费人成在线观看播放a| 亚洲日本一区二区一本一道| 久久WWW免费人成—看片| 亚洲AV日韩精品久久久久| 亚洲黄色免费电影| 亚洲欧美aⅴ在线资源| jjzz亚洲亚洲女人| 中国在线观看免费的www| 久久亚洲AV无码精品色午夜| 国产精品成人免费一区二区| 香蕉视频免费在线| 亚洲av无码av制服另类专区| 免费精品国产日韩热久久| 国产成人 亚洲欧洲| 亚洲国产精品SSS在线观看AV| 2021精品国产品免费观看| 亚洲精华国产精华精华液网站| 亚洲精品国精品久久99热| 香蕉免费一区二区三区| 456亚洲人成在线播放网站| 啊v在线免费观看| 久久久久久一品道精品免费看| 国产.亚洲.欧洲在线| 国产偷窥女洗浴在线观看亚洲| 日本高清免费观看| 精品成人一区二区三区免费视频| 亚洲av日韩av无码| yy6080久久亚洲精品|