自己動手設(shè)計代碼編輯器——(五)分析源代碼

因?yàn)橹暗脑O(shè)計不合理,就重新開了工程?;藥滋鞎r間,重做的進(jìn)度趕上之前的了
今天來說說分析源代碼
源碼分析中會用到的其它東西,之前都介紹過了。
接下來看UParser這個類中最后的函數(shù)GetNextCut,是用來把源碼分析為UCodeCut集合的。其中包括關(guān)鍵字設(shè)別,變量設(shè)別等等

開始

    UCodeCut cut = new UCodeCut();  
    List<byte> cutData = new List<byte>();  
    byte b;  
    UCutType currType = UCutType.None; // 狀態(tài)機(jī)的狀態(tài)  
      
    bool inString = false; // 是否解析字符串  
      
    if (EndOfCode == true)  
    {  
    cut.CutType = UCutType.End;  
    return cut;  
    }  

開頭就是定義一些變量,并且判斷分析是否結(jié)束
cut用于保存分析玩的CodeCut
cutData是cut中的具體數(shù)據(jù)
currType是當(dāng)前狀態(tài)機(jī)的狀態(tài)

接下來是狀態(tài)機(jī)的循環(huán)

while (!EndOfCode)  
{  
b = GetNextByte();  
cutData.Add(b);  
...  
...  
} 

然后是狀態(tài)機(jī)的入口

    #region UCutType.None  
                        if (currType == UCutType.None)  
                        {  
                            if (b == UConfig.Space)  
                            {  
                                currType = UCutType.Space;  
                                continue;  
                            }  
      
                            if (b == UConfig.Tab)  
                            {  
                                currType = UCutType.Tab;  
                                break;  
                            }  
      
                            if (b == UConfig.DoubleQuote)  
                            {  
                                currType = UCutType.String;  
                                inString = true;  
                                continue;  
                            }  
      
                            if (b == UConfig.NewLine)  
                            {  
                                currType = UCutType.NewLine;  
      
                                // 跳過回車符  
                                if (PeekNextByte() == UConfig.Enter)  
                                {  
                                    GetNextByte();  
                                }  
      
                                break;  
                            }  
      
                            if (b == UConfig.BackSlash)  
                            {  
                                if (PeekNextByte() == UConfig.BackSlash)  
                                {  
                                    currType = UCutType.Annotation;  
                                    continue;  
                                }  
                            }  
      
                            if (UHelper.IsSymbol(b))  
                            {  
                                currType = UCutType.Symbol;  
                                break;  
                            }  
      
                            if (UHelper.IsCharacter(b))  
                            {  
                                currType = UCutType.Normal;  
                                continue;  
                            }  
      
                            if (UHelper.IsDigit(b))  
                            {  
                                currType = UCutType.Digit;  
                                continue;  
                            }  
                        }  
                        #endregion  

接下來是Normal狀態(tài),就是一些普通的字符串

    #region UCutType.Normal  
                        if (currType == UCutType.Normal)  
                        {  
                            if (UHelper.IsCutEnd(b))  
                            {  
                                BackToLastByte();  
      
                                cutData.RemoveAt(cutData.Count - 1);  
                                break;  
                            }  
                        }  
                        #endregion  

接下來是String狀態(tài),是字符串

#region UCutType.String  
                    if (currType == UCutType.String)  
                    {  
                        if (b == UConfig.NewLine)  
                        {  
                            BackToLastByte();  
  
                            currType = UCutType.Normal;  
                            break;  
                        }  
  
                        if (b == UConfig.DoubleQuote)  
                        {  
                            inString = false;  
                            break;  
                        }  
  
                        if (b == UConfig.Slash)  
                        {  
                            // 添加 \ 后的字符  
                            if (inString)  
                            {  
                                //ch = (char)GetNextChar();  
                                //cutData.Add((byte)ch);  
                                cutData.Add(GetNextByte());  
                                continue;  
                            }  
                        }  
                    }  
                    #endregion

接下來 數(shù)字狀態(tài)、注釋等等,都一樣的判斷格式
最后循環(huán)結(jié)束,檢查關(guān)鍵字

    // 替換Tab為Space  
                    if (currType == UCutType.Tab)  
                    {  
                        cut.Data = UConfig.TabString;  
                        cut.CutType = UCutType.Space;  
                    }  
                    else  
                    {  
                        cut.Data = UHelper.GetStringByBytes(cutData.ToArray());  
                        cut.CutType = currType;  
                          
                        // 如果是普通的一段文本,判斷是否為關(guān)鍵字  
                        if (currType == UCutType.Normal)  
                        {  
                            if (UHelper.IsKeyWord(cut.Data))  
                            {  
                                cut.CutType = UCutType.KeyWord;  
                            }  
                        }  
                    }  

最后檢查,是否為變量或者類

if (currType == UCutType.Normal)  
                {  
                    if (LastCut2.CutType == UCutType.KeyWord && LastCut1.CutType == UCutType.Space)  
                    {  
                        // 類的定義  
                        if (LastCut2.Data == UConfig.ClassString)  
                        {  
                            cut.CutType = UCutType.ClassName;  
                        }  
                        else  
                        {  
                            // 函數(shù)名的定義  
                            foreach (string str in UConfig.FunctionDefineString)  
                            {  
                                if (str == LastCut2.Data)  
                                {  
                                    if (PeekNextByte() == (byte)'(')  
                                    {  
                                        cut.CutType = UCutType.FunctionName;  
                                    }  
                                    else  
                                    {  
                                        cut.CutType = UCutType.VariableName;  
                                    }  
                                    break;  
                                }  
                            }  
                        }  
                    }  
                }  
  
                LastCut2 = LastCut1;  
                LastCut1 = cut; 

這就是一個CodeCut的分析過程了
接下來只要不斷的調(diào)用GetNextCut()就能分析完整個代碼

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 179,036評論 25 709
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法,內(nèi)部類的語法,繼承相關(guān)的語法,異常的語法,線程的語...
    子非魚_t_閱讀 34,696評論 18 399
  • 戴旺 昨天有幸學(xué)習(xí)了兩位李老師的語文課,讓我對語文教學(xué)有了新的認(rèn)識。 ...
    戴言旺語閱讀 266評論 0 1
  • 今天給自己打70分,上午聽了任紅霞十九大宣講報告,下午開了會,宣布了任命,可我一點(diǎn)感覺都沒有,換做幾年前或許會高興...
    小熊熊1234閱讀 192評論 0 0

友情鏈接更多精彩內(nèi)容