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()));