亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 學院 > 開發設計 > 正文

c#中struct和class的區別

2019-11-14 16:32:51
字體:
來源:轉載
供稿:網友

簡單來說,struct是值類型,創建一個struct類型的實例被分配在棧上。class是引用類型,創建一個class類型實例被分配在托管堆上。但struct和class的區別遠不止這么簡單。本篇主要包括:

 

■ 概括來講
■ 從賦值的角度體驗struct和class的不同
■ 從參數傳值角度體驗struct和class的不同
■ 從struct類型的struct類型屬性和struct引用類型屬性體驗struct和class的不同
■ 從構造函數體驗struct和class的不同
■ 從給類型成員賦初值體驗struct和class的不同
■ 何時使用struct,何時使用class?

 

  概括來講

概括來講,struct和class的不同體現在:

 

● 類是引用類型,struct是值類型
● 在托管堆上創建類的實例,在棧上創建struct實例
● 類實例的賦值,賦的是引用地址,struct實例的賦值,賦的是值
● 類作為參數類型傳遞,傳遞的是引用地址,struct作為參數類型傳遞,傳遞的是值
● 類沒有默認無參構造函數,struct有默認無參構造函數
● 類支持繼承,struct不支持繼承
● 類偏向于"面向對象",用于復雜、大型數據,struct偏向于"簡單值",比如小于16字節,結構簡單
● 類的成員很容易賦初值,很難給struct類型成員賦初值
● 類的實例只能通過new SomeClass()來創建,struct類型的實例既可以通過new SomeStruct()來創建,也可以通過SomeStruct myStruct;來創建

 

  從賦值的角度體驗struct和class的不同

  引用類型賦值,是把地址賦值給了變量

class PRogram    {        static void Main(string[] args)        {            SizeClass sizeClass = new SizeClass(){Width = 10, Length = 10};            Console.WriteLine("賦值前:width={0},length={1}", sizeClass.Width, sizeClass.Length);            var copyOfSizeClass = sizeClass;            copyOfSizeClass.Length = 5;            copyOfSizeClass.Width = 5;            Console.WriteLine("賦值后:width={0},length={1}",sizeClass.Width, sizeClass.Length);            Console.ReadKey();        }    }    public class SizeClass    {        public int Width { get; set; }        public int Length { get; set; }    }    public struct SizeStruct    {        public int Width { get; set; }        public int Length { get; set; }    }

1

以上,當把sizeClass賦值給copyOfSize變量的時候,是把sizeClass所指向的地址賦值給了copyOfSize變量,2個變量同時指向同一個地址。所以,當改變copyOfSizeClass變量的值,也相當于改變了sizeClass的值。

 

  struct類型賦值,是完全拷貝,在棧上多了一個完全一樣的變量

class Program    {        static void Main(string[] args)        {            SizeStruct sizeStruct = new SizeStruct(){Length = 10, Width = 10};            Console.WriteLine("賦值前:width={0},length={1}", sizeStruct.Width, sizeStruct.Length);            var copyOfSizeStruct = sizeStruct;            copyOfSizeStruct.Length = 5;            copyOfSizeStruct.Width = 5;            Console.WriteLine("賦值后:width={0},length={1}", sizeStruct.Width, sizeStruct.Length);            Console.ReadKey();        }    }

2
以上,當把sizeStruct賦值給copyOfSizeStruct變量的時候,是完全拷貝,改變copyOfSizeStruct的值不會影響到sizeStruct。

 

  從參數傳值角度體驗struct和class的不同

  引用類型參數傳遞的是地址

class Program    {        static void Main(string[] args)        {            List<string> temp = new List<string>(){"my","god"};            temp.ForEach(t => Console.Write(t + " "));            Console.ReadKey();        }        public static void ChangeReferenceType(List<string> list)        {            list = new List<string>(){"hello", "world"};        }    }

結果顯示:my god


為什么不是hello world?
→棧上的temp指向托管堆上的一個集合實例
→當temp放到ChangeReferenceType(temp)方法中,本質是把temp指向的地址賦值給了變量list
→在ChangeReferenceType(List<string> list)方法內部,又把變量list的指向了另外一個集合實例地址
→但temp的指向地址一直沒有改變


我們再來改變ChangeReferenceType(List<string> list)內部實現方式,其它不變。

class Program    {        static void Main(string[] args)        {            List<string> temp = new List<string>(){"my","god"};                        ChangeReferenceType(temp);            temp.ForEach(t => Console.Write(t + " "));            Console.ReadKey();        }        public static void ChangeReferenceType(List<string> list)        {            list.Clear();            list.Add("hello");            list.Add("world");        }    }

結果顯示:hello world


為什么不是my god? 
→棧上的temp指向托管堆上的一個集合實例
→當temp放到ChangeReferenceType(temp)方法中,本質是把temp指向的地址賦值給了變量list
→在ChangeReferenceType(List<string> list)方法內部,把temp和list共同指向的實例清空,又添加"hello"和"world"2個元素
→由于list和temp指向的實例是一樣的,所以改變list指向的實例就等同于改變temp指向的實例

以上,很好地說明了:引用類型參數傳遞的是地址。

 

  值類型struct參數傳遞的是值

class Program    {        static void Main(string[] args)        {            Size s = new Size(){Length = 10, Width = 10};            ChangeStructType(s);            Console.Write("Length={0},Width={1}", s.Length,s.Width);            Console.ReadKey();        }        public static void ChangeStructType(Size size)        {            size.Length = 0;            size.Width = 0;        }    }    public struct Size    {        public int Length { get; set; }        public int Width { get; set; }    }

3

 

為什么Length和Width不是0呢?
→在棧上變量size
→當通過ChangeStructType(size),把s變量賦值給ChangeStructType(Size size)中的size變量,其本質是在棧上又創建了一個變量size,size的值和s是完全一樣的
→在ChangeStructType(Size size)內部改變size的值,與變量s毫無關系

 

  從struct類型的struct類型屬性和struct引用類型屬性體驗struct和class的不同

  假設有一個struct,它有struct類型的屬性

以下, struct類型Room有struct類型的屬性TableSize和TvSize,我們如何通過Room實例來修改其struct類型的屬性值呢?

class Program    {        static void Main(string[] args)        {            Room r = new Room()            {                TableSize = new Size(){Length = 100, Width = 80},                TvSize = new Size(){Length = 10, Width = 8}            };            r.TableSize.Length = 0;                        Console.WriteLine("table目前的尺寸是:length={0},width={1}", r.TableSize.Length, r.TableSize.Width);            Console.ReadKey();        }    }    public struct Size    {        public int Length { get; set; }        public int Width { get; set; }    }    public struct Room    {        public Size TableSize { get; set; }        public Size TvSize { get; set; }    }

以上,r.TableSize.Length = 0;此處會報錯:不能修改r.TableSize的值,因為不是變量。的確,r.TableSize只是Size的一份拷貝,而且也沒有賦值給其它變量,所以r.TableSize是臨時的,會被自動回收,對其賦值也是沒有意義的。

 

如果要修改r.TableSize,只需把r.TableSize.Length = 0;改成如下:
r.TableSize = new Size(){Length = 0, Width = 0};

 

4

可見,改變struct類型的struct類型屬性的某個屬性是行不通的,因為像以上r.TableSize只是一份拷貝,是臨時的,會被自動回收的。要改變struct類型的struct類型屬性,就需要像上面一樣,給r.TableSize賦上一個完整的Size實例。

 

  假設有一個struct,它有引用類型的屬性呢?

以下,struct類型的Room有引用類型屬性,TableSize和TvSize,如何通過Room實例來修改其引用類型的屬性值呢?并且,我們在類Size中定義了一個事件,當給Size的屬性賦值時就觸發事件,提示size類的屬性值發生了改變。

class Program    {        static void Main(string[] args)        {            var oneSize = new Size() {Length = 10, Width = 10};            var twoSize = oneSize;            oneSize.Changed += (s, e) => Console.Write("Size發生了改變~~");            oneSize.Length = 0;            Console.ReadKey();        }    }    public class Size    {        private int _length;        private int _width;        public event System.EventHandler Changed;        public int Length        {            get { return _length; }            set            {                _length = value;                OnChanged();            }        }        public int Width        {            get { return _width; }            set { _width = value; OnChanged(); }        }        private void OnChanged()        {            if (Changed != null)            {                Changed(this, new EventArgs());            }        }    }    public struct Room    {        public Size TableSize { get; set; }        public Size TvSize { get; set; }    }

運行,顯示:Size發生了改變~~

對oneSize.Length的修改,實際上修改的是oneSize.Length指向托管堆上的實例。

 

  從構造函數體驗struct和class的不同

  struct類型包含隱式的默認無參構造函數

class Program    {        static void Main(string[] args)        {            var size = new SizeStruct();            Console.WriteLine("length={0},width={1}", size.Length, size.Width);            Console.ReadKey();        }    }    public struct SizeStruct    {        public int Length { get; set; }        public int Width { get; set; }    }

5

 

為什么我們沒有給SizeStruct定義無參構造函數,而沒有報錯?
--因為,struct類型有一個隱式的無參構造函數,并且給所有的成員賦上默認值,int類型屬性成員的默認值是0。

 

  類不包含隱式無參構造函數

class Program    {        static void Main(string[] args)        {            var size = new SizeClass();            Console.WriteLine("length={0},width={1}", size.Length, size.Width);            Console.ReadKey();        }    }    public class SizeClass    {        public int Length { get; set; }        public int Width { get; set; }        public SizeClass(int length, int width)        {            Length = length;            Width = Width;        }    }

運行,報錯:SizeClass不包含0個參數的構造函數

 

  從給類型成員賦初值體驗struct和class的不同

如果直接給字段賦初值。

public struct SizeStruct    {        public int _length = 10;    }

運行,報錯:結構中不能有實例字段初始值設定項

 

如果通過構造函數給字段賦初值。

public struct SizeStruct    {        public int _length;        public SizeStruct()        {            _length = 10;        }    }

運行,報錯:結構中不能包含顯式無參數構造函數

 

可見,給struct類型成員賦初值是不太容易的,而給class成員賦初值,no problem。

 

  何時使用struct,何時使用class?

在多數情況下,推薦使用class類,因為無論是類的賦值、作為參數類型傳遞,還是返回類的實例,實際拷貝的是托管堆上引用地址,也就大概4個字節,這非常有助于性能的提升。

 

而作為struct類型,無論是賦值,作為參數類型傳遞,還是返回struct類型實例,是完全拷貝,會占用棧上的空間。根據Microsoft's Value Type Recommendations,在如下情況下,推薦使用struct:

● 小于16個字節
● 偏向于值,是簡單數據,而不是偏向于"面向對象"
● 希望值不可變


 

參考資料:
C# Fundamentals: The Differences Between Struct and Class
Difference between class and struct in C# .Net
Difference Between Class and Structure


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久久人成影片一区二区三区| 日韩av综合中文字幕| 亚洲女人被黑人巨大进入al| 91精品国产成人| 欧美亚洲国产日本| 久久理论片午夜琪琪电影网| 亚洲人av在线影院| 成人h视频在线观看播放| 国产91精品久久久久| 欧美日韩国产综合新一区| 2018中文字幕一区二区三区| 国产一级揄自揄精品视频| 久久91超碰青草是什么| 91情侣偷在线精品国产| 国产精品久久久久7777婷婷| 精品成人69xx.xyz| 国产精品嫩草影院一区二区| 国产深夜精品福利| 久久精品国产精品| 欧美日韩免费网站| 欧美精品一区在线播放| 日韩久久精品成人| 日韩免费在线观看视频| 欧美在线一区二区视频| 国模私拍一区二区三区| 国产精品视频大全| 97超碰蝌蚪网人人做人人爽| 欧美视频中文在线看| 国产69精品久久久久99| 亚洲精品一区av在线播放| 亚洲人成亚洲人成在线观看| 最近中文字幕mv在线一区二区三区四区| 国产精品久久久91| 亚洲色图第三页| 亚洲乱亚洲乱妇无码| 欧美一二三视频| 91久久夜色精品国产网站| 亚洲欧洲av一区二区| 久久精品2019中文字幕| 都市激情亚洲色图| 国产精品视频最多的网站| 国产日韩欧美夫妻视频在线观看| 精品国产欧美一区二区三区成人| 亚洲欧洲偷拍精品| 欧美日韩第一视频| 日韩欧美中文字幕在线观看| 91久久国产精品| 欧美中文在线视频| 91免费看国产| 日本aⅴ大伊香蕉精品视频| 久久久噜噜噜久久中文字免| 亚洲精品一区二区在线| 国产欧美一区二区三区视频| 欧洲精品在线视频| 按摩亚洲人久久| 欧美成人性色生活仑片| 欧美性猛交xxxx免费看| 亚洲美女动态图120秒| 日韩色av导航| 亚洲激情免费观看| 日韩暖暖在线视频| 97香蕉久久超级碰碰高清版| 成人午夜在线影院| 另类少妇人与禽zozz0性伦| 久久精品国产一区二区电影| 成人做爰www免费看视频网站| 欧美日韩免费区域视频在线观看| 欧美日韩国产精品| 国产精品日韩在线一区| 精品一区二区三区三区| 福利二区91精品bt7086| 1769国产精品| 欧美性视频在线| 91麻豆桃色免费看| 国产一区二区三区在线观看视频| 国产视频久久网| 日韩专区在线播放| 国产精品亚发布| 欧美在线欧美在线| 深夜福利91大全| 在线看福利67194| 亚洲国产精品成人精品| 91成人在线视频| 亚洲国产黄色片| 亚洲国产精品女人久久久| 亚洲国产成人久久| 亚洲视频在线观看免费| 欧美电影在线观看网站| 成人福利在线观看| 国产精品午夜一区二区欲梦| 国产成人av在线| 亚洲石原莉奈一区二区在线观看| 欧美国产欧美亚洲国产日韩mv天天看完整| 成人久久精品视频| 欧美孕妇孕交黑巨大网站| 国产精品黄视频| 国内久久久精品| 欧美极品少妇全裸体| 精品日本高清在线播放| 欧美一级视频在线观看| 欧美一级视频一区二区| 亚洲男人第一网站| 欧美猛少妇色xxxxx| 欧美一级淫片aaaaaaa视频| 国产不卡在线观看| 久久久久国产精品一区| 国内精品伊人久久| 欧美xxxx18性欧美| 国产精品自产拍在线观看中文| 精品国内自产拍在线观看| 日韩视频欧美视频| 日韩视频免费观看| 久久激情五月丁香伊人| 成人观看高清在线观看免费| 国产91色在线播放| 成人精品福利视频| 国产区精品在线观看| 久久成人亚洲精品| 国产精品精品久久久| 亲子乱一区二区三区电影| 一本色道久久综合亚洲精品小说| 国语自产精品视频在免费| 亚洲精品美女在线观看| 欧美一区二区三区免费视| 久久久天堂国产精品女人| 国内精品久久久久久| 日韩在线观看视频免费| 欧美激情精品久久久久久变态| 日韩精品在线第一页| 国产精品a久久久久久| 亚洲天堂精品在线| 亚洲欧美成人一区二区在线电影| 欧美成人高清视频| 亚洲精品美女在线观看播放| 成人网欧美在线视频| 亚洲精品国产精品国产自| 97在线看免费观看视频在线观看| 亚洲精品小视频在线观看| 国产成人高潮免费观看精品| 国产精品96久久久久久又黄又硬| 国产成人精品久久| 久久精品一偷一偷国产| 日韩经典一区二区三区| 欧美性极品xxxx娇小| 精品国产美女在线| 久久久在线视频| 久久久www成人免费精品| 2019中文在线观看| 成人性生交大片免费看小说| 亚洲天堂第二页| 岛国av一区二区| 亚洲美女av电影| 亚洲国产精品中文| 青青草国产精品一区二区| 欧美日韩国产成人高清视频| 日韩在线观看视频免费| 永久555www成人免费| 国产精品一区二区久久国产| 欧洲成人性视频| 成人免费视频网址| 欧美日韩中文字幕综合视频| 亚洲天堂网站在线观看视频| 中文字幕亚洲欧美一区二区三区| 日韩成人激情影院|