色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

C#模式匹配完全指南

新機器視覺 ? 來源:hez2010 ? 2023-09-13 17:33 ? 次閱讀

前言 自從 2017 年 C# 7.0 版本開始引入聲明模式和常數模式匹配開始,到 2022 年的 C# 11 為止,最后一個板塊列表模式和切片模式匹配也已經補齊,當初計劃的模式匹配內容已經基本全部完成。 C# 在模式匹配方面下一步計劃則是支持活動模式(active pattern),這一部分將在本文最后進行介紹,而在介紹未來的模式匹配計劃之前,本文主題是對截止 C# 11 模式匹配的~~(不)~~完全指南,希望能對各位開發者們提升代碼編寫效率、可讀性和質量有所幫助。 模式匹配 要使用模式匹配,首先要了解什么是模式。在使用正則表達式匹配字符串時,正則表達式自己就是一個模式,而對字符串使用這段正則表達式進行匹配的過程就是模式匹配。而在代碼中也是同樣的,我們對對象采用某種模式進行匹配的過程就是模式匹配。

C# 11 支持的模式有很多,包含:

  • 聲明模式(declaration pattern)
  • 類型模式(type pattern)
  • 常數模式(constant pattern)
  • 關系模式(relational pattern)
  • 邏輯模式(logical pattern)
  • 屬性模式(property pattern)
  • 位置模式(positional pattern)
  • var 模式(var pattern)
  • 丟棄模式(discard pattern)
  • 列表模式(list pattern)
  • 切片模式(slice pattern)

而其中,不少模式都支持遞歸,也就意味著可以模式嵌套模式,以此來實現更加強大的匹配功能。

模式匹配可以通過switch表達式來使用,也可以在普通的switch語句中作為case使用,還可以在if條件中通過is來使用。本文主要在switch表達式中使用模式匹配。

那么接下來就對這些模式進行介紹。

實例:表達式計算器

為了更直觀地介紹模式匹配,我們接下來利用模式匹配來編寫一個表達式計算器。

為了編寫表達式計算器,首先我們需要對表達式進行抽象:

publicabstractpartialclassExpr<T>whereT:IBinaryNumber<T>
{
publicabstractTEval(params(stringName,TValue)[]args);
}

我們用上面這個Expr來表示一個表達式,其中T是操作數的類型,然后進一步將表達式分為常數表達式ConstantExpr參數表達式ParameterExpr、一元表達式UnaryExpr、二元表達式BinaryExpr和三元表達式TernaryExpr。最后提供一個Eval方法,用來計算表達式的值,該方法可以傳入一個args來提供表達式計算所需要的參數。

有了一、二元表達式自然也需要運算符,例如加減乘除等,我們也同時定義Operator來表示運算符:

publicabstractrecordOperator
{
publicrecordUnaryOperator(OperatorsOperator):Operator;
publicrecordBinaryOperator(BinaryOperatorsOperator):Operator;
}

然后設置允許的運算符,其中前三個是一元運算符,后面的是二元運算符:

publicenumOperators
{
[Description("~")]Inv,[Description("-")]Min,[Description("!")]LogicalNot,
[Description("+")]Add,[Description("-")]Sub,[Description("*")]Mul,[Description("/")]Div,
[Description("&")]And,[Description("|")]Or,[Description("^")]Xor,
[Description("==")]Eq,[Description("!=")]Ne,
[Description(">")]Gt,[Description("<")]Lt,[Description(">=")]Ge,[Description("<=")]Le,
[Description("&&")]LogicalAnd,[Description("||")]LogicalOr,
}

你可以能會好奇對T的運算能如何實現邏輯與或非,關于這一點,我們直接使用0來代表false,非0代表true

接下來就是分別實現各類表達式的時間!

常數表達式

常數表達式很簡單,它保存一個常數值,因此只需要在構造方法中將用戶提供的值存儲下來。它的Eval實現也只需要簡單返回存儲的值即可:

publicabstractpartialclassExpr<T>whereT:IBinaryNumber<T>
{
publicclassConstantExpr:Expr<T>
{
publicConstantExpr(Tvalue)=>Value=value;

publicTValue{get;}
publicvoidDeconstruct(outTvalue)=>value=Value;

publicoverrideTEval(params(stringName,TValue)[]args)=>Value;
}
}

參數表達式

參數表達式用來定義表達式計算過程中的參數,允許用戶在對表達式執行Eval計算結果的時候傳參,因此只需要存儲參數名。它的Eval實現需要根據參數名在args中找出對應的參數值:

publicabstractpartialclassExpr<T>whereT:IBinaryNumber<T>
{
publicclassParameterExpr:Expr<T>
{
publicParameterExpr(stringname)=>Name=name;

publicstringName{get;}
publicvoidDeconstruct(outstringname)=>name=Name;

//對args進行模式匹配
publicoverrideTEval(params(stringName,TValue)[]args)=>argsswitch
{
//如果args有至少一個元素,那我們把第一個元素拿出來存為(name,value),
//然后判斷 name 是否和本參數表達式中存儲的參數名 Name 相同。
//如果相同則返回 value,否則用 args 除去第一個元素剩下的參數繼續匹配。
[var(name,value),..vartail]=>name==Name?value:Eval(tail),
//如果args是空列表,則說明在args中沒有找到名字和Name相同的參數,拋出異常
[]=>thrownewInvalidOperationException($"Expectedanargumentnamed{Name}.")
};
}
}

模式匹配會從上往下依次進行匹配,直到匹配成功為止。

上面的代碼中你可能會好奇[var (name, value), .. var tail]是個什么模式,這個模式整體看是列表模式,并且列表模式內組合使用聲明模式、位置模式和切片模式。例如:

  • []:匹配一個空列表。
  • [1, _, 3]:匹配一個長度是 3,并且首尾元素分別是 1、3 的列表。其中_是丟棄模式,表示任意元素。
  • [_, .., 3]:匹配一個末元素是 3,并且 3 不是首元素的列表。其中..是切片模式,表示任意切片。
  • [1, ..var tail]:匹配一個首元素是 1 的列表,并且將除了首元素之外元素的切片賦值給tail。其中var tailvar模式,用于將匹配結果賦值給變量。
  • [var head, ..var tail]:匹配一個列表,將它第一個元素賦值給head,剩下元素的切片賦值給tail,這個切片里可以沒有元素。
  • [var (name, value), ..var tail]:匹配一個列表,將它第一個元素賦值給(name, value),剩下元素的切片賦值給tail,這個切片里可以沒有元素。其中(name, value)是位置模式,用于將第一個元素的解構結果根據位置分別賦值給namevalue,也可以寫成(var name, var value)

一元表達式

一元表達式用來處理只有一個操作數的計算,例如非、取反等。

publicabstractpartialclassExpr<T>whereT:IBinaryNumber<T>
{
publicclassUnaryExpr:Expr<T>
{
publicUnaryExpr(UnaryOperatorop,Exprexpr)=>(Op,Expr)=(op,expr);

publicUnaryOperatorOp{get;}
publicExprExpr{get;}
publicvoidDeconstruct(outUnaryOperatorop,outExprexpr)=>(op,expr)=(Op,Expr);

//對Op進行模式匹配
publicoverrideTEval(params(stringName,TValue)[]args)=>Opswitch
{
//如果Op是UnaryOperator,則將其解構結果賦值給op,然后對op進行匹配,op是一個枚舉,而.NET中的枚舉值都是整數
UnaryOperator(varop)=>opswitch
{
//如果op是Operators.Inv
Operators.Inv=>~Expr.Eval(args),
//如果op是Operators.Min
Operators.Min=>-Expr.Eval(args),
//如果op是Operators.LogicalNot
Operators.LogicalNot=>Expr.Eval(args)==T.Zero?T.One:T.Zero,
//如果op的值大于LogicalNot或者小于0,表示不是一元運算符
>Operators.LogicalNotor0=>thrownewInvalidOperationException($"Expectedanunaryoperator,butgot{op}.")
},
//如果Op不是UnaryOperator
_=>thrownewInvalidOperationException("Expectedanunaryoperator.")
};
}
}

上面的代碼中,首先利用了 C# 元組可作為左值的特性,分別使用一行代碼就做完了構造方法和解構方法的賦值:(Op, Expr) = (op, expr)(op, expr) = (Op, Expr)。如果你好奇能否利用這個特性交換多個變量,答案是可以!

Eval中,首先將類型模式、位置模式和聲明模式組合成UnaryOperator(var op),表示匹配UnaryOperator類型、并且能解構出一個元素的東西,如果匹配則將解構出來的那個元素賦值給op

然后我們接著對解構出來的op進行匹配,這里用到了常數模式,例如Operators.Inv用來匹配op是否是Operators.Inv。常數模式可以使用各種常數對對象進行匹配。

這里的> Operators.LogicalNot< 0則是關系模式,分別用于匹配大于Operators.LogicalNot的值和小于0的指。然后利用邏輯模式or將兩個模式組合起來表示或的關系。邏輯模式除了or之外還有andnot

由于我們在上面窮舉了枚舉中所有的一元運算符,因此也可以將> Operators.LogicalNot or < 0換成丟棄模式_或者 var 模式var foo,兩者都用來匹配任意的東西,只不過前者匹配到后直接丟棄,而后者聲明了個變量foo將匹配到的值放到里面:

opswitch
{
//...
_=>thrownewInvalidOperationException($"Expectedanunaryoperator,butgot{op}.")
}

opswitch
{
//...
varfoo=>thrownewInvalidOperationException($"Expectedanunaryoperator,butgot{foo}.")
}

二元表達式

二元表達式用來表示操作數有兩個的表達式。有了一元表達式的編寫經驗,二元表達式如法炮制即可。

publicabstractpartialclassExpr<T>whereT:IBinaryNumber<T>
{
publicclassBinaryExpr:Expr<T>
{
publicBinaryExpr(BinaryOperatorop,Exprleft,Exprright)=>(Op,Left,Right)=(op,left,right);

publicBinaryOperatorOp{get;}
publicExprLeft{get;}
publicExprRight{get;}
publicvoidDeconstruct(outBinaryOperatorop,outExprleft,outExprright)=>(op,left,right)=(Op,Left,Right);

publicoverrideTEval(params(stringName,TValue)[]args)=>Opswitch
{
BinaryOperator(varop)=>opswitch
{
Operators.Add=>Left.Eval(args)+Right.Eval(args),
Operators.Sub=>Left.Eval(args)-Right.Eval(args),
Operators.Mul=>Left.Eval(args)*Right.Eval(args),
Operators.Div=>Left.Eval(args)/Right.Eval(args),
Operators.And=>Left.Eval(args)&Right.Eval(args),
Operators.Or=>Left.Eval(args)|Right.Eval(args),
Operators.Xor=>Left.Eval(args)^Right.Eval(args),
Operators.Eq=>Left.Eval(args)==Right.Eval(args)?T.One:T.Zero,
Operators.Ne=>Left.Eval(args)!=Right.Eval(args)?T.One:T.Zero,
Operators.Gt=>Left.Eval(args)>Right.Eval(args)?T.One:T.Zero,
Operators.Lt=>Left.Eval(args)Left.Eval(args)>=Right.Eval(args)?T.One:T.Zero,
Operators.Le=>Left.Eval(args)<=?Right.Eval(args)???T.One?:?T.Zero,
????????????????Operators.LogicalAnd?=>Left.Eval(args)==T.Zero||Right.Eval(args)==T.Zero?T.Zero:T.One,
Operators.LogicalOr=>Left.Eval(args)==T.Zero&&Right.Eval(args)==T.Zero?T.Zero:T.One,
Operators.LogicalOr=>thrownewInvalidOperationException($"Unexpectedabinaryoperator,butgot{op}.")
},
_=>thrownewInvalidOperationException("Unexpectedabinaryoperator.")
};
}
}

同理,也可以將< Operators.Add or > Operators.LogicalOr換成丟棄模式或者 var 模式。

三元表達式

三元表達式包含三個操作數:條件表達式Cond、為真的表達式Left、為假的表達式Right。該表達式中會根據Cond是否為真來選擇取Left還是Right,實現起來較為簡單:

publicabstractpartialclassExpr<T>whereT:IBinaryNumber<T>
{
publicclassTernaryExpr:Expr<T>
{
publicTernaryExpr(Exprcond,Exprleft,Exprright)=>(Cond,Left,Right)=(cond,left,right);

publicExprCond{get;}
publicExprLeft{get;}
publicExprRight{get;}
publicvoidDeconstruct(outExprcond,outExprleft,outExprright)=>(cond,left,right)=(Cond,Left,Right);

publicoverrideTEval(params(stringName,TValue)[]args)=>Cond.Eval(args)==T.Zero?Right.Eval(args):Left.Eval(args);
}
}

完成。我們用了僅僅幾十行代碼就完成了全部的核心邏輯!這便是模式匹配的強大之處:簡潔、直觀且高效。

表達式判等

至此為止,我們已經完成了所有的表達式構造、解構和計算的實現。接下來我們為每一個表達式實現判等邏輯,即判斷兩個表達式(字面上)是否相同。

例如a == b ? 2 : 4a == b ? 2 : 5不相同,a == b ? 2 : 4c == d ? 2 : 4不相同,而a == b ? 2 : 4a == b ? 2 : 4相同。

為了實現該功能,我們重寫每一個表達式的EqualsGetHashCode方法。

常數表達式

常數表達式判等只需要判斷常數值是否相等即可:

publicoverrideboolEquals(object?obj)=>objisConstantExpr(varvalue)&&value==Value;
publicoverrideintGetHashCode()=>Value.GetHashCode();

參數表達式

參數表達式判等只需要判斷參數名是否相等即可:

publicoverrideboolEquals(object?obj)=>objisParameterExpr(varname)&&name==Name;
publicoverrideintGetHashCode()=>Name.GetHashCode();

一元表達式

一元表達式判等,需要判斷被比較的表達式是否是一元表達式,如果也是的話則判斷運算符和操作數是否相等:

publicoverrideboolEquals(object?obj)=>objisUnaryExpr({Operator:varop},varexpr)&&(op,expr).Equals((Op.Operator,Expr));
publicoverrideintGetHashCode()=>(Op,Expr).GetHashCode();

上面的代碼中用到了屬性模式{ Operator: var op },用來匹配屬性的值,這里直接組合了聲明模式將屬性Operator的值賦值給了expr。另外,C# 中的元組可以組合起來進行判等操作,因此不需要寫op.Equals(Op.Operator) && expr.Equals(Expr),而是可以直接寫(op, expr).Equals((Op.Operator, Expr))

二元表達式

和一元表達式差不多,區別在于這次多了一個操作數:

publicoverrideboolEquals(object?obj)=>objisBinaryExpr({Operator:varop},varleft,varright)&&(op,left,right).Equals((Op.Operator,Left,Right));
publicoverrideintGetHashCode()=>(Op,Left,Right).GetHashCode();

三元表達式

和二元表達式差不多,只不過運算符Op變成了操作數Cond

publicoverrideboolEquals(object?obj)=>objisTernaryExpr(varcond,varleft,varright)&&cond.Equals(Cond)&&left.Equals(Left)&&right.Equals(Right);
publicoverrideintGetHashCode()=>(Cond,Left,Right).GetHashCode();

到此為止,我們為所有的表達式都實現了判等。

一些工具方法

我們重載一些Expr的運算符方便我們使用:

publicstaticExproperator~(Exproperand)=>newUnaryExpr(new(Operators.Inv),operand);
publicstaticExproperator!(Exproperand)=>newUnaryExpr(new(Operators.LogicalNot),operand);
publicstaticExproperator-(Exproperand)=>newUnaryExpr(new(Operators.Min),operand);
publicstaticExproperator+(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Add),left,right);
publicstaticExproperator-(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Sub),left,right);
publicstaticExproperator*(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Mul),left,right);
publicstaticExproperator/(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Div),left,right);
publicstaticExproperator&(Exprleft,Exprright)=>newBinaryExpr(new(Operators.And),left,right);
publicstaticExproperator|(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Or),left,right);
publicstaticExproperator^(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Xor),left,right);
publicstaticExproperator>(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Gt),left,right);
publicstaticExproperator<(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Lt),left,right);
publicstaticExproperator>=(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Ge),left,right);
publicstaticExproperator<=(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Le),left,right);
publicstaticExproperator==(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Eq),left,right);
publicstaticExproperator!=(Exprleft,Exprright)=>newBinaryExpr(new(Operators.Ne),left,right);
publicstaticimplicitoperatorExpr(Tvalue)=>newConstantExpr(value);
publicstaticimplicitoperatorExpr(stringname)=>newParameterExpr(name);
publicstaticimplicitoperatorExpr(boolvalue)=>newConstantExpr(value?T.One:T.Zero);

publicoverrideboolEquals(object?obj)=>base.Equals(obj);
publicoverrideintGetHashCode()=>base.GetHashCode();

由于重載了==!=,編譯器為了保險起見提示我們重寫EqualsGetHashCode,這里實際上并不需要重寫,因此直接調用base上的方法保持默認行為即可。

然后編寫兩個擴展方法用來方便構造三元表達式,和從Description中獲取運算符的名字:

publicstaticclassExtensions
{
publicstaticExprSwitch(thisExprcond,Exprleft,Exprright)whereT:IBinaryNumber=>newExpr.TernaryExpr(cond,left,right);
publicstaticstring?GetName(thisTop)whereT:Enum=>typeof(T).GetMember(op.ToString()).FirstOrDefault()?.GetCustomAttribute()?.Description;
}

由于有參數表達式參與時需要我們提前提供參數值才能調用Eval進行計算,因此我們寫一個交互式的Eval來在計算過程中遇到參數表達式時提示用戶輸入值,起名叫做InteractiveEval

publicTInteractiveEval()
{
varnames=Array.Empty<string>();
returnEval(GetArgs(this,refnames,refnames));
}
privatestaticTGetArg(stringname,refstring[]names)
{
Console.Write($"Parameter{name}:");
string?str;
do{str=Console.ReadLine();}
while(strisnull);
names=names.Append(name).ToArray();
returnT.Parse(str,NumberStyles.Number,null);
}
privatestatic(stringName,TValue)[]GetArgs(Exprexpr,refstring[]assigned,refstring[]filter)=>exprswitch
{
TernaryExpr(varcond,varleft,varright)=>GetArgs(cond,refassigned,refassigned).Concat(GetArgs(left,refassigned,refassigned)).Concat(GetArgs(right,refassigned,refassigned)).ToArray(),
BinaryExpr(_,varleft,varright)=>GetArgs(left,refassigned,refassigned).Concat(GetArgs(right,refassigned,refassigned)).ToArray(),
UnaryExpr(_,varuexpr)=>GetArgs(uexpr,refassigned,refassigned),
ParameterExpr(varname)=>filterswitch
{
[varhead,..]whenhead==name=>Array.Empty<(stringName,TValue)>(),
[_,..vartail]=>GetArgs(expr,refassigned,reftail),
[]=>new[]{(name,GetArg(name,refassigned))}
},
_=>Array.Empty<(stringName,TValue)>()
};

這里在GetArgs方法中,模式[var head, ..]后面跟了一個when head == name,這里的when用來給模式匹配指定額外的條件,僅當條件滿足時才匹配成功,因此[var head, ..] when head == name的含義是,匹配至少含有一個元素的列表,并且將頭元素賦值給head,且僅當head == name時匹配才算成功。

最后我們再重寫ToString方法方便輸出表達式,就全部大功告成了。

測試

接下來讓我測試測試我們編寫的表達式計算器:

Expr<int>a=4;
Expr<int>b=-3;
Expr<int>x="x";
Expr<int>c=!((a+b)*(a-b)>x);
Expr<int>y="y";
Expr<int>z="z";
Expr<int>expr=(c.Switch(y,z)-a>x).Switch(z+a,y/b);
Console.WriteLine(expr);
Console.WriteLine(expr.InteractiveEval());

運行后得到輸出:

((((! ((((4) + (-3)) * ((4) - (-3))) > (x))) ? (y) : (z)) - (4)) > (x)) ? ((z) + (4)) : ((y) / (-3))

然后我們給xyz分別設置成 42、27 和 35,即可得到運算結果:

Parameterx:42
Parametery:27
Parameterz:35
-9

再測測表達式判等邏輯:

Expr<int>expr1,expr2,expr3;
{
Expr<int>a=4;
Expr<int>b=-3;
Expr<int>x="x";
Expr<int>c=!((a+b)*(a-b)>x);
Expr<int>y="y";
Expr<int>z="z";
expr1=(c.Switch(y,z)-a>x).Switch(z+a,y/b);
}

{
Expr<int>a=4;
Expr<int>b=-3;
Expr<int>x="x";
Expr<int>c=!((a+b)*(a-b)>x);
Expr<int>y="y";
Expr<int>z="z";
expr2=(c.Switch(y,z)-a>x).Switch(z+a,y/b);
}

{
Expr<int>a=4;
Expr<int>b=-3;
Expr<int>x="x";
Expr<int>c=!((a+b)*(a-b)>x);
Expr<int>y="y";
Expr<int>w="w";
expr3=(c.Switch(y,w)-a>x).Switch(w+a,y/b);
}

Console.WriteLine(expr1.Equals(expr2));
Console.WriteLine(expr1.Equals(expr3));

得到輸出:

True
False

活動模式

在未來,C# 將會引入活動模式,該模式允許用戶自定義模式匹配的方法,例如:

staticboolEven(thisTvalue)whereT:IBinaryInteger=>value%2==0;

上述代碼定義了一個T的擴展方法Even,用來匹配value是否為偶數,于是我們便可以這么使用:

varx=3;
vary=xswitch
{
Even()=>"even",
_=>"odd"
};

此外,該模式還可以和解構模式結合,允許用戶自定義解構行為,例如:

staticboolInt(thisstringvalue,outintresult)=>int.TryParse(value,outresult);

然后使用的時候:

varx="3";
vary=xswitch
{
Int(varresult)=>result,
_=>0
};

即可對x這個字符串進行匹配,如果x可以被解析為int,就取解析結果result,否則取 0。

總結

模式匹配極大的方便了我們編寫出簡潔且可讀性高的高質量代碼,并且會自動幫我們做窮舉檢查,防止我們漏掉情況。此外,使用模式匹配時,編譯器也會幫我們優化代碼,減少完成匹配所需要的比較次數,最終減少分支并提升運行效率。

本文中的例子為了覆蓋到全部的模式,不一定采用了最優的寫法,這一點各位讀者們也請注意。

本文中的表達式計算器全部代碼可以前往


聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 字符串
    +關注

    關注

    1

    文章

    585

    瀏覽量

    20577
  • 代碼
    +關注

    關注

    30

    文章

    4821

    瀏覽量

    68890
  • 模式
    +關注

    關注

    0

    文章

    65

    瀏覽量

    13414

原文標題:C# 模式匹配完全指南

