【轉】 C#版本和.NET版本以及VS版本的對應關系


轉自: http://blog.csdn.net/wulex/article/details/73499500

 

之所以在這里分享這個對應關系,是因為在C#基礎知識系列的文章發布之后,有些初學者對.NET版本和C#語言特性之間的對應關系有點不清楚,有時候會弄混淆了。

並且通過這個對應關系,也可以幫助大家對C#和.NET 類庫有個全面的把控,可以幫助大家理清楚C#和.NET 類庫中各個知識點,使他們可以對號入坐。具體他們的之間對應關系見下表:

版本 .NET Framework版本 Visual Studio版本 發布日期 特性
C# 1.0 .NET Framework 1.0 Visual Studio .NET 2002 2002.1 委托
         
        事件
         
C# 1.1 .NET Framework 1.1 Visual Studio .NET 2003 2003.4 APM
C# 2.0 .NET Framework 2.0 Visual Studio 2005(開始命名為Visual Studio) 2005.11 泛型
         
        匿名方法
         
        迭代器
         
        可空類型
         
         
C# 3.0 .NET Framework 3.0 Visual Studio 2008 2007.11 隱式類型的部變量
         
  .NET Framework 3.5     對象集合初始化
         
        自動實現屬性
         
        匿名類型
         
        擴展方法
         
        查詢表達式
         
        Lambda表達式
         
        表達式樹
         
        分部類和方法
         
        Linq
C# 4.0 .NET Framework 4.0 Visual Studio 2010 2010.4 動態綁定
         
        命名和可選參數
         
        泛型的協變和逆變
         
        互操作性
C# 5.0 .NET Framework 4.5 Visual Studio 2012 2012.8 異步和等待(async和await)
         
        調用方信息(Caller Information)

這里寫圖片描述


 

C# 5.0中新增特性

C# 5.0隨着VisualStudio 2012一起正式發布了,讓我們來看看C#5.0中增加了哪些功能。

1. 異步編程

在.Net 4.5中,通過async和await兩個關鍵字,引入了一種新的基於任務的異步編程模型(TAP)。在這種方式下,可以通過類似同步方式編寫異步代碼,極大簡化了異步編程模型。如下式一個簡單的實例:

 static async void DownloadStringAsync2(Uri uri)
    {
        var webClient = new WebClient();
        var result = await webClient.DownloadStringTaskAsync(uri);
        Console.WriteLine(result);
    }

而之前的方式是這樣的:

static void DownloadStringAsync(Uri uri)
    {
        var webClient = new WebClient();
        webClient.DownloadStringCompleted += (s, e) =>
            {
                Console.WriteLine(e.Result);
            };
        webClient.DownloadStringAsync(uri);
    }

也許前面這個例子不足以體現async和await帶來的優越性,下面這個例子就明顯多了:

public void CopyToAsyncTheHardWay(Stream source, Stream destination)
    {
        byte[] buffer = new byte[0x1000];
        Action<IAsyncResult> readWriteLoop = null;
        readWriteLoop = iar =>
        {
            for (bool isRead = (iar == null); ; isRead = !isRead)
            {
                switch (isRead)
                {
                    case true:
                        iar = source.BeginRead(buffer, 0, buffer.Length,
                            readResult =>
                            {
                                if (readResult.CompletedSynchronously) return;
                                readWriteLoop(readResult);
                            }, null);
                        if (!iar.CompletedSynchronously) return;
                        break;
                    case false:
                        int numRead = source.EndRead(iar);
                        if (numRead == 0)
                        {
                            return;
                        }
                        iar = destination.BeginWrite(buffer, 0, numRead,
                            writeResult =>
                            {
                                if (writeResult.CompletedSynchronously) return;
                                destination.EndWrite(writeResult);
                                readWriteLoop(null);
                            }, null);
                        if (!iar.CompletedSynchronously) return;
                        destination.EndWrite(iar);
                        break;
                }
            }
        };
        readWriteLoop(null);
    }

    public async Task CopyToAsync(Stream source, Stream destination)
    {
        byte[] buffer = new byte[0x1000];
        int numRead;
        while ((numRead = await source.ReadAsync(buffer, 0, buffer.Length)) != 0)
        {
            await destination.WriteAsync(buffer, 0, numRead);
        }
    }

關於基於任務的異步編程模型需要介紹的地方還比較多,不是一兩句能說完的,有空的話后面再專門寫篇文章來詳細介紹下。另外也可參看微軟的官方網站:Visual Studio Asynchronous Programming,其官方文檔Task-Based Asynchronous Pattern Overview介紹的非常詳細, VisualStudio中自帶的CSharp Language Specification中也有一些說明。

2. 調用方信息

很多時候,我們需要在運行過程中記錄一些調測的日志信息,如下所示:

public void DoProcessing()
    {
        TraceMessage("Something happened.");
    } 

為了調測方便,除了事件信息外,我們往往還需要知道發生該事件的代碼位置以及調用棧信息。在C++中,我們可以通過定義一個宏,然后再宏中通過FILELINE來獲取當前代碼的位置,但C#並不支持宏,往往只能通過StackTrace來實現這一功能,但StackTrace卻有不是很靠譜,常常獲取不了我們所要的結果。

針對這個問題,在.Net 4.5中引入了三個Attribute:CallerMemberNameCallerFilePathCallerLineNumber。在編譯器的配合下,分別可以獲取到調用函數(准確講應該是成員)名稱,調用文件及調用行號。上面的TraceMessage函數可以實現如下:

public void TraceMessage(string message,
            [CallerMemberName] string memberName = "",
            [CallerFilePath] string sourceFilePath = "",
            [CallerLineNumber] int sourceLineNumber = 0)
    {
        Trace.WriteLine("message: " + message);
        Trace.WriteLine("member name: " + memberName);
        Trace.WriteLine("source file path: " + sourceFilePath);
        Trace.WriteLine("source line number: " + sourceLineNumber);
    } 

另外,在構造函數,析構函數、屬性等特殊的地方調用CallerMemberName屬性所標記的函數時,獲取的值有所不同,其取值如下表所示:

調用的地方 CallerMemberName獲取的結果
方法、屬性或事件 方法,屬性或事件的名稱
構造函數 字符串 “.ctor”
靜態構造函數 字符串 “.cctor”
析構函數 該字符串 “Finalize”
用戶定義的運算符或轉換 生成的名稱成員,例如, “op_Addition”。
特性構造函數 特性所應用的成員的名稱

例如,對於在屬性中調用CallerMemberName所標記的函數即可獲取屬性名稱,通過這種方式可以簡化 INotifyPropertyChanged 接口的實現。關於調用方信息更詳細的資料,請參看MSDN:http://msdn.microsoft.com/zh-cn/library/hh534540.aspx

 

C# 5.0五大新特性

第一:綁定運算符,:=:

這個只是簡化了數據綁定,跟ASP.NET MVC3不斷改進一樣,其實不是什么亮點改進。

 comboBox1.Text :=: textBox1.Text; //將文本框的內容綁定到下拉框。

第二:帶參數的泛型構造函數:

這個的加入給一些設計增加了強大功能,泛型早在C#2.0加入后就有着強大的應用,一般稍微設計比較好的框架,都會用到泛型,c#5.0加入帶參數泛型構造函數,則在原有基礎上對C#泛型完善了很多。:)

public class T MyClass : T: class, new()  
public class T MyClass : T:class, new(int)  

第三:支持null類型運算:

此功能,個人覺得並非什么大的亮點,但至少對null類型,特別是有數據計算的這種null類型的支持,寫代碼還是方便不少。

注意對於Nullable Types,在C#2.0就加入進來了,但是不支持計算,比如:

int? x = null;

int? y = x + 40; 
那么y值是多少?不支持計算,得到的是null,想必大家知道為什么結果是null了吧?但C#5.0可以,40加一個null的整數,我們要的結果是40,不過份吧?

    int x? = null;  
    int y? = x + 40;  
    Myobject obj = null;  
    Myotherobj obj2 = obj.MyProperty ??? new Myotherobj();  

第四:case支持表達式:

這個是一個我很早就想如果能這樣就好了,沒想到在C#5.0里就加入此功能,以前case里只能寫一個具體的常量,而現在可以加表達式了,靈活多了。

 switch(myobj){  
    llorEmpty(myotherobj):  
    //邏輯代碼  
    case myotherobj.Trim().Lower:  
    //邏輯代碼  
    }  

第五:擴展屬性。

我們在C#3.0里有擴展方法,那么在C#5.0里將會加入擴展屬性的感念,對照擴展方法,不難理解擴展屬性的概念了。以下為擴展屬性的定義舉例:

[Associate(string)]

public static int Zivsoft_ExtensionProperty { get;set;}

C#5.0 遠遠不只是上面描述的5點新功能,它如同C#4.0加入dynamic概念一樣,會加入異步處理概念,這個不是幾行代碼就能表達,而是將在設計,架構上,又會掀起一次飛躍……

為了大家搶先看,就給一段C#5.0一段簡單的異步操作的代碼例子,注意(C#5.0兩個新加的關鍵字async, await):

    Task<Movie> GetMovieAsync(string title);  

    Task PlayMovieAsync(Movie movie);  

    async void GetAndPlayMoviesAsync(string[] titles)  
    {  
        foreach (var title in titles)  
         {  
             var movie = await GetMovieAsync(title);  

             await PlayMovieAsync(movie);  
         }  
    }  

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM