C#中雙問號、雙冒號等幾個特殊關鍵字

1、@

這個東東看似和郵件有關啊,但是在C#的世界裏,可跟郵件沒有一毛錢關係,它是string的女朋友(當然了string有N多女友),二者結合就可以發揮作用了。你可以給它起個名字,叫做“逐字字符串”,或者別的什麼東東都行,這都不重要,關鍵你要會泡它~

在處理字符串時,那些個字符串轉義字符我們可傷不起,一個個的記又記不住,用了還要查,麻煩的不行,那@就是爲解決這個而誕生的,直接讓字符串原樣輸出有木有?什麼sql字符串了,還有路徑啥了的,統統搞定

string sql = @"select * from student where id = '001'";
//實際sql輸出select * from student where id = '001'
string path = @"C:\MDocu\student.xml";
//實際path輸出C:\MDocu\student.xml,注意這裏可不是\\哦

如果字符串裏邊包含雙引號時,要稍微處理一下,那就是在雙引號的外邊再加雙引號即可

string s = @"He said,""yes""";
//s輸出He said,"yes"

2、is

目的就一個,檢查變量是不是給定的類型,是就返回true,不是就false了,比較簡單,一筆帶過了

int i = 5;
bool check = i is int;//check = true

3、as

用於在兼容的引用類型之間執行轉換。例如

string s = someObject as string;
if (s != null)
{
    // someObject is a string.
}

as運算符類似於強制轉換操作;但是,如果轉換不可行,as會返回null而不是引發異常。更嚴格地說,這種形式的表達式 等效於

expression is type ? (type)expression : (type) null

as 運算符只執行引用轉換和裝箱轉換。as運算符無法執行其他轉換,如用戶定義的轉換,這類轉換應使用cast表達式來執行。

4、sizeof

用於獲取非託管類型的大小(以字節爲單位)

int intSize = sizeof(int);//intSize = 4

5、typeof

返回Type對象,該對象保存類型信息

Type myType = typeof(int);
console.writeline("Type:{}",myType);
//輸出Type:System.Int32

6、checked

檢測操作的溢出情況

short a =20000,b=20000;
short myShort = checked((short)(a+b));
//error

7、unchecked

忽略溢出關鍵字,接受結果而不管溢出情況,默認是不檢查溢出的

short a =20000,b=20000;
short myShort = checked((short)(a+b));
//忽略error

8、Guid

全局唯一標示符,是一個128位的字符串,在任何要以唯一方式來表示某個事物時就可以用該屌絲。

uniquecode = Guid.NewGuid ();
console.WriteLine("myCode:{}",uniquecode.ToString());
//輸出:myCode:cabfe0ba-fa72-4c5c-969f-e76821949ff1

9、?

可空類型

 

public class student
{
private string name;
private int? age=null;
public string Name
{
get { return name; }
set { name = value; }
}
public int? Age
{
get { return age; }
set { age = value; }
}
}
student s = new student();
s.Age = null;//是允許的

 

 

10、??

null接合操作符,也可以說是雙問號操作符,意思是取所賦值??左邊的,如果左邊爲null,取所賦值??右邊的

 

 

DateTime? createDate = null;
DateTime? defaultDate= null;
DateTime secondDate = DateTime.Now;
createDate = createDate ??defaultDate??secondDate;

// 如果createDate 爲空,則對defaultDate求值,如果defaultDate不爲空,則將defaultDate賦值給createDate 。否則繼續計算secondDate,是不是null都賦值給createDate ,因爲是最後一個表達式

 

 

11、::

兩個冒號表示作用域操作符。::操作符在其左操作數的作用域內找到其右操作數的名字。用於訪問某個命名空間中的名字,如std::cout,表明名字cout來自命名空間std。同樣的可以用來從某個類取名字,如string::size_type,表明size_type是string類定義的。

這裏面::前面是GAC的標示符global,用法比較特殊,和.不是一個類型的東西。

global 是 C# 2.0 中新增的關鍵字,理論上說,如果代碼寫得好的話,根本不需要用到它。 

假設你現在寫了一個類,名字叫 System。那麼當你再在代碼裏寫 System 的時候,編譯器就不知道你是要指你寫的 System 類還是系統的 System 命名空間,而 System 命名空間已經是根命名空間了,無法再通過完全限名來指定。在以前的 C# 版本中,這就是一個無法解決的問題。現在,可以通過 

global::System 

來表示 System 根命名空間,而用你自己的 
MyNamespace.System 
來表示自己的類。

另外,也可以用在命名空間別名上(也可以用  .  點號)在此示例中,命名空間 System 用於包括類 TestClass,因此必須使用 global::System.Console 來引用 System.Console 類,該類被System 命名空間隱藏。 而且,別名 colAlias 用於引用命名空間 System.Collections;因此,將使用此別名而不是命名空間來創建System.Collections.Hashtable 的實例。

 

 

using colAlias = System.Collections;
namespace System
{
    class TestClass
    {
        static void Main()
        {
            // Searching the alias:
            colAlias::Hashtable test = new colAlias::Hashtable();

            // Add items to the table.
            test.Add("A", "1");
            test.Add("B", "2");
            test.Add("C", "3");

            foreach (string name in test.Keys)
            {
                // Searching the global namespace:
                global::System.Console.WriteLine(name + " " + test[name]);
            }
        }
    }
}

 

 

12、=>

Lambda表達式的運算符是=>,運算符左邊列舉出了需要的參數,右邊定義了賦予Lambda變量的方法的實現代碼

List<User> user = new List<User>{ new User{Id=1,Name="LiSi",Age=22},  new User{Id=2,Name="ZhangSan",Age=25} }; 
//獲取特定人時所用的過濾條件,p參數屬於User類型  
var results = user.Where(p => p.Name == "LiSi").ToList();  //用User對象的Age值計算平均年齡  
var average = user.Average(p => p.Age); 

13、ref

 ref 關鍵字使參數按引用傳遞,也就是說它能夠讓你直接對原數進行操作,而不是對那個原數的Copy進行操作。若要使用 ref 參數,則方法定義和調用方法都必須顯式使用 ref 關鍵字,而且傳遞到 ref 參數的參數必須最先初始化,例如:

 

class RefExample
{
static void Method(ref int i)
{
i = 44;
}
static void Main()
{
int val = 0;
Method(ref val); // val is now 44
}
}

 

 

14、out

out是傳出參數,與ref有點像,但偏重於輸出,而且不用初始化,通過執行使用out參數的方法邏輯,out後面的數接受並返回這個值,比如你寫一個方法返回dataset,同時你還想返回頁數,怎麼辦?方法一般不能返回多個值啊,這個時候out就可以返回多個值,是不是很爽,你需要多個值得時候別忘了out這廝啊

 

 

public DataSet getData(out int count) 
{ 
    dataset ds=bll.getdata(10,20); 
獲取第11條到第20條數據,但是不可能只顯示共有10條記錄吧,那麼我們就可以用out了 
    int rcount=bll.GetCount();//比方說這個是取總記錄數的
    count=rcount; 

    return ds; 
} 

//顯示的時候 

public void showdata() 
{ 
    int count=0; 

    gridview1.datasource=getData(out count); 
    gridview1.databind(); 
   label1.text="共有"+count.tostring()+"條記錄"; 
} 

 

 

15、params

params主要的用處是在給函數傳參數的時候用,就是當函數的參數不固定的時候。在方法聲明中的 params 關鍵字之後不允許任何其他參數,並且在方法聲明中只允許一個 params 關鍵字!

注意事項:

(1)若形參表中含一個參數數組,則該參數數組必須位於形參列表的最後;

(2)參數數組必須是一維數組;

(3)不允許將params修飾符與ref和out修飾符組合起來使用;

(4)與參數數組對應的實參可以是同一類型的數組名,也可以是任意多個與該數組的元素屬於同一類型的變量;

(5)若實參是數組則按引用傳遞,若實參是變量或表達式則按值傳遞。

(6)用法:可變的方法參數,也稱數組型參數,適合於方法的參數個數不知的情況,用於傳遞大量的數組集合參數;當使用數組參數時,可通過使用params關鍵字在形參表中指定多種方法參數,並在方法的參數表中指定一個數組,形式爲:方法修飾符 返回類型 方法名(params 類型[] 變量名)

如帶有參數的SQL 語句,不同的表的字段數量也不同,當你更新修改的時候就可以用params

 16、using

這個再也熟悉不過了,常見三種用法

(1)引用命名空間,例如:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;

(2)創建別名(類或者命名空間的別名)

 

 

using MyControle=System.Console;
class UsingEx
{
   public static void Main()
   {
    MyConsole.WriteLine("應用了類的別名");
    }
}

 

 

(3)自動清理資源

using (SqlConnection conn = new SqlConnection(_connstr))
//這樣你就不用手工清理連接資源了

17、this

(1)表示當前實例

