C#面試題(轉載) SQL Server 數據庫基礎筆記分享(下) SQL Server 數據庫基礎筆記分享(上) Asp.Net MVC4中的全局過濾器 C#語法——泛型的多種應用

C#面試題(轉載)

原文地址:100道C#面試題(.net開發人員必備)  https://blog.csdn.net/u013519551/article/details/51220841html

 

1. .NET和C#有什麼區別程序員

答:.NET通常指 .NET FrameWork框架,它是一種平臺,一種技術。web

C#是一種編程語言,能夠基於.NET平臺的應用。面試

2.一列數的規則以下: 一、一、二、三、五、八、1三、2一、34...... 求第30位數是多少,用遞歸算法實現。
答:public class MainClass 
                 { 
                 public static void Main() 
                 { 
                 Console.WriteLine(Foo(30)); 
                 } 
                 public static int Foo(int i) 
                 { 
                 if (i <= 0) 
                 return 0; 
                 else if(i > 0 && i <= 2) 
                 return 1; 
                 else return Foo(i -1) + Foo(i - 2); 
                 } 
                 }算法

3. C#中的委託是什麼?事件是否是一種委託?
答 : 委託能夠把一個方法做爲參數代入另外一個方法。
                 委託能夠理解爲指向一個函數的引用。
                 是,是一種特殊的委託sql

4. 簡述 private、 protected、 public、internal 修飾符的訪問權限。數據庫

答 . private : 私有成員, 在類的內部才能夠訪問。編程

protected: 保護成員,該類內部和繼承類中能夠訪問。c#

public: 公共成員,徹底公開,沒有訪問限制。設計模式

internal:在同一命名空間內能夠訪問。

5. override與重載的區別

答 :override 與重載的區別。重載是方法的名稱相同。參數或參數類型不一樣,進行多

次重載以適應不一樣的須要

Override是進行基類中函數的重寫。爲了適應須要。

6.若是在一個B/S結構的系統中須要傳遞變量值,可是又不能使用Session、Cookie、Application,您有幾種方法進行處理?
答 :   this.Server.Transfer

7. 請編程遍歷頁面上全部TextBox控件並給它賦值爲string.Empty?
答:
foreach (System.Windows.Forms.Control control in this.Controls)
{
if (control is System.Windows.Forms.TextBox)
{
System.Windows.Forms.TextBox tb = (System.Windows.Forms.TextBox)control ; 
tb.Text = String.Empty ;
}
}

8. 請編程實現一個冒泡排序算法?
答:
int [] array = new int [*] ;
int temp = 0 ;
for (int i = 0 ; i < array.Length - 1 ; i++)
{
for (int j = i + 1 ; j < array.Length ; j++)
{
if (array[j] < array[i])
{
temp = array[i] ;
array[i] = array[j] ;
array[j] = temp ;
}
}
}

9. 描述一下C#中索引器的實現過程,是否只能根據數字進行索引?
答:不是。能夠用任意類型。

10. 求如下表達式的值,寫出您想到的一種或幾種實現方法:1-2+3-4+……+m
答:
int Num = this.TextBox1.Text.ToString() ;
int Sum = 0 ;
for (int i = 0 ; i < Num + 1 ; i++)
{
if((i%2) == 1)
{
Sum += i ;
}
else
{
Sum = Sum - I ;
}
}
System.Console.WriteLine(Sum.ToString());
System.Console.ReadLine() ;

11. 在下面的例子裏
using System;
class A
{
public A()
{
PrintFields();
}
public virtual void PrintFields(){}
}
class B:A
{
int x=1;
int y;
public B()
{
y=-1;
}
public override void PrintFields()
{
Console.WriteLine("x={0},y={1}",x,y);
}
當使用new B()建立B的實例時,產生什麼輸出?
答:X=1,Y=0;x= 1 y = -1

12. CTS、CLS、CLR分別做何解釋?
答:CTS:通用語言系統。CLS:通用語言規範。CLR:公共語言運行庫。

13. 什麼是裝箱和拆箱?
答:從值類型接口轉換到引用類型裝箱。從引用類型轉換到值類型拆箱。

14. 什麼是受管制的代碼?
答:unsafe:非託管代碼。不通過CLR運行。

15.什麼是強類型系統?

答:RTTI:類型識別系統。

16.net中讀寫數據庫須要用到那些類?他們的做用?

答:DataSet:數據存儲器。

DataCommand:執行語句命令。

DataAdapter:數據的集合,用語填充。

17.列舉ASP.NET頁面之間傳遞值的幾種方式。 
答. 1).使用QueryString, 如....?id=1; response. Redirect().... 
                 2).使用Session變量 
                 3).使用Server.Transfer

18.什麼是Code-Behind技術?

答:代碼後植。

19.在.net中,配件的意思是?

答:程序集。(中間語言,源數據,資源,裝配清單)

20.經常使用的調用WebService的方法有哪些?

答:1.使用WSDL.exe命令行工具。

2.使用VS.NET中的Add Web Reference菜單選項

21..netRemoting 的工做原理是什麼?

答:服務器端向客戶端發送一個進程編號,一個程序域編號,以肯定對象的位置

22.在C#中,string str = null 與 string str = 「」 請儘可能使用文字或圖

象說明其中的區別。

答:string str = null 是不給他分配內存空間,而string str = "" 給它分配

長度爲空字符串的內存空間。

23.請詳述在dotnet中類(class)與結構(struct)的異同?

答:Class能夠被實例化,屬於引用類型,是分配在內存的堆上的,Struct屬於值類

型,是分配在內存的棧上的.

24.分析如下代碼,完成填空

stringstrTmp = "abcdefg某某某";

inti= System.Text.Encoding.Default.GetBytes(strTmp).Length;

intj= strTmp.Length;

以上代碼執行完後,i= j=

答:i=13,j=10

25.SQLSERVER服務器中,給定表table1 中有兩個字段 ID、LastUpdateDate,

ID表示更新的事務號,LastUpdateDate表示更新時的服務器時間,請使用一句

SQL語句得到最後更新的事務號

答:Select ID FROM table1 Where LastUpdateDate = (Select MAX

(LastUpdateDate)FROM table1)

26.簡要談一下您對微軟.NET構架下remoting和webservice兩項技術的理解以及

實際中的應用。

答:WS主要是可利用HTTP,穿透防火牆。而Remoting能夠利用TCP/IP,二進制傳

送提升效率。

27.公司要求開發一個繼承System.Windows.Forms.ListView類的組件,要求達到

如下的特殊功能:點擊ListView各列列頭時,能按照點擊列的每行值進行重排視

圖中的全部行 (排序的方式如DataGrid類似)。根據您的知識,請簡要談一下您的

思路

答:根據點擊的列頭,包該列的ID取出,按照該ID排序後,在給綁定到ListView中。

 

28.寫出一條Sql語句:取出表A中第31到第40記錄(SQLServer,以自動增加的ID

做爲主鍵,注意:ID可能不是連續的。

答:解1: select top 10 * from A where id not in (select top 30 id

fromA)

解2: select top 10 * from A where id >(select max(id) from (select

top30 id from A )as A)

29.面向對象的語言具備________性、_________性、________性

答:封裝、繼承、多態。

30.能用foreach遍歷訪問的對象須要實現 ________________接口或聲明

________________方法的類型。

答:IEnumerable 、 GetEnumerator。

31.GC是什麼? 爲何要有GC?

答:GC是垃圾收集器。程序員不用擔憂內存管理,由於垃圾收集器會自動進行管

理。要請求垃圾收集,能夠調用下面的方法之一:

System.gc()

Runtime.getRuntime().gc()

32.Strings = new String("xyz");建立了幾個String Object?

答:兩個對象,一個是「xyx」,一個是指向「xyx」的引用對象s。

 

33.啓動一個線程是用run()仍是start()?

答:啓動一個線程是調用start()方法,使線程所表明的虛擬處理機處於可運行狀

態,這意味着它能夠由JVM調度並執行。這並不意味着線程就會當即運行。run()

方法能夠產生必須退出的標誌來中止一個線程。

34.接口是否可繼承接口? 抽象類是否可實現(implements)接口? 抽象類是否可

繼承實體類(concrete class)?

答:接口能夠繼承接口。抽象類能夠實現(implements)接口,抽象類是否可繼承

實體類,但前提是實體類必須有明確的構造函數。

35.構造器Constructor是否可被override?

答:構造器Constructor不能被繼承,所以不能重寫Overriding,但能夠被重載

Overloading。

36.是否能夠繼承String類?

答:String類是final類故不能夠繼承。

37.try{}裏有一個return語句,那麼緊跟在這個try後的finally {}裏的code會

不會被執行,何時被執行,在return前仍是後?

答:會執行,在return前執行。

38.兩個對象值相同(x.equals(y)== true),但卻可有不一樣的hash code,這句

話對不對?

答:不對,有相同的hash code。

39.swtich是否能做用在byte上,是否能做用在long上,是否能做用在String上?

答:switch(expr1)中,expr1是一個整數表達式。所以傳遞給 switch 和

case語句的參數應該是 int、 short、 char 或者 byte。long,string 都不

能做用於swtich。

40.當一個線程進入一個對象的一個synchronized方法後,其它線程是否可進入此

對象的其它方法?

不能,一個對象的一個synchronized方法只能由一個線程訪問。

41.abstract的method是否可同時是static,是否可同時是native,是否可同時是

synchronized?

答:都不能。

42.List,Set, Map是否繼承自Collection接口?

答:List,Set是Map不是

43.Set裏的元素是不能重複的,那麼用什麼方法來區分重複與否呢? 是用==仍是

equals()?它們有何區別?

答:Set裏的元素是不能重複的,那麼用iterator()方法來區分重複與否。

equals()是判讀兩個Set是否相等。

equals()和==方法決定引用值是否指向同一對象equals()在類中被覆蓋,爲的是

當兩個分離的對象的內容和類型相配的話,返回真值。

44.數組有沒有length()這個方法? String有沒有length()這個方法?

答:數組沒有length()這個方法,有length的屬性。String有有length()這個方

法。

45.sleep()和 wait() 有什麼區別?

答:sleep()方法是使線程中止一段時間的方法。在sleep 時間間隔期滿後,線程

不必定當即恢復執行。這是由於在那個時刻,其它線程可能正在運行並且沒有被

調度爲放棄執行,除非(a)「醒來」的線程具備更高的優先級

(b)正在運行的線程由於其它緣由而阻塞。

wait()是線程交互時,若是線程對一個同步對象x 發出一個wait()調用,該線程

會暫停執行,被調對象進入等待狀態,直到被喚醒或等待時間到。

46.shorts1 = 1; s1 = s1 + 1;有什麼錯? short s1 = 1; s1 += 1;有什麼錯

?

答:short s1 = 1; s1 = s1 + 1;有錯,s1是short型,s1+1是int型,不能顯式

轉化爲short型。可修改成s1 =(short)(s1 + 1) 。short s1 = 1; s1 += 1正

確。

47.談談final,finally, finalize的區別。

答:

final—修飾符(關鍵字)若是一個類被聲明爲final,意味着它不能再派生出新

的子類,不能做爲父類被繼承。所以一個類不能既被聲明爲 abstract的,又被

聲明爲final的。將變量或方法聲明爲final,能夠保證它們在使用中 不被改變。

被聲明爲final的變量必須在聲明時給定初值,而在之後的引用中只能讀取,不可

修改。被聲明爲 final的方法也一樣只能使用,不能重載

finally—再異常處理時提供finally 塊來執行任何清除操做。若是拋出一個異

常,那麼相匹配的 catch 子句就會 執行,而後控制就會進入 finally 塊(若是

有的話)。

finalize—方法名。Java 技術容許使用 finalize() 方法在垃圾收集器將對象

從內存中清除出去以前作必要的清理工做。這個方法是由垃圾收集器在肯定這個

對象沒有被引用時對這個對象調用的。它是在 Object 類中定義的 ,所以全部的

類都繼承了它。子類覆蓋 finalize() 方法以整理系統資源或者執行其餘清理工

做。finalize() 方法是在垃圾收集器刪除對象以前對這個對象調用的。

48.如何處理幾十萬條併發數據?

答:用存儲過程或事務。取得最大標識的時候同時更新..注意主鍵不是自增量方

式這種方法併發的時候是不會有重複主鍵的..取得最大標識要有一個存儲過程來

獲取.

49.Session有什麼重大BUG,微軟提出了什麼方法加以解決?

答:是iis中因爲有進程回收機制,系統繁忙的話Session會丟失,能夠用Sate

server或SQL Server數據庫的方式存儲Session不過這種方式比較慢,並且沒法

捕獲Session的END事件。

50.進程和線程的區別?

答:進程是系統進行資源分配和調度的單位;線程是CPU調度和分派的單位,一個

進程能夠有多個線程,這些線程共享這個進程的資源。

51.堆和棧的區別?

答: 棧:由編譯器自動分配、釋放。在函數體中定義的變量一般在棧上。

堆:通常由程序員分配釋放。用new、malloc等分配內存函數分配獲得的就是在堆

上。

52.成員變量和成員函數前加static的做用?

答:它們被稱爲常成員變量和常成員函數,又稱爲類成員變量和類成員函數。分

別用來反映類的狀態。好比類成員變量能夠用來統計類實例的數量,類成員函數

負責這種統計的動做。

53.ASP。NET與ASP相比,主要有哪些進步?

答:asp解釋形,aspx編譯型,性能提升,有利於保護源碼。

54.請說明在.net中經常使用的幾種頁面間傳遞參數的方法,並說出他們的優缺點。

答:session(viewstate) 簡單,但易丟失

application全局

cookie簡單,但可能不支持,可能被僞造

inputttype="hidden" 簡單,可能被僞造

url參數 簡單,顯示於地址欄,長度有限

數據庫 穩定,安全,但性能相對弱

55.請指出GAC的含義?

答:全局程序集緩存。

56.向服務器發送請求有幾種方式?

答:get,post。get通常爲連接方式,post通常爲按鈕方式。

57.DataReader與Dataset有什麼區別?

答:一個是隻能向前的只讀遊標,一個是內存中虛擬的數據庫。

58.軟件開發過程通常有幾個階段?每一個階段的做用?

答:需求分析,架構設計,代碼編寫,QA,部署

59.在c#中using和new這兩個關鍵字有什麼意義,請寫出你所知道的意義?using

指令 和語句 new 建立實例 new 隱藏基類中方法。

答:using 引入名稱空間或者使用非託管資源

new新建實例或者隱藏父類方法

60.須要實現對一個字符串的處理,首先將該字符串首尾的空格去掉,若是字符串中

間還有連續空格的話,僅保留一個空格,即容許字符串中間有多個空格,但連續的空

格數不可超過一個.

答:string inputStr=" xx xx ";

inputStr=Regex.Replace(inputStr.Trim(),"*"," ");

 

61.什麼叫作SQL注入,如何防止?請舉例說明。

答:利用sql關鍵字對網站進行攻擊。過濾關鍵字'等

62.什麼是反射?

答:動態獲取程序集信息

63.用Singleton如何寫設計模式

答:static屬性裏面new ,構造函數private

64.什麼是ApplicationPool?

答:Web應用,相似Thread Pool,提升併發性能。

65.什麼是虛函數?什麼是抽象函數?

答:虛函數:沒有實現的,可由子類繼承並重寫的函數。抽象函數:規定其非虛

子類必須實現的函數,必須被重寫。

66.什麼是XML?

答:XML便可擴展標記語言。eXtensible Markup Language.標記是指計算機所能

理解的信息符號,經過此種標記,計算機之間能夠處理包含各類信息的文章等。

如何定義這些標記,便可以選擇國際通用的標記語言,好比HTML,也可使用象

XML這樣由相關人士自由決定的標記語言,這就是語言的可擴展性。XML是從SGML

中簡化修改出來的。它主要用到的有XML、XSL和XPath等。

67.什麼是WebService?UDDI?

答:Web Service即是基於網絡的、分佈式的模塊化組件,它執行特定的任務,遵

守具體的技術規範,這些規範使得Web Service能與其餘兼容的組件進行互操做。

UDDI的目的是爲電子商務創建標準;UDDI是一套基於Web的、分佈式的、爲

WebService提供的、信息註冊中心的實現標準規範,同時也包含一組使企業能將

自身提供的Web Service註冊,以使別的企業可以發現的訪問協議的實現標準。

68.什麼是ASP.net中的用戶控件?

答:用戶控件通常用在內容多爲靜態,或者少量會改變的狀況下..用的比較大..類

似ASP中的include..可是功能要強大的多。

69.列舉一下你所瞭解的XML技術及其應用

答:xml用於配置,用於保存靜態數據類型.接觸XML最多的是web Services..和

config

70.ADO.net中經常使用的對象有哪些?分別描述一下。

答:Connection 數據庫鏈接對象

Command數據庫命令

DataReader數據讀取器

DataSet數據集

71.什麼是code-Behind技術。

答:ASPX,RESX和CS三個後綴的文件,這個就是代碼分離.實現了HTML代碼和服務

器代碼分離.方便代碼編寫和整理.

72.什麼是SOAP,有哪些應用。

答:simple object access protocal,簡單對象接受協議.以xml爲基本編碼結構

,創建在已有通訊協議上(如http,不過聽說ms在搞最底層的架構在tcp/ip上的

soap)的一種規範WebService使用的協議..

73.C#中 property 與 attribute的區別,他們各有什麼用處,這種機制的好處

在哪裏?

答:一個是屬性,用於存取類的字段,一個是特性,用來標識類,方法等的附加

性質

74.XML與 HTML 的主要區別

答:1. XML是區分大小寫字母的,HTML不區分。

2.在HTML中,若是上下文清楚地顯示出段落或者列表鍵在何處結尾,那麼你能夠

省略</p>或者</li>之類的結束標記。在XML中,絕對不能省略掉結束標記。

3.在XML中,擁有單個標記而沒有匹配的結束標記的元素必須用一個 / 字符做爲

結尾。這樣分析器就知道不用查找結束標記了。

4.在XML中,屬性值必須分裝在引號中。在HTML中,引號是可用可不用的。

5.在HTML中,能夠擁有不帶值的屬性名。在XML中,全部的屬性都必須帶有相應

的值。

75.c#中的三元運算符是?

答:?:。

76.當整數a賦值給一個object對象時,整數a將會被?

答:裝箱。

77.類成員有_____種可訪問形式?

答:this.;new Class().Method;

78.publicstatic const int A=1;這段代碼有錯誤麼?是什麼?

答:const不能用static修飾。

79.floatf=-123.567F; int i=(int)f;i的值如今是_____?

答:-123。

80.委託聲明的關鍵字是______?

答:delegate.

81.用sealed修飾的類有什麼特色?

答:密封,不能繼承。

82.在Asp.net中全部的自定義用戶控件都必須繼承自________?

答:Control。

83.在.Net中全部可序列化的類都被標記爲_____?

答:[serializable]

84.在.Net託管代碼中咱們不用擔憂內存漏洞,這是由於有了______?

答:GC。

85.當類T只聲明瞭私有實例構造函數時,則在T的程序文本外部,___能夠___(可

以 or 不能夠)從T派生出新的類,不能夠____(能夠 or 不能夠)直接建立T的

任何實例。

答:不能夠,不能夠。

86.下面這段代碼有錯誤麼?

switch(i){

case():答://case()條件不能爲空

CaseZero();

break;

case1:

CaseOne();

break;

case2:

dufault;答://wrong,格式不正確

CaseTwo();

break;

}

87.在.Net中,類System.Web.UI.Page 能夠被繼承麼?

答:能夠。

88..net的錯誤處理機制是什麼?

答:.net錯誤處理機制採用try->catch->finally結構,發生錯誤時,層層上拋

,直到找到匹配的Catch爲止。

89.利用operator聲明且僅聲明瞭==,有什麼錯誤麼?

答:要同時修改Equale和GetHash() ? 重載了"==" 就必須重載 "!="

90.在.net(C# or vb.net)中如何取消一個窗體的關閉。

答:private void Form1_Closing(object sender,

System.ComponentModel.CancelEventArgse)

{

e.Cancel=true;

}

91.在.net(C# or vb.net)中,Appplication.Exit 仍是 Form.Close有什麼

不一樣?

答:一個是退出整個應用程序,一個是關閉其中一個form。

 

92.某一密碼僅使用K、L、M、N、O共5個字母,密碼中的單詞從左向右排列,密

碼單詞必須遵循以下規則:

(1) 密碼單詞的最小長度是兩個字母,能夠相同,也能夠不一樣

(2) K不多是單詞的第一個字母

(3) 若是L出現,則出現次數不止一次

(4) M不能使最後一個也不能是倒數第二個字母

(5) K出現,則N就必定出現

(6) O若是是最後一個字母,則L必定出現

問題一:下列哪個字母能夠放在LO中的O後面,造成一個3個字母的密碼單詞?

A)K B)L C) M D) N

答案:B

問題二:若是能獲得的字母是K、L、M,那麼可以造成的兩個字母長的密碼單詞的

總數是多少?

A)1個 B)3個 C)6個 D)9個

答案:A

問題三:下列哪個是單詞密碼?

A)KLLN B) LOML C) MLLO D)NMKO

答案:C

93.62-63=1 等式不成立,請移動一個數字(不能夠移動減號和等於號),使得等

式成立,如何移動?

答案:62移動成2的6次方

94.C#中 property 與 attribute的區別,他們各有什麼用處,這種機制的好

處在哪裏?

答:attribute:自定義屬性的基類;property :類中的屬性

95.在C#中,string str = null 與 string str = "" 請儘可能使用文字或圖

象說明其中的區別。

答:null是沒有空間引用的;

"" 是空間爲0的字符串;

96.abstract class和interface有什麼區別?

答:聲明方法的存在而不去實現它的類被叫作抽像類(abstract class),它用於要建立一個體現某些基本行爲的類,併爲該類聲明方法,但不能在該類中實現該類的狀況。不能建立abstract 類的實例。然而能夠建立一個變量,其類型是一個抽像類,並讓它指向具體子類的一個實例。不能有抽像構造函數或抽像靜態方法。Abstract 類的子類爲它們父類中的全部抽像方法提供實現,不然它們也是抽像類爲。取而代之,在子類中實現該方法。知道其行爲的其它類能夠在類中實現這些方法。

接口(interface)是抽像類的變體。在接口中,全部方法都是抽像的。多繼承性可經過實現這樣的接口而得到。接口中的全部方法都是抽像的,沒有一個有程序體。接口只能夠定義static final成員變量。接口的實現與子類類似,除了該實現類不能從接口定義中繼承行爲。當類實現特殊接口時,它定義(即將程序體給予)全部這種接口的方法。而後,它能夠在實現了該接口的類的任何對像上調用接口的方法。因爲有抽像類,它容許使用接口名做爲引用變量的類型。一般的動態聯編將生效。引用能夠轉換到接口類型或從接口類型轉換,instanceof 運算符能夠用來決定某對象的類是否實現了接口。

97.<%# %> 和 <% %> 有什麼區別?

答:<%# %>表示綁定的數據源

<%%>是服務器端代碼塊

98.重載與覆蓋的區別?

答:一、方法的覆蓋是子類和父類之間的關係,是垂直關係;方法的重載是同一個

類中方法之間的關係,是水平關係

二、覆蓋只能由一個方法,或只能由一對方法產生關係;方法的重載是多個方法之

間的關係。

99.Overloaded的方法是否能夠改變返回值的類型?

答:Overloaded的方法是能夠改變返回值的類型。

100.C#能否對內存進行直接的操做?

答:在.net下,.net引用了垃圾回收(GC)功能,它替代了程序員不過在C#中。

 

 

SQL Server 數據庫基礎筆記分享(下)

 

前言

本文是我的學習SQL Server 數據庫時的以往筆記的整理,內容主要是對數據庫的基本增刪改查的SQL語句操做約束,視圖,存儲過程,觸發器的基本瞭解。

注:內容比較基礎,適合入門者對SQL Server 數據庫的瞭解!!!

正文

1.子查詢

--把一個查詢結果做爲另一個查詢的查詢源 
select * from (select * from Student where tbage between 3 and 5)
as ct where tbname=5 --ct是新創的表名

--把另一個查詢的結果做爲當前查詢的條件來使用。
--子查詢中=、!= 、< 、> 、<= 、>=以後只能返回單個值,若是多個值就會報錯
--解決辦法 能夠用in 代替
select * from Student
where tbage in(select tbage from Student where tbname=3)

select * from Student
where tbage=(select tbage from Student where tbname=3)

》》》》》》子查詢分頁《《《《《《

--1》顯示第一頁的數據
--分頁查詢的時候首先是將數據排序
select * from Student order by id desc

--2》第一頁 顯示5條數據
select Top 5 * from Student order by id desc

--3》第二頁
select top 5 * from Student 
where id not in (select top 5 * from Student order by id desc)
order by id desc

--4》第三頁
select top 5 * from Student 
where id not in (select top (2*5) * from Student order by id desc)
order by id desc

》》》開窗函數分頁《《《

--第七頁數據 每頁5條數據
--over屬於開窗函數

select * from
(
select * ,row_number() over( order by id desc) as paixu from Student
) as tbl
where tbl.paixu between 6*5+1 and 7*5

2.連表查詢

--查詢全部學生的姓名、年齡及所在班級 (班級在另外一個表中)
--當多個列在不一樣的表中時,要跨表查詢,因此通常可使用inner join
--tc ts是對錶名起的別名
select
ts.tsname,
ts.tsage,
tc.tclassname
from TblStudent as ts
inner join TblClass as tc on ts.tsclassid=tc.tclassid(只查詢兩個表中都有的數據)

--》》》full join 是查詢全部的數據(沒有的爲空)

---子查詢寫法
select
tsname,
tsage,
班級名稱=(select tclassname from TblClass where TblClass.tclassid=TblStudent.tsclassid)
from TblStudent

--查詢學生姓名、年齡、班級及成績(成績屬於第三張表)
select 
ts.tsname,
ts.tsage,
tc.tclasssname,
tscore.tenglish,
tscore.tmath
from TblStudent as ts
inner join TblClass as tc on ts.tsclassid=tc.tclassid 
inner join TblScore as tscore on tscore.tsid=ts.tsid

 

--》》》左外聯接(左聯接)

--查詢沒有參加考試的學生的姓名與編號
--把左表(left join 關鍵字左邊的表)中的所有記錄都顯示出來,對於那些在右表中能找到匹配的記錄,顯示對應匹配數據,對於那些右表中找不到匹配的記錄顯示爲null
select
ts.tsid,
ts.tsname,
TblScore.*
from TblStudent as ts
left outer join TblSore.tsid=ts.tsid   --outer能夠不寫

--》》》右外聯接
--表示要將右表(right join 右邊的表)中的全部數據都顯示,左表中只顯示那些匹配的數據。

select
ts.tsid,
ts.tsname,
TblScore.*
from TblStudent as ts
right outer join TblSore.tsid=ts.tsid

--右外聯與左外聯都是先將匹配的數據找到,而後再將那些沒有匹配的數據添加進來,(注意:不是一塊兒查詢出來的,有前後順序)

--》》》練習:查詢全部學生(參加和未參加的考試)的學生姓名、年齡、成績,若是沒有參加考試顯示缺考,若是小於english或者math 小於60分顯示不及格
select
ts.tsname,
ts.tsage,
tscore.tsid,
case
when tscore.tenglish is null then '缺考'
else convert(varchar(10),tscore.tenglish)
end as 英語成績,
case
when tscore.tmath id null then '缺考'
else convert (varchar(10),tscore.tmath)
end as 數學成績,
是否報考=
case
when tscore.tscoreid is null then '是'
else '否'
end
from TblStudent as ts
left join TblScore as tscore on ts.tsid=tscore.tsid

3.視圖

視圖自己並不存儲數據,只是存儲的查詢語句,若是把真實表中的數據修改後,則經過視圖查詢到的結果也變了。

視圖的目的是方便查詢,因此通常狀況下不能對視圖進行增刪改查


--在視圖中的查詢語句,必須爲每一列建立一個列名
create view vw2
as
select
tsname,
case
when tsage>13 and tsage<=16 then '少年'
when tsage>50 then '老年'
else '青壯年'
end as 稱呼
from TblStudent

--在視圖中不能使用order by語句。除非:另外還指定了top 或for xml
--錯誤
create view vw3
as 
select * from TblStudent order by tsage desc

--正確
create view vw3
as 
select top 3 * from TblStudent order by tsage desc

4.聲明變量與使用

--》》》局部變量
--聲明變量
declare @name varchar(10)
declare @age int

--賦值
set @name='yhz'
set @age=17

--輸出值
print @name
print @age

--使用set與select爲變量賦值的區別
declare @rcount int 
set @rcount=(select count(*) from TblStudent)
print @rcount

declare @rcount int 
select @rcount=count(*) from TblStudent
print @rcount


--》》》全局變量
print @@language
print @@version
print 'aaa'+100
--經過判斷@@error變量中是否不爲0,就能夠判斷上一條sql語句執行是否出錯了
--若是@@error爲0,表示上一條sql語句執行沒出錯,若是@@error不爲0,則表示上一條sql語句出錯了。
print@@error

--》》》經過while計算1-100之間全部奇數的和

--聲明變量並初始化

declare @sum int=0
declare @i int =1
while @i<=100
begin
if @i%2<>0
begin
set @sum=@sum+@i
end
end
print @sum

5.事務

事務有四個屬性:原子性 一致性 隔離性 持久性
原子性:對於數據修改,要麼全都執行,要麼全都不執行
一致性:當數據完成時,數據必須處於一致狀態
隔離性:對數據進行修改的全部併發事務時彼此隔離的。這代表事務必須是獨立的,它不該以任何方式依賴於或影響其餘事務
永久性:事務完成後,他對數據庫的修改被永久保持,事務日誌可以保持事務的永久性

--打開事務
begin transaction

--提交事務
commit transaction

--回滾事務
rollback transaction


--帳戶A給帳戶B轉帳 當一方出問題時,兩個語句都不執行
begin tran
declare @sum int=0
update bank set balance =balance-1000 where cid='0001'
set @sum=@sum+@@error
update banl set balance =balance+1000 where cid='0002'
set @sum=@sum+@@error

if @sum<>0
begin
rollback tran
print '回滾'
end
else 
begin
commit tran
print '提交了'
end

6.存儲過程

--建立一個自定義的存儲過程
create proc usp_HelloWorld
as 
begin
print 'hello world'
end

--輸出存儲過程
exec usp_HelloWorld

--建立一個存儲過程計算兩個數的和
create procedure usp_Add
@num1 int,
@num2 int
as 
begin
print @num1+@num2
end

--輸出值
exec usp_Add 100,230


--存儲過程當中的參數的問題
--存儲過程若是有參數,則調用的時候必須爲參數賦值
exec usp_Add --不傳參數則報錯


--第二個參數若是用戶不傳,則有一個默認值
create procedure usp_Add
@num1 int,
@num2 int 1000 --爲存儲過程的參數設置默認值
as 
begin
print @num1+@num2
end

--建立分頁存儲過程
create proc usp_PageBum
@pageSize int, --每頁顯示的數量
@pageIndex int --第幾頁
as
begin
select * from (select *,row_number()over (order by CityID asc)as num from S_City )as s 
where s.num between (@pageIndex -1)*@pageSize +1 and @pageIndex *@pageSize 
end
--查詢第5頁內容每頁顯示10條數據
exec usp_PageBum 10,5

--刪除一個存儲過程
drop proc usp_Add

7.觸發器

儘可能避免在觸發器中執行耗時操做,由於觸發器會與sql語句認爲在同一個事務中(事務不結束,就沒法釋放鎖)

--建立插入數據觸發器
create trigger tri_Teacher_insert_after
on Teacher after insert
as
begin
declare @id int
declare @name varchar(10)
declare @phone int
declare @mail varchar(50)
select @id=tcid,@name=tcname,@phone=tcphone,@mail=tcmail from inserted

print @id
print @name
print @phone
print @mail
end

--插入數據
insert into Teacher values('網名好','12352536','Wjifdfji@qq.com')


--建立刪除數據觸發器
--不能有主鍵
create trigger tri_Teacher_after
on Teacher after delete
as
begin
insert into TeacherBak
select * from deleted
end

--刪除數據
--sql server中的觸發器是表級觸發器,不管刪除多少行或者插入多少行,只觸發一次
--是按語句來觸發的,每次執行一次語句,觸發一次觸發器
delete from Teacher where tcid>18

8.遊標

--1.定義遊標
declare cur_Student cursor fast_forward for select * from Student

--2.打開遊標
open cur_Student

--2.1 對遊標的操做
--將每條數據讀取並輸出

--2.1.1將遊標向後移動一條
fetch next from cur_Student

--將遊標循環向後移動,直到末尾
while @@fetch_status=0
begin
fetch next from cur_Student
end


--3.關閉遊標
close cur_Student

--4.釋放資源
deallocate cur_Student

9.(補充)全局臨時表,局部臨時表

局部臨時表:表名以#爲開頭。只在當前會話中有效,不能跨鏈接訪問。若是直接在鏈接會話中建立,則當前鏈接斷開後刪除,若是是在存儲過程當中建立的,則存儲過程執行完畢後刪除

全局臨時表:表名以##爲開頭。多個會話可共享全局臨時表。當建立全局臨時表的會話斷開,而且沒有用戶正在訪問全局臨時表時刪除

10.(補充)約束

--刪除一列(EmpAddress列)
alter table Class drop column EmpAddress

--增長一列(增長一列EmpAddr varchar(1000))
alter table Class Add EmpAddr varchar(1000)

--修改一下Emp 的數據類型(varchar(200))
alter table Class alter column Emp varchar(200)

--爲EmpId增長一個主鍵約束
alter table Class add constraint PK_Class_EmpId primary key(EmpId)

--爲EmpName增長一個惟一約束
alter table Class add constraint UQ_Class_EmpName unique(EmpName)

--爲性別增長一個默認約束,默認爲男
alter table Class add constraint DF_Class_EmpGender default('男') for EmpGender

--爲年齡增長一個檢查約束,年齡必須在1—120歲之間(包含)
alter table Class add constraint CK_Class_EmpAge check(EmpAge>=0 and EmpAge<=120)

--增長外鍵約束,表Class中有一列EmpDeptId引用Student表中的DeptId
alter table Class add EmpDeptId int not null
alter table Student add constraint PK_Student_DeptId primary key(DeptId)

alter table Class add constraint FK_Class_Student foreign key(EmpDeptId)
references Student(DeptId)


--一條語句刪除多個約束,約束名用 逗號 隔開
alter table Class drop constraint
PK_Student_DeptId,
FK_Class_Student,
CK_Class_EmpAge


--用一條語句爲表增長多個約束
alter table Class add 
constraint PK_Student_DeptId primary key(DeptId),
constraint CK_Class_EmpAge check(EmpAge>=0 and EmpAge<=120),
add constraint DF_Class_EmpGender default('男') for EmpGender

後記

筆記不全,還請見諒!但願對你有所提升。

 

SQL Server 數據庫基礎筆記分享(上)

 

前言

本文是我的學習SQL Server 數據庫時的以往筆記的整理,內容主要是對數據庫的基本增刪改查的SQL語句操做約束,視圖,存儲過程,觸發器的基本瞭解。

注:內容比較基礎,適合入門者對SQL Server 數據庫的瞭解!!!

正文

1.主鍵:

主鍵的做用:保證表中的每條數據的惟一性
特色: 主鍵不能重複 不能爲空
分類:
邏輯主鍵:選擇爲表中增長的那些「自動編號」列或者「GUID」列爲主鍵(沒有實際業務上的意義)的主鍵 (建議使用邏輯主鍵)
業務主鍵:選擇表中那些在業務中有實際意義的列做爲主鍵
》》》》》》》》》選擇主鍵的策略,選什麼樣的列做爲主鍵《《《《《《《《《
1》主鍵,建議選擇那些通常不會被修改的列
2》選擇單列,不選擇多列(不用組合主鍵)
3》選擇那些簡單列(整數列(自動編號))

 

2.char(),nchar(),varchar()之間的區別

》》》》》》》》》char(10)與varchar(10)的區別《《《《《《《《《
char(10) 固定長度,表示在數據庫中存儲的時候佔用10個字節的空間,若是超出10個則報錯,若是不夠10個則用空格補全。 
varchar(10) 可變長度,表示該列最多能夠存儲10個字節,若是實際存儲不夠10個字節,則會在存儲的時候自動計算一下實際的存儲個數,而動態的改變長度。【節省空間】

》》》》》》》》》char(10)與nchar(10)的區別《《《《《《《《《

char(10) 能夠存儲10個字母或者5個漢字。 用來存儲數據的時候,英文站1個字節,中文站2個字節。

nchar(10) 表示能夠存儲10個字母或10個漢字,由於每一個字符都是按照unicode方法來存儲的。當使用nchar(10),來存儲數據的時候不管存儲的是中文仍是英文都是每一個字符佔2個。

 

3. 建立數據庫

--建立一個數據庫
create database School

--刪除數據庫
drop database School

--建立數據庫的時候,指定一些數據庫的相關參數。
create database School
on primary --主數據文件
(
name='School',
size=10mb,
filename='c:school.mdf',
filegrowth=10%,
maxsize=100mb
)
log on --日誌文件
(
name='School_log',
filename='c:school.ldf',
size=5mb,
filegrowth=5mb,
maxsize=50mb
)

--切換數據庫
use school
go

4. 建立表

--建立表
create table Class
(
ClassId int identity(1,1) primary key,
ClassName varchar(50) not null,
ClassDesc varchar(50) not null
)

--刪除表
drop table Class

--向Class表中插入數據
insert into Class(ClassName,ClsDesc)values('大三','三年');

--insert into...values.. 這種寫法每次只能插入一條數據

--向Class表中插入多條數據
--重複數據不重複插入,union關鍵字自己就具備去掉重複的意思
--union | union all (重複插入)
insert into Class
select '大三','三年' union
select '三五','間諜' union
select '一一','多久' union
select '六七','獲得'


--將Class表中的數據備份到Student表中
--這種寫法會將Class表中的全部數據插入到Student表中
--前提是Student表不存在,若是這個表存在則報錯。
select * into Student from Class


--向一個已經存在的表中插入數據,數據的來源是Class表
insert into Student(ClassName,ClsDesc) 
select ClassName,ClsDesc from Class

 

--查詢表中數據
select * from Class

5.update 數據

--將全部年齡小於20歲的人的年齡都改爲19(tage是Class表後加屬性)
update Class set tage=19 where tage<20

--將年齡爲19歲的而且性別爲0的人的姓名兩邊★改成☆
update Class set ClassName =replace (tname,'★','☆') where tage=19 and tgender=0

6.刪除數據

delete from Class --刪除全部數據 自動編號沒有恢復到默認值 能夠根據條件來刪除
truncate table Class --從新設置了自動編號 刪除只能一次性都清空,不能根據條件來刪除 清除速度(性能)比delete語句快的多


delete from Class where tage=19 or tage is null --刪除19歲或者空值

》》》》》》》》》刪除重複數據只保留一條(id最小的一條)《《《《《《《《《
》》》》》》》》》刪除表中多餘的重複記錄,重複記錄是根據單個字段(peopleId)來判斷,只留有rowid最小的記錄 《《《《《《《《《
delete from people 
where peopleName in (select peopleName from people group by peopleName having count(peopleName) > 1) 
and peopleId not in (select min(peopleId) from people group by peopleName having count(peopleName)>1)

 

7.條件查詢,模糊查詢

--查詢數學沒有及格的學生的學號
select 
fid as 學號,
fmath as 分數
from MyStudent where fmath<60

--查詢年齡在20-30歲之間的男學生
select
fname as 姓名 from MyStudent where fage between 20 and 30 and fgender='男'

--查詢班級id 1 2 3 的全部學生
select * from MyStudent where classid in (1,2,3)

--查詢全部姓趙的同窗 (通配符%表示:任意多個任意字符)
select * from MyStudent where fname like '趙%'

--查詢出姓名中只要包含一個‘民’字便可。
select * from MyStudent where fname like '%民%'

--查詢全部姓趙的同窗,而且姓名字數是3個
--通配符 _ :表示任意的單個字符。
select * from MyStudent where fname like '趙__'
select * from MyStudent where fname like '趙%' and len(fname)=3

--查詢出姓名中包含‘民’或‘用’的同窗
--通配符[]:表示中括號中的任意個字符,只選一個匹配
--通配符 ^a :表示除了a這個字符都行。
select * from MyStudent where fname like '%[民用]%'

8.聚合函數

--查詢數學成績最高低分
select max(fMath) as 數學成績最高分 from MyStudent
select min(fMath) as 數學成績最低分 from MyStudent

--平均分(計算平均分的時候對空值不處理)
select avg(fMath) as 平均分 from MyStudent

--求數據記錄中的總條數(總人數)
select count(*) as 班級總人數 from MyStudent


select
最高分=(select max(fMath) as 數學成績最高分 from MyStudent),
最低分=(select min(fMath) as 數學成績最低分 from MyStudent),
平均分=(select avg(fMath) as 平均分 from MyStudent)


--分數評級
--90以上 優秀
--80以上 良好
--70以上 中
--70如下 差
select chengji,
評級=
case
when shuxue>=90 then '優秀'
when shuxue>=80 then '良好'
when shuxue>=70 then '中'
else '差'
end
from Student

9.null 問題

--請查詢出學生表中全部數學成績爲null的人的信息
--null在數據庫中表示unknow(不知道),判斷一個值是否爲null,也就不能用=或者<>來判斷
select * from MyStudent where fMath=null 錯誤(不返回任何數據)

正確 select * from MyStudent where fMath is null

--查詢全部fmath爲非null的值
select * from MyStudent where fMath is not null


--null值與任何數據運算後獲得的仍是null值。
update MyStudent set fage=fage+1 where fid=1

10.分組group by

--統計出mystudent表中,男女同窗的個數

select 
fgender as 性別, --這時,count(*)統計的是每一組的記錄條數, 不是總條數
count(*) as 人數
from MyStudent group by fgender --先執行group by語句分組,分完組在統計每 組個數。 分出來幾個組,那麼count(*)就統 計幾回


--查詢班級的男同窗的人數大於2的信息

--having是group by的條件對分組後的數據進行篩選(與where相似,都是篩選,只不過having是用來篩選分組後的組的)
select 
classid as 班級號,
count(*) as 班級人數
from TblStudent
where fgender='男'
group by classid
having count(*)>2

》》》》》》》》》語句執行順序《《《《《《《《《

select
--distinct / top 之類的關鍵字
fgender as 性別, --5》選擇列
count(*) as 人數
from MyStudent --1》先從表中拿到數據
where fage>30 --2》從MyStudent的數據中篩選出全部年齡大於30歲的任的信息
group by fgender --3》按照性別分組,分完組獲得一個新的結果集
having count(*)>500 --4》基於分組之後的結果集,而後再篩選,篩選出那些組中記錄大於500的組
order by 人數 asc --6》最後把顯示出來的結果排序


--語句執行順序
from > where > group by > having > select > order by

11.日期函數

--請查詢出全部入職一年以上的員工信息
select * from TblStudent
where dateadd(year,1,tsday)<getdate()


--計算兩個時間差
--查詢90年距今是多少年
select datediff(year,'1990-9-9',getdate())


--查詢一個日期的特定部分
select year(getdate())
select datepart(year,getdate())

--輸出全部數據中通話時間最長的5條記錄。
select top 5 *,'通話時長(秒)'=datediff(second,Startdatetime,Enddatetime) from Calltecords order by datediff(second,Stardatetime,enddatetime) desc

後記

下篇分享視圖、觸發器等,分頁查詢、子查詢、連表查詢等

 

Asp.Net MVC4中的全局過濾器

 

能夠對整個項目進行全局監控。

              新建一個MVC4項目,能夠在global.asax文件中看到以下代碼:  FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);

              表示註冊全局過濾器. 

               GlobalFilters是全局過濾器的集合,能夠經過add方法添加過濾器,默認狀況下,HandleErrorAttribute過濾器被添加到集合中。

                接下來咱們建立一個自定義過濾器,而後添加到全局過濾器集合中。

               1.建立自定義過濾器

                  建立自定義過濾器要繼承ActionFilterAttribute類。咱們建立一個名稱爲CustomerFilterAttribute的過濾器,在action裏面記錄下時間。

                    代碼以下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
public  class  CustomerFilterAttribute : ActionFilterAttribute 
  
   
       public  override  void  OnActionExecuting(ActionExecutingContext filterContext) 
      
           base .OnActionExecuting(filterContext); 
           filterContext.HttpContext.Response.Write( "開始時間:" +DateTime.Now.ToString()+ "<br/>" ); 
      
   
       public  override  void  OnActionExecuted(ActionExecutedContext filterContext) 
      
           base .OnActionExecuted(filterContext); 
           var  controllerName = filterContext.RouteData.Values[ "controller" ].ToString(); 
           var  actionName = filterContext.RouteData.Values[ "action" ].ToString(); 
   
           filterContext.HttpContext.Response.Write( "結束時間:"  + DateTime.Now.ToString() +  "<br/>" ); 
           filterContext.HttpContext.Response.Write( "controller:"  +controllerName+ ",action:" +actionName); 
      
  

  

       2.註冊全局過濾器

                     過濾器建立完成後,咱們把這個過濾器添加到全局過濾器中,使用  filters.Add(new CustomerFilterAttribute());方法,

                       代碼以下:

1
2
3
4
5
6
7
8
public  class  FilterConfig 
   
        public  static  void  RegisterGlobalFilters(GlobalFilterCollection filters) 
       
            filters.Add( new  HandleErrorAttribute()); 
            filters.Add( new  CustomerFilterAttribute()); 
       
   

    接下來咱們運行項目中的每個頁面,都會看到頁面中輸出時間和controller名稱,效果圖以下:

                                

 

 

C#語法——泛型的多種應用

 

本篇文章主要介紹泛型的應用。

泛型是.NET Framework 2.0 版類庫就已經提供的語法,主要用於提升代碼的可重用性、類型安全性和效率。

泛型的定義

下面定義了一個普通類和一個泛型類,咱們能夠明確看到泛型類和普通類最大的區別就是多了一個<T>。

因此,這個<T>就標記了,這個類是泛型類。其中這個T,也能夠寫成A,B,C,D或其餘字符。

1
2
3
4
public  class  Generic
{
     public  String Name;
}
1
2
3
4
public  class  Generic<T>
{
     public  T Name;
}

泛型,顧名思義,就是泛指的類型。比如男人,女人,白人,黑人,能夠泛稱爲【人】。

但類型只能是一個類型。 那麼泛型和類型之間是什麼關係呢?

其實很簡單,泛型在定義的時候,是泛指類型;在使用的時候,就須要被指定,到底使用哪一個類型。

即,使用時,就不在是泛指類型,而是特定類型。

比如,定義時,定義了一我的。但在使用時,必須明確指定,究竟是黑人仍是白人。

泛型的使用

泛型類跟普通類的使用方式同樣,都須要實例化對象,再由對象來調用內部的屬性或方法。

下面代碼實例化了泛型Generic,實例化時,還指定了該泛型Generic的指定類型爲String。

因此要給泛型Generic的屬性Name賦值,就須要賦值字符串類型的值。

1
2
3
4
5
public  static  void  Excute()
{
     Generic<String> gs =  new  Generic<String>();
     gs.Name =  "Kiba518" ;
}

下面代碼定義了一個Int類型的泛型Generic。

1
2
3
4
5
public  static  void  Excute()
{
     Generic< int > gs =  new  Generic< int >();
     gs.Name = 518;
}

泛型的默認值

泛型的默認值,以下面代碼所示。須要使用default(T)來賦值。

無論泛型究竟是String,int,bool或者是一個Class類型,均可以被自動賦值。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public  static  void  Excute()
{
     Generic< int > gs =  new  Generic< int >();
     gs.Name = 518;
     Generic<Task> gsTask =  new  Generic<Task>();
     gsTask.Name =  new  Task(()=> {
         Console.WriteLine( "Kiba518" );
     });
}
 
public  class  Generic<T>
{
     public  T Name =  default (T);
}

泛型的約束

在泛型類中,有個特別的約束可供咱們使用。

當咱們不顯示的聲明時,這個約束不存在。但當咱們顯示的聲明的時候,這個約束就會執行。

下面,咱們來看看這個特別的約束。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
public  static  void  Excute()
{
     Generic<FanXing> gFanXing =  new  Generic<FanXing>();
     Generic<Base> gFanXingBase =  new  Generic<Base>();
     //Generic<string> gs = new Generic<string>(); 這樣定義會報錯
}
public  class  Generic<T>  where  T : Base
{
     public  T Name =  default (T);
}
public  class  Base 
{
     public  string  Name {  get set ; }
}
public  class  FanXing : Base
{
     public  new  string  Name {  get set ; }
}

如上面代碼所示,【where T : Base】就是這個特別的約束。

當顯示聲明這個約束的時候,定義會限制泛型的類型。

什麼是限制泛型的類型呢?

很簡單,泛型T,是泛指某一個類型。咱們在定義泛型類時,還需顯示的指定類型,此時咱們顯示指定的類型,要受這個限制。

這個限制就是指【where T : Base】。

它的限制是,要求咱們指定的類型T必須是Base,或者該類型繼承自Base,如FanXing類。

泛型的函數

在C#中,泛型不只能夠用於類,還能夠直接用於函數。

具體使用方式以下:

1
2
3
4
5
6
7
8
9
10
11
12
public  static  void  Excute()
{
     GenericFunc gf =  new  GenericFunc();
     gf.FanXingFunc<FanXing>( new  FanXing() { Name= "Kiba518" });
}
public  class  GenericFunc
{
     public  void  FanXingFunc<T>(T obj)
     {
         Console.WriteLine(obj.GetType());
     }
}

很簡單,調用泛型函數的時候,指定泛型函數的[指定類型]便可。

可是,這裏咱們發現一個問題,那就是,在泛型函數裏,使用泛型對象的時候,咱們發現對象都是object類型的。

那咱們若是想使用泛型對象裏的屬性和方法時,要怎麼辦呢?

也很簡單,反射就能夠了。

下面咱們添加一個反射函數GetPropertyValue,專門用來獲取屬性。

1
2
3
4
5
6
7
8
9
10
11
12
13
public  class  GenericFunc
{
     public  void  FanXingFunc<T>(T obj)
     {
         var  name = GetPropertyValue(obj,  "Name" );
         Console.WriteLine(name);
     }
     public  object  GetPropertyValue( object  obj,  string  name)
     {
         object  drv1 = obj.GetType().GetProperty(name).GetValue(obj,  null );
         return  drv1;
     }
}

輸出結果以下:

這樣咱們就獲得了咱們想要的結果,若是想使用泛型類裏的函數,道理也同樣,只須要用反射來調用便可。

結語

看到這裏,有些同窗可能會以爲泛型很複雜,連使用其對象下的屬性,都得反射,太繁瑣了,還不如不用呢。

有這樣想法的同窗,內心想一想就行了,若是對老司機這麼說,他確定會心裏默默的微笑,而後對你說,你想的沒錯。

而後,你就沒有而後了。

泛型的應用,開篇已經說了,主要用在提升代碼的可重用性、類型安全性和效率上。

若是隻是定義一個類,調用一個屬性,那泛型的存在就是雞肋。

但事實上,咱們的系統永遠只有更復雜,更復雜,更復雜。所以泛型纔有了用武之地。

C#語法——委託,架構的血液

C#語法——元組類型

C#語法——await與async的正確打開方式

相關文章
相關標籤/搜索