C#中struct和class的区别详解

本文详细分析了C#中struct和class的区别,对于C#初学者来说是有必要加以了解并掌握的。

简单来说,struct是值类型,创建一个struct类型的实例被分配在栈上。class是引用类型,创建一个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; }
  }

运行结果如下图所示:

以上,当把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();
    }
  }

程序运行结果如下图所示:

以上,当把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; }
  }

运行结果如下图所示:

为什么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};

运行结果如下图所示:

可见,改变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; }
  }

运行结果如下图所示:

为什么我们没有给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个字节
● 偏向于值,是简单数据,而不是偏向于"面向对象"
● 希望值不可变

(0)

相关推荐

  • C语言的Struct Hack笔记

    最近在搞Compiler的CodeGenerator实验,有一部分需要把Java程序翻译成C程序,比如: 复制代码 代码如下: int [] array;array = new int[10];System.out.println(array.length); //10 这段代码翻译成C很自然的想法是: 复制代码 代码如下: int * array; // int array[] not support in Carray = (int*)malloc(sizof(int)*10);printf

  • php读取二进制流(C语言结构体struct数据文件)的深入解析

    尽管php是用C语言开发的,不过令我不解的是php没有提供对结构体struct的直接支持.不过php提供了pack和unpack函数,用来进行二进制数据(binary data)和php内部数据的互转: 复制代码 代码如下: string pack ( string $format [, mixed $args [, mixed $...]] )   //Pack given arguments into binary string according to format.  array unp

  • C#中结构(struct)的部分初始化和完全初始化实例分析

    本文实例分析了C#中结构(struct)的部分初始化和完全初始化,分享给大家供大家参考.具体分析如下: 假设有这样一个值类型struct,如下所示: public struct Size { public int Length; public int Width; public int Area() { return Length*Width; } } 一.客户端,给所有struct字段初始化后调用方法 class Program { static void Main(string[] args

  • Go语言struct类型详解

    struct Go语言中,也和C或者其他语言一样,我们可以声明新的类型,作为其它类型的属性或字段的容器.例如,我们可以创建一个自定义类型person代表一个人的实体.这个实体拥有属性:姓名和年龄.这样的类型我们称之struct.如下代码所示: 复制代码 代码如下: type person struct {     name string     age int } 看到了吗?声明一个struct如此简单,上面的类型包含有两个字段. 1.一个string类型的字段name,用来保存用户名称这个属性

  • C语言中结构体(struct)的几种初始化方法

    本文给大家总结的struct数据有3种初始化方法 1.顺序 2.C风格的乱序 3.C++风格的乱序 下面通过示例代码详细介绍这三种初始化方法. 1)顺序 这种方法很常见,在一般的介绍C的书中都有介绍.顺序初始化的特点是: 按照成员定义的顺序,从前到后逐个初始化:允许只初始化部分成员:在被初始化的成员之前,不能有未初始化的成员. 示例: struct User oneUser = {10, "Lucy", "/home/Lucy"}; 2)乱序(C风格) 顺序的缺陷是

  • C语言中结构体struct编写的一些要点解析

    一.关于结构体的声明 1.匿名声明.如: struct { int i,j; }point; 说明: 这段代码的含义是,声明一个无名(anonymous)的结构体,并创建了一个结构体变量point.如果这段声明是放在全局域(在任意函数(比如main函数)外)内,那么point内的变量将被初始化为默认值,换句话说,以这种方式声明结构体变量时就已经为它分配了内存空间. 适用于该结构体只需要产生一个变量!本例中,该匿名结构体将有且仅有point这个结构体变量! 不同的匿名结构体变量,类型是不同的!如

  • 解析C语言中结构体struct的对齐问题

    首先看一下结构体对齐的三个概念值: 数据类型的默认对齐值(自身对齐): 1.基本数据类型:为指定平台上基本类型的长度.如在32位机器中,char对齐值为1,short为2,int,float为4,double为8: 结构体:其数据成员中默认对齐值最大的那个值. 2.指定对齐值:#pragma pack (value)时的指定对齐值value. 3.数据类型的有效对齐值:默认对齐值和指定对齐值中小的那个值. 有了这些值,我们就可以很方便的来讨论具体数据结构的成员和其自身的对齐方式.有效对齐值N是最

  • C语言 结构体(Struct)详解及示例代码

    前面的教程中我们讲解了数组(Array),它是一组具有相同类型的数据的集合.但在实际的编程过程中,我们往往还需要一组类型不同的数据,例如对于学生信息登记表,姓名为字符串,学号为整数,年龄为整数,所在的学习小组为字符,成绩为小数,因为数据类型不同,显然不能用一个数组来存放. 在C语言中,可以使用结构体(Struct)来存放一组不同类型的数据.结构体的定义形式为: struct 结构体名{     结构体所包含的变量或数组 }; 结构体是一种集合,它里面包含了多个变量或数组,它们的类型可以相同,也可

  • 浅析c与c++中struct的区别

    这里有两种情况下的区别.(1)C的struct与C++的class的区别.(2)C++中的struct和class的区别.在第一种情况下,struct与class有着非常明显的区别.C是一种过程化的语言,struct只是作为一种复杂数据类型定义,struct中只能定义成员变量,不能定义成员函数(在纯粹的C语言中,struct不能定义成员函数,只能定义变量).例如下面的C代码片断: 复制代码 代码如下: struct Point        {                int x; //

  • C#中struct与class的区别详解

    目录 1.最大的区别 2.struct成员无法被声明为protected. 3.struct是隐式的sealed类 4.struct中无法重载默认构造函数 5.关于对象的初始化 6.结构体没有析构函数,也无法自己去给结构体定义一个析构函数 1.最大的区别 struct类型定义的变量是值类型,class定义的变量是引用类型.因此struct类型定义的对象是分配在栈上面的,而class定义的对象是分配在堆上的. 2.struct成员无法被声明为protected. 举例: struct Test1

  • C#中struct和class的区别详解

    本文详细分析了C#中struct和class的区别,对于C#初学者来说是有必要加以了解并掌握的. 简单来说,struct是值类型,创建一个struct类型的实例被分配在栈上.class是引用类型,创建一个class类型实例被分配在托管堆上.但struct和class的区别远不止这么简单. 概括来讲,struct和class的不同体现在: ● 类是引用类型,struct是值类型 ● 在托管堆上创建类的实例,在栈上创建struct实例 ● 类实例的赋值,赋的是引用地址,struct实例的赋值,赋的是

  • C++中的struct和class的区别详解

    目录 1. C++的struct和class的区别 1.1 成员访问范围的差异 struct class 1.1 继承关系访问范围的差异 struct : struct struct : class struct : private class class : class class : public class class : struct class : public struct 1.3 {}初始化的差异 struct – 纯数据+一般方法 struct – 带构造函数 struct –

  • C++ class和struct到底有什么区别详解

    C++ 中保留了C语言的 struct 关键字,并且加以扩充.在C语言中,struct 只能包含成员变量,不能包含成员函数.而在C++中,struct 类似于 class,既可以包含成员变量,又可以包含成员函数. C++中的 struct 和 class 基本是通用的,唯有几个细节不同: 使用 class 时,类中的成员默认都是 private 属性的:而使用 struct 时,结构体中的成员默认都是 public 属性的. class 继承默认是 private 继承,而 struct 继承默

  • 基于python中staticmethod和classmethod的区别(详解)

    例子 class A(object): def foo(self,x): print "executing foo(%s,%s)"%(self,x) @classmethod def class_foo(cls,x): print "executing class_foo(%s,%s)"%(cls,x) @staticmethod def static_foo(x): print "executing static_foo(%s)"%x a=A(

  • node.js中grunt和gulp的区别详解

    node.js中grunt和gulp的区别详解 自nodeJS登上前端舞台,自动化构建变得越来越流行.目前最流行的当属grunt和gulp,这两个光看名字挺像,功能也差不多,不过gulp能在grunt这位大哥如日中天的境况下开辟出自己的一片天地,有着她独到的优点. 易用 Gulp相比Grunt更简洁,而且遵循代码优于配置策略,维护Gulp更像是写代码. 高效 Gulp相比Grunt更有设计感,核心设计基于Unix流的概念,通过管道连接,不需要写中间文件. 高质量 Gulp的每个插件只完成一个功能

  • 基于js中this和event 的区别(详解)

    今天在看javascript入门经典-事件一章中看到了 this 和 event 两种传参形式.因为作为一个初级的前端开发人员平时只用过 this传参,so很想弄清楚,this和event的区别是什么,什么情况下用什么比较合适. onclick = changeImg(this)       vs     onclick = changeImg(event) <img src='usa.gif' onclick="changeImg(event)" /> <scrip

  • iOS中setValue和setObject的区别详解

    网上关于setValue和setObject的区别的文章很多,说的并不准确,首先我们得知道: setObject:ForKey: 是NSMutableDictionary特有的:setValue:ForKey:是KVC的主要方法 话不多说,上代码: - (void)viewDidLoad { [super viewDidLoad]; //setObject和setvalue的区别 NSMutableDictionary *dic = [NSMutableDictionary dictionary

  • python中import reload __import__的区别详解

    import 作用:导入/引入一个python标准模块,其中包括.py文件.带有__init__.py文件的目录(自定义模块). import module_name[,module1,...] from module import *|child[,child1,...] 注意:多次重复使用import语句时,不会重新加载被指定的模块,只是把对该模块的内存地址给引用到本地变量环境. 实例: pythontab.py #!/usr/bin/env python #encoding: utf-8

  • JavaScript中object和Object的区别(详解)

    JavaScript中object和Object有什么区别,为什么用typeof检测对象,返回object,而用instanceof 必须要接Object呢 这个问题和我之前遇到的问题非常相似,我认为这里有两个问题需要解决,一个是运算符new的作用机制,一个是function关键字和Funtion内置对象之间的区别.看了一些前辈的博客和标准,这里帮提问者总结一下. 1.new new运算符的作用是创建一个对象实例.这个对象可以是用户自定义的,也可以是带构造函数的一些系统自带的对象.如果 new

随机推荐