(2)索引器關鍵字

(3)隱藏父類同名方法的關鍵字

(4)擴展方法的關鍵字

18、<%= %>主要用於在前臺輸入後臺變量,比如後臺中有個public string a = "abc";前臺aspx頁面<%=a%>就可以取到後臺中a的值:abc

19、<%: %>是在asp.net mvc項目中綁ViewData用的,而且前提是視圖引擎用的是aspx的才行

20、__makeref、__reftype、__refvalue、__arglist 這類關鍵字才叫奇葩

看IL指令到mkrefany, 文檔中說它的作用是: "push a typed reference on the stack", 不知道在C#的何種語法會用上這條指令, 於是Google之, 發現了從來沒有看過的C#關鍵字:

 

Object obj = new Object();

TypedReference typedref = __makeref(obj);

Type type = __reftype(typedref);

Object sameObj = __refvalue( typedref,Object);

對應的IL是:

.method private hidebysig static void  Main(string[] args) cil managed
{
  .entrypoint
  // Code size       33 (0x21)
  .maxstack  1
  .locals init ([0] object obj,
           [1] typedref 'typedref',
           [2] class [mscorlib]System.Type 'type',
           [3] object sameObj)
  IL_0000:  nop
  IL_0001:  newobj     instance void [mscorlib]System.Object::.ctor()
  IL_0006:  stloc.0
  IL_0007:  ldloca.s   obj
  IL_0009:  mkrefany   [mscorlib]System.Object
  IL_000e:  stloc.1
  IL_000f:  ldloc.1
  IL_0010:  refanytype
  IL_0012:  call       class [mscorlib]System.Type [mscorlib]System.Type::GetTypeFromHandle(valuetype [mscorlib]System.RuntimeTypeHandle)
  IL_0017:  stloc.2
  IL_0018:  ldloc.1
  IL_0019:  refanyval  [mscorlib]System.Object
  IL_001e:  ldind.ref
  IL_001f:  stloc.3
  IL_0020:  ret
} // end of method Program::Main

可以發現:

TypedReference對象在IL中是typedref類型;

IL_0009使用mkrefany生成了一個類型爲Object的typedref;

IL_0010使用refanytype從typedref中得到了一個RuntimeTypeHandle, 隨即調用一個方法得到Type對象;

IL_0019使用refanyval從typedref中獲取了一個類型爲Object的引用, 從後面一句ldind.ref可以知道refanyval壓棧的是一個managed pointer(&類型), 而不是普通的reference, ldind.ref把棧頂的managed pointer轉換成了普通的reference.

 

這三個操作對應的也可以直接用TypedReference的靜態方法實現:

 

MyObj obj = new MyObj(99);

TypedReference tr = __makeref(obj); // TypedReference.MakeTypedReference          

Type type = Type.GetTypeFromHandle(TypedReference.TargetTypeToken(tr));

MyObj sameObj = (MyObj)TypedReference.ToObject(tr);

在C#的unsafe語境中可以使用&運算符獲取一個值類型量/對象的地址, 但不可以獲取一個引用類型對象的地址, 因爲引用類型字段值的分配完全受運行時控制, 但TypedReference可以看成爲任何託管對象的指針, typeref在CLI裏存在的理由是給所謂的動態語言提供一種動態的方式來訪問對象.

21、__arglist
大家都知道printf是一個不定參數數量的函數, 它的第二個參數是用...聲明的, 如果要在C#中使用P/Invoke應用這個函數該如何聲明呢? params是.NET中特有的不定參數數量的實現, 但我用

extern static int printf(string format, params object[] args);

聲明的printf永遠都不能正常工作(誰能?), 還好C#提供了一個__arglist關鍵字來支持古老的vararg, 如何使用__arglist聲明和調用printf見下面的代碼:

[DllImport("msvcrt.dll")]

extern static int printf(string format, __arglist);

static unsafe void Main(string[] args)

{

    printf("%d %.2f %s", __arglist(3, 0.4567, "asdf"));

}

 

我們甚至可以自己寫一個帶vararg參數的方法, TypedReference也派上用場了:

 

static void MyPrint(__arglist)

{

    ArgIterator itr = new ArgIterator(__arglist);

    while (itr.GetRemainingCount() > 0)

    {

        Console.WriteLine(TypedReference.ToObject(itr.GetNextArg()));

    }

}

 

static unsafe void Main(string[] args)

{

    MyPrint(__arglist("asdfasdf", 2, 1.2f, 2.4d, 123L, new object()));

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章