文章出處:【微信號:vision263com,微信公眾號:新機器視覺】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    C#模式匹配入門指南

    自從 2017 年 C# 7.0 版本開始引入聲明模式和常數模式匹配開始,到 2022 年的 C# 11 為止,最后一個板塊列表
    的頭像 發表于 09-18 09:36 ?734次閱讀

    C#上位機顯示不完全

    我用原子的板子和C#上位機通訊。上位機顯示不完全。但是用網絡助手通訊的話,顯示是完全的。所以應該是我C#上位機的問題。大家幫忙看看,是什么問題。以下 是代碼CSharpTCP.rar
    發表于 04-02 23:26

    C#完全手冊

    C#語言概述點NET編程語言C#運行環境編寫第一個程序C#程序設計基礎數據類型變量和和常量......
    發表于 05-21 22:00 ?144次下載

    C#教程之DisplayRowCount

    C#教程之DisplayRowCount,很好的C#資料,快來學習吧。
    發表于 04-20 09:59 ?8次下載

    C#教程之FileBatchCopy

    C#教程之FileBatchCopy,很好的C#資料,快來學習吧。
    發表于 04-20 09:59 ?12次下載

    C#教程之彈出模式窗口顯示進度條

    C#教程之彈出模式窗口顯示進度條,很好的C#資料,快來學習吧。
    發表于 04-20 10:49 ?7次下載

    C#教程之組合

    C#教程之組合,很好的C#資料,快來學習吧。
    發表于 04-20 10:50 ?15次下載

    C#教程之LoadFLASH

    C#教程之LoadFLASH,很好的C#資料,快來學習吧。
    發表于 04-20 10:50 ?9次下載

    C#教程之PrintRemitBill

    C#教程之PrintRemitBill,很好的C#資料,快來學習吧。
    發表于 04-20 13:50 ?16次下載

    C#教程之PrintTopFive

    C#教程之PrintTopFive,很好的C#資料,快來學習吧。
    發表于 04-20 13:50 ?5次下載

    C#教程之VPrintt

    C#教程之VPrintt,很好的C#資料,快來學習吧。
    發表于 04-20 14:06 ?10次下載

    C#教程之WordToHtml

    C#教程之WordToHtml,很好的C#資料,快來學習吧。
    發表于 04-20 14:46 ?15次下載

    《Visual C# 2005開發技術》C#與.NET Fram

    《Visual C# 2005開發技術》C#與.NET Framework簡介
    發表于 02-07 15:11 ?0次下載

    《Visual C# 2005開發技術》C#程序設計基礎

    《Visual C# 2005開發技術》C#程序設計基礎
    發表于 02-07 15:11 ?0次下載

    C#上位機實戰開發指南

    C#上位機實戰開發指南
    發表于 11-22 19:25 ?0次下載
    主站蜘蛛池模板: 国产美熟女乱又伦AV| 亚洲精品蜜桃AV久久久| 皮皮色狼网| 亚洲 视频 在线 国产 精品| 13一18TV处流血TV| 国产成人精品免费视频软件 | 影音先锋色小姐| 成人在线视频免费看| 久久99精品AV99果冻| 伸到同桌奶罩里捏她胸h| 91久久精一区二区三区大全| 国产网红主播精品福利大秀专区| 欧美Av无码高清在线| 一边捏奶头一边啪高潮会怎么样| 宫交拔不出来了h黑人| 美女胸网站| 伊人久99久女女视频精品免| 国产精品人妻无码免费A片导航 | 国语大学生自产拍在线观看| 日本69xxxxx| 99精品免费在线观看| 久久国产av偷拍在线| 亚洲色在线| 含羞草在线免费观看| 乌克兰少妇大胆大BBW| 办公室里做好紧好爽H| 欧美video巨大粗暴18| 91av欧美| 免费观看成人毛片| 99国产精品| 乱VODAFONEWIFI熟妇| 中文字幕日本久久2019| 久久精品亚洲| 永久免费精品精品永久-夜色| 国内免费视频成人精品| 亚洲精品人成电影网| 狠日狠干日曰射| 亚洲视频中文字幕| 久久91精品久久久久久水蜜桃| 亚洲视频在线观看免费| 精品成人片深夜|