〇、關於本文算法
本文中的兩個函數函數
1)用於統計擴展名爲 .h .c .cpp .cs 文件的代碼行數 spa
public static int LinesOfCode(string filename)code
2)用於遞歸統計一個文件夾內全部擴展名爲 .h .c .cpp .cs 文件的代碼行數 遞歸
public static int LinesOfFolder(string foldername)字符串
1、什麼樣的狀況算一行代碼string
須要注意以下幾點:it
1)若是一行爲空,則不算做一行代碼。在字符串中的空行除外,如:io
Console.WriteLine(@"fasfew fewafa");
2)Windows中的換行爲\r\n,Linux中的換行爲\n,所以判斷隔行能夠統一以\n計class
(r:Carriage Return,回車;n:Linefeed,換行)
所以,判斷算法採用如下步驟:
1)遇到' '、'\r'、'\t'是無效字符,直接略過
2)遇到'\n',若是該行有有效字符,則認爲該行有代碼,不然認爲沒有
3)遇到字符'\"',則字符串開始,直到找到下一個字符'\"',中間忽略任何字符。注意字符若是找到的'\"'前有奇數個"\\",則跳過繼續搜索。若是遇到'\n',則按代碼行數自增1
4)遇到形如 /*...*/ 的註釋,找到'/'和'*'相連的狀況,則繼續找'*'和'/'相連的狀況。中間如有'\n',則看註釋開始前該行是否有效,有效則算一行,無效則不算
5)遇到形如 //... 的註釋,則看註釋開始前該行是否有效,有效則算一行,無效則不算
6)遍歷完整個文件後,因爲最後一行可能不以'\n'結尾,所以遍歷完畢後最後一行有沒有有效字符,有則最後一行算做一行,沒有則不算
2、查看文件中的代碼行數
/// <summary> /// 檢測一個C代碼文件中的有效代碼行數 /// </summary> /// <param name="filename">文件名</param> /// <returns>代碼行數</returns> public static int LinesOfCode(string filename) { System.IO.StreamReader sr = System.IO.File.OpenText(filename); string s = sr.ReadToEnd(); sr.Close(); bool isLine = false; //一行中擁有有效字符時爲true,該行可記入代碼行數 bool isCommitLf = false; //註釋/*...*/中出現至少一個折行時爲true int lines = 0; //代碼行數統計 for (int i = 0; i < s.Length; i++) { //無效字符 if (s[i] == ' ' || s[i] == '\r' || s[i] == '\t') { continue; } //搜索到換行,若該行有有效字符 if (s[i] == '\n') { if (isLine) { lines++; isLine = false; } continue; } //字符串,佔多少行按多少行算 if (s[i] == '\"') { while (true) { i++; //若是文件遍歷完畢則強行停止 if (i >= s.Length) { break; } //再次遇到字符'"'且前方沒有或有偶數個'//'時,停止循環並退出 if (s[i] == '\"') { int sign = 0, counter = 0; while (true) { sign++; if (i - sign < 0) { break; } else if (s[i - sign] == '\\') { counter++; } else { break; } } if (counter % 2 == 0) { break; } } //字符串中的換行,直接算做一行代碼 if (s[i] == '\n') { lines++; isLine = true; } } isLine = true; continue; } //遇到形如 /*...*/ 的註釋 if (s[i] == '/' && i < s.Length - 1) { if (s[i + 1] == '*') { i++; while (true) { i++; //若是文件遍歷完畢則強行停止 if (i >= s.Length) { break; } if (s[i] == '\n') { if (isCommitLf == false) { if (isLine == true) { lines++; isLine = false; } isCommitLf = true; } } if (s[i] == '*' && i < s.Length - 1) { if (s[i + 1] == '/') { i++; break; } } } isCommitLf = false; continue; } } //遇到形如 // 的註釋 if (s[i] == '/' && i < s.Length - 1 && s[i + 1] == '/') { if (isLine == true) { lines++; isLine = false; } while (true) { i++; if (i >= s.Length || s[i] == '\n') { break; } } continue; } //該行有了有效字符,算做一行 isLine = true; } //最後一行可能沒有字符'\n'結尾 if (isLine) { lines++; } return lines; }
3、查看文件夾中因此代碼文件的代碼行數
/// <summary> /// 檢測一個文件夾中全部C代碼的行數 /// </summary> /// <param name="foldername">文件夾名稱</param> /// <returns>代碼行數</returns> public static int LinesOfFolder(string foldername) { //行數統計 int lines = 0; //文件夾信息 System.IO.DirectoryInfo dif = new System.IO.DirectoryInfo(foldername); //遍歷文件夾中的各子文件夾 foreach (System.IO.DirectoryInfo di in dif.GetDirectories()) { lines += LinesOfFolder(di.FullName); } //統計本文件夾中C語言文件代碼 foreach (System.IO.FileInfo f in dif.GetFiles()) { if (f.Extension == ".cs" || f.Extension == ".cpp" || f.Extension == ".c" || f.Extension == ".h") { lines += LinesOfCode(f.FullName); } } return lines; }
4、Main函數
輸入命令 checkfile:文件名 或 checkfolder:文件夾路徑 查詢
static void Main(string[] args) { Console.WriteLine("請輸入要統計的文件或文件夾"); Console.WriteLine("輸入示例:checkfile:xxx / checkfolder:xxx"); try { string order = Console.ReadLine(); string[] temp = order.Split(':'); if (temp.Length < 2) { Console.WriteLine("語法錯誤,程序結束"); } switch (temp[0]) { case "checkfile": { int count = LinesOfCode(order.Substring(10).Trim()); Console.WriteLine("共有代碼 " + count + " 行"); } break; case "checkfolder": { int count = LinesOfFolder(order.Substring(12).Trim()); Console.WriteLine("共有代碼 " + count + " 行"); } break; default: Console.WriteLine("未知命令"); break; } } catch (Exception ex) { Console.WriteLine(ex.Message); } Console.ReadLine(); }
5、運行結果示例