日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問(wèn)題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營(yíng)銷解決方案
深究.NET中程序的編譯時(shí)間

我們?cè)诰帉?NET程序時(shí),經(jīng)常會(huì)在該程序的“關(guān)于本軟件”對(duì)話框中給出這個(gè)程序的編譯時(shí)間,如下圖所示:

創(chuàng)新互聯(lián)服務(wù)項(xiàng)目包括皋蘭網(wǎng)站建設(shè)、皋蘭網(wǎng)站制作、皋蘭網(wǎng)頁(yè)制作以及皋蘭網(wǎng)絡(luò)營(yíng)銷策劃等。多年來(lái),我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,皋蘭網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到皋蘭省份的部分城市,未來(lái)相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!

上圖中的編譯時(shí)間是如果得到的呢?其實(shí)是在其 C# 源程序中有這么一句:

 
 
 
 
  1. [assembly: AssemblyVersion("1.3.*")] 

上述語(yǔ)句使用了 System.Reflection.AssemblyVersionAttribute 類,該類用于指定正在特性化的程序集的版本。在 MSDN 文檔中有以下描述:

 
 
 
 
  1. 程序集版本號(hào)是程序集標(biāo)識(shí)的一部分,在綁定到程序集時(shí)以及在版本策略中扮演著關(guān)鍵的角色。  
  2. 版本號(hào)包含以下四部分:<主版本(Major)>.<次版本(Minor)>.<內(nèi)部版本號(hào)(Build)>.<修訂號(hào)(Revision)> 
  3. 您可以指定所有這些值,也可使用星號(hào) (*) 表示接受默認(rèn)的內(nèi)部版本號(hào)、修訂號(hào),或者接受二者。 例如,  
  4. [assembly:AssemblyVersion("2.3.25.1")] 指示主版本為 2,次版本為 3,內(nèi)部版本號(hào)為 25,修訂號(hào)為 1。 
  5. 版本號(hào) [assembly:AssemblyVersion("1.2.*")] 指定主版本為 1,次版本為 2,并接受默認(rèn)的內(nèi)部版本號(hào)和修訂號(hào)。 
  6. 版本號(hào) [assembly:AssemblyVersion("1.2.15.*")] 指定主版本為 1,次版本為 2,內(nèi)部版本號(hào)為 15,并接受默認(rèn)的修訂號(hào)。  
  7. 默認(rèn)的內(nèi)部版本號(hào)每日增加。 默認(rèn)修訂號(hào)是隨機(jī)的。  

具體來(lái)說(shuō),默認(rèn)的內(nèi)部版本號(hào)表示自2000年1月1日以來(lái)的天數(shù),而默認(rèn)修訂號(hào)也不是隨機(jī)的,表示自該天午夜零時(shí)以來(lái)的秒數(shù)的一半。于是就可以使用下面的表達(dá)式獲得 .NET 程序的編譯時(shí)間:

 
 
 
 
  1. new DateTime(2000, 1, 1).AddDays(version.Build).AddSeconds(version.Revision * 2) 

但是,還有很多 .NET 程序的程序集版本號(hào)沒(méi)有使用星號(hào)來(lái)接受默認(rèn)的內(nèi)部版本號(hào)、修訂號(hào),就不能使用這個(gè)方法了。我們知道,.NET 程序也是一個(gè)標(biāo)準(zhǔn)的32位或64位的 Microsoft Windows 可執(zhí)行體(PE32,Portable Executable)文件。而PE文件頭中包含一個(gè)時(shí)間標(biāo)記來(lái)指出文件的生成時(shí)間,請(qǐng)參見(jiàn):“Microsoft可移植可執(zhí)行文件和通用目標(biāo)文件格式文件規(guī)范v8.1修訂版”和“Microsoft Portable Executable and Common Object File Format Specification”。具體來(lái)說(shuō)就是:

PE文件頭由Microsoft MS-DOS?占位程序、PE文件簽名、COFF文件頭以及可選文件頭組成,COFF目標(biāo)文件頭由COFF文件頭和可選文件頭組成。在這兩種情況下,文件頭后面緊跟著的都是節(jié)頭。
MS-DOS占位程序是一個(gè)運(yùn)行于MS?DOS下的合法應(yīng)用程序,它被放在EXE映像的最前面。在位置0x3C處,這個(gè)占位程序包含PE文件簽名的偏移地址。
在MS-DOS占位程序后面、在偏移0x3C指定的文件偏移處,是一個(gè)4字節(jié)的簽名,它用來(lái)標(biāo)識(shí)文件為一個(gè)PE格式的映像文件。這個(gè)簽名是“PE\0\0”(字母“P”和“E”后跟著兩個(gè)空字節(jié))。
緊跟著映像文件簽名之后,是一個(gè)標(biāo)準(zhǔn)COFF文件頭。在這個(gè)COFF文件頭的偏移為4開(kāi)始的4個(gè)字節(jié)就是從UTC時(shí)間1970年1月1日00:00起的總秒數(shù)(一個(gè)C運(yùn)行時(shí)time_t類型的值)的低32位,它指出文件何時(shí)被創(chuàng)建。

一個(gè)PE文件的例子如下圖所示:

使用 DumpBin.exe 可得到如下信息:

 
 
 
 
  1. Microsoft (R) COFF/PE Dumper Version 10.00.30319.01  
  2. Copyright (C) Microsoft Corporation.  All rights reserved.  
  3.  
  4. Dump of file PowerWord2Snb.exe  
  5.  
  6. PE signature found  
  7.  
  8. File Type: EXECUTABLE IMAGE  
  9.  
  10. FILE HEADER VALUES  
  11.              14C machine (x86)  
  12.                3 number of sections  
  13.         4D0165D7 time date stamp Fri Dec 10 07:27:19 2010  
  14.                0 file pointer to symbol table  
  15.                0 number of symbols  
  16.               E0 size of optional header  
  17.              102 characteristics  
  18.                    Executable  
  19.                    32 bit word machine 

此外,在操作系統(tǒng)的文件系統(tǒng)中,也記錄了每個(gè)文件(不要求是PE文件)的創(chuàng)建和修改時(shí)間,如下圖所示:

我們寫一個(gè) C# 程序來(lái)獲取這三個(gè)時(shí)間吧:

 
 
 
 
  1. using System;  
  2. using System.IO;  
  3.  using System.Reflection;  
  4. [assembly: AssemblyVersion("1.0.*")]  
  5. namespace Skyiv.BuildTime  
  6.  {  
  7.    sealed class Program  
  8.    {  
  9.      delegate DateTime GetTime(string fileName);  
  10.     TextWriter writer;  
  11.      Program(TextWriter writer)  
  12.     {  
  13.       this.writer = writer;  
  14.     }  
  15.     static void Main(string[] args)  
  16.   {  
  17.      Console.WriteLine("OS  Version: " + Environment.OSVersion);  
  18.       Console.WriteLine("CLR Version: " + Environment.Version);  
  19.       Console.WriteLine();  
  20. var fileName = (args.Length > 0) ? args[0] : Assembly.GetExecutingAssembly().Location;  
  21.      new Program(Console.Out).Write(fileName);  
  22.     }  
  23.     void Write(string fileName)  
  24.     {  
  25.       writer.WriteLine(fileName);  
  26.       Write("文件系統(tǒng)  ", GetFileCreationTime, fileName);  
  27.        Write("PE32      ", GetPe32Time, fileName);  
  28. rite("裝配件版本", GetAssemblyVersionTime, fileName);  
  29.     }  
  30.  void Write(string msg, GetTime getTime, string fileName)  
  31.     {  
  32.      string time;  
  33.       try 
  34.       {  
  35.          time = getTime(fileName).ToString("yyyy-MM-dd HH:mm:ss");  
  36.     }  
  37.       catch (Exception ex)  
  38.       {  
  39.        time = ex.Message;  
  40.      }  
  41.      writer.WriteLine("{0}: {1}", msg, time);  
  42.      }  
  43.   DateTime GetFileCreationTime(string fileName)  
  44.     {  
  45.       return new FileInfo(fileName).CreationTime;  
  46.   }  
  47.   DateTime GetAssemblyVersionTime(string fileName)  
  48.   {  
  49.      var version = Assembly.LoadFrom(fileName).GetName().Version;  
  50. return new DateTime(2000, 1, 1).AddDays(version.Build).AddSeconds(version.Revision * 2);  
  51.     }  
  52.    DateTime GetPe32Time(string fileName)  
  53.    {  
  54.        int seconds;  
  55. using (var br = new BinaryReader(new FileStream(fileName, FileMode.Open, FileAccess.Read)))  
  56.        {  
  57.          var bs = br.ReadBytes(2);  
  58.        var msg = "非法的PE32文件";  
  59.   if (bs.Length != 2) throw new Exception(msg);  
  60.   if (bs[0] != 'M' || bs[1] != 'Z') throw new Exception(msg);  
  61.        br.BaseStream.Seek(0x3c, SeekOrigin.Begin);  
  62.         var offset = br.ReadByte();  
  63.          br.BaseStream.Seek(offset, SeekOrigin.Begin);  
  64.          bs = br.ReadBytes(4);  
  65.    if (bs.Length != 4) throw new Exception(msg);  
  66.  if (bs[0] != 'P' || bs[1] != 'E' || bs[2] != 0 || bs[3] != 0) throw new Exception(msg);  
  67.         bs = br.ReadBytes(4);  
  68.         if (bs.Length != 4) throw new Exception(msg);  
  69.         seconds = br.ReadInt32();  
  70.      }  
  71.     return DateTime.SpecifyKind(new DateTime(1970, 1, 1), DateTimeKind.Utc).  
  72.         AddSeconds(seconds).ToLocalTime();  
  73.     }  
  74.   }  
  75.  } 

這個(gè)程序的運(yùn)行結(jié)果如下所示:

 
 
 
 
  1. E:\work> BuildTime PowerWord2Snb.exe  
  2. OS  Version: Microsoft Windows NT 6.0.6002 Service Pack 2  
  3. CLR Version: 2.0.50727.4206  
  4.  
  5. PowerWord2Snb.exe  
  6. 文件系統(tǒng)  : 2010-12-10 07:32:14  
  7. PE32      : 2010-12-10 07:27:19  
  8. 裝配件版本: 2010-12-10 07:27:18  
  9.  
  10. E:\work> BuildTime  
  11. OS  Version: Microsoft Windows NT 6.0.6002 Service Pack 2  
  12. CLR Version: 2.0.50727.4206  
  13.  
  14. E:\work\BuildTime.exe  
  15. 文件系統(tǒng)  : 2010-12-18 20:26:24  
  16. PE32      : 2010-12-18 19:56:41  
  17. 裝配件版本: 2010-12-18 19:56:40 

我們來(lái)看看著名的 .NET Reflector 的信息吧:

 
 
 
 
  1. E:\work> BuildTime d:\bin\reflector\reflector.exe  
  2. OS  Version: Microsoft Windows NT 6.0.6002 Service Pack 2  
  3. CLR Version: 2.0.50727.4206  
  4.  
  5. d:\bin\reflector\reflector.exe  
  6. 文件系統(tǒng)  : 2008-07-10 19:36:45  
  7. PE32      : 2010-07-15 00:42:36  
  8. 裝配件版本: 2000-01-01 00:00:00  
  9.  
  10. E:\work> 

再來(lái)更多的例子:

 
 
 
 
  1. E:\work> BuildTime C:\windows\regedit.exe  
  2. OS  Version: Microsoft Windows NT 6.0.6002 Service Pack 2  
  3. CLR Version: 2.0.50727.4206  
  4.  
  5. C:\windows\regedit.exe  
  6. 文件系統(tǒng)  : 2008-04-23 19:35:29  
  7. PE32      : 2008-01-19 13:30:16  
  8. 裝配件版本: 未能加載文件或程序集“file:///C:\windows\regedit.exe”  
  9. 或它的某一個(gè)依賴項(xiàng)。該模塊應(yīng)包含一個(gè)程序集清單。  
  10.  
  11. E:\work> BuildTime BuildTime.cs  
  12. OS  Version: Microsoft Windows NT 6.0.6002 Service Pack 2  
  13. CLR Version: 2.0.50727.4206  
  14.  
  15. BuildTime.cs  
  16. 文件系統(tǒng)  : 2010-12-18 20:50:03  
  17. PE32      : 非法的PE32文件  
  18. 裝配件版本: 未能加載文件或程序集“file:///E:\work\BuildTime.cs”  
  19. 或它的某一個(gè)依賴項(xiàng)。該模塊應(yīng)包含一個(gè)程序集清單。  
  20.  
  21. E:\work> 

我們?cè)倏纯丛?Linux 操作系統(tǒng)中的情況(openSUSE 11.3, mono 2.8.1):

 
 
 
 
  1. ben@ben1520:~/work> dmcs BuildTime.cs  
  2. ben@ben1520:~/work> mono BuildTime.exe  
  3. OS  Version: Unix 2.6.34.7  
  4. CLR Version: 4.0.30319.1  
  5.  
  6. /home/ben/work/BuildTime.exe  
  7. 文件系統(tǒng)  : 2010-12-18 21:29:14  
  8. PE32      : 2010-12-18 21:29:14  
  9. 裝配件版本: 2000-01-01 00:00:00  
  10. ben@ben1520:~/work> mono BuildTime.exe /usr/lib/mono/4.0/dmcs.exe  
  11. OS  Version: Unix 2.6.34.7  
  12. CLR Version: 4.0.30319.1  
  13.  
  14. /usr/lib/mono/4.0/dmcs.exe  
  15. 文件系統(tǒng)  : 2010-11-12 22:52:48  
  16. PE32      : 2010-11-12 22:44:24  
  17. 裝配件版本: 2000-01-02 00:00:00  
  18. ben@ben1520:~/work> mono BuildTime.exe /usr/bin/gcc-4.5  
  19. OS  Version: Unix 2.6.34.7  
  20. CLR Version: 4.0.30319.1  
  21.  
  22. /usr/bin/gcc-4.5  
  23. 文件系統(tǒng)  : 2010-07-02 02:29:53  
  24. PE32      : 非法的PE32文件  
  25. 裝配件版本: Could not load file or assembly '/usr/bin/gcc-4.5' or one of its dependencies.  
  26.  An attempt was made to load a program with an incorrect format.  
  27. ben@ben1520:~/work>  

由上可見(jiàn),在 Linux 操作系統(tǒng)中 mono 對(duì) System.Reflection.AssemblyVersionAttribute 類的支持很成問(wèn)題。在 Microsoft 的 MSDN 文檔中說(shuō)可以用星號(hào)表示接受默認(rèn)的內(nèi)部版本號(hào)、修訂號(hào),默認(rèn)的內(nèi)部版本號(hào)每日增加。 默認(rèn)修訂號(hào)是隨機(jī)的。而 mono 的 C# 編譯器直接將默認(rèn)的內(nèi)部版本號(hào)和默認(rèn)的修訂號(hào)都設(shè)置為零了。

綜上所述,要得到 .NET 程序的編譯時(shí)間,還是去讀該程序的PE文件頭中的文件創(chuàng)建時(shí)間最靠譜。

原文鏈接:http://www.cnblogs.com/skyivben/archive/2010/12/18/1910180.html

【編輯推薦】

  1. .NET Framework字符串相關(guān)操作細(xì)節(jié)介紹
  2. 詳解.NET字符串解析的具體過(guò)程
  3. 漫談.NET開(kāi)發(fā)中的字符串編碼
  4. .NET Lambda表達(dá)式的語(yǔ)義:字符串列表范例
  5. 解釋.NET之匿名對(duì)象(AnonymousObject)


新聞名稱:深究.NET中程序的編譯時(shí)間
網(wǎng)頁(yè)URL:http://www.dlmjj.cn/article/dpgigjj.html