net的一個很重要的特性就是跨語言的程式設計,用C#寫的dll可以在VB.net裡調用,例如:
用C#寫的一個類,編譯到dll中,然後在VB.net中調用:
using System;
namespace CLSsample
{
public class CLSTest
{
public CLSTest()
{
}
public void ABC()
{
Console.WriteLine("ABC");
}
}
}
在VB.net中調用:
Dim c As CLSsample.CLSTest = New CLSsample.CLSTest
c.ABC()
現在給dll中的CLSTest類加一個函數:
public void abc()
{
Console.WriteLine("abc");
}
先編譯C#寫的這個dll,再編譯VB.net工程,編譯出現問題,提示資訊如下:
重載決策失敗,原因是沒有可通路的“ABC”最适合這些參數:
'Public Sub abc()': 不是最适合。
'Public Sub ABC()': 不是最适合。
原因很簡單,因為C#是區分大小寫的,但是VB.net不區分。
而真正的原因在于用C#寫的這個類是不符合CLS(公共語言規範)的。
現在在命名空間前面加上一句,标示編譯時確定不包含其它語言無法使用的内容:
[assembly:CLSCompliant(true)]
namespace CLSsample
{
......
public void ABC()
{
Console.WriteLine("ABC");
}
public void abc()
{
Console.WriteLine("abc");
}
}
這時候再編譯,就會出現錯誤,提示資訊:
隻在大小寫不同的辨別符“CLSsample.CLSTest.abc()”不符合 CLS
要編譯通過,在函數abc前加上:
[CLSCompliantAttribute(false)]
指出 abc函數 是不符合 CLS 的
下面是MSDN中對CLS的部分說明:
CLS 在設計上足夠大,可以包括開發人員經常需要的語言構造;同時也足夠小,
大多數語言都可以支援它。此外,任何不可能快速驗證代碼類型安全性的語言
構造都被排除在 CLS 之外,以便所有符合 CLS 的語言都可以生成可驗證的代碼
(如果它們選擇這樣做)。
也就是說開發的類庫必須遵守CLS才可以更好的被其它的語言所使用。否則就像
上面的情況,用C#開發的動态連結庫在VB.net中就出現了問題,特别是開發一些
通用的類庫的時候,就更需要注意這一點。
看看對.net framework中Int32反編譯的部分結果(用的是Reflector):
[CLSCompliant(false)]
uint System.IConvertible.ToUInt32(IFormatProvider provider);
這是因為有的語言不支援UInt32類型。
再看看對UInt32反編譯的結果,UInt32的聲明:
[Serializable, StructLayout(LayoutKind.Sequential), CLSCompliant(false)]
public struct UInt32 : IComparable, IFormattable, IConvertible
{
}
在前陣子的一個随筆:什麼是CLS?
“CLS是程式設計語言設計者和類庫設計者之間的一個約定”
現在體會得更深了。
剛開始系統學習.net架構,有錯誤的地方,還請大家不吝賜教