Visual

C#语言入门详解(017)——字段,属性,索引器,常量.mp4 (192.03MB) 这四种成员都是用来表示数据的,所以放在一起讲。

017 字段、属性、索引器、常量 - 图2

字段

017 字段、属性、索引器、常量 - 图3

  1. 字段历史悠久,C 语言就有了
  2. ID、Name 各占一块空间,这也是字段称为 field 的由来

017 字段、属性、索引器、常量 - 图4

什么是字段

示例:实例字段与静态字段

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. var stuList = new List<Student>();
  6. for (int i = 0; i < 100; i++)
  7. {
  8. var stu = new Student
  9. {
  10. Age = 24,
  11. Score = i,
  12. };
  13. stuList.Add(stu);
  14. }
  15. int totalAge=0;
  16. int totalScore=0;
  17. foreach (var stu in stuList)
  18. {
  19. totalAge += stu.Age;
  20. totalScore += stu.Score;
  21. }
  22. Student.AverageAge = totalAge / Student.Amout;
  23. Student.AverageScore = totalScore / Student.Amout;
  24. Student.ReportAmount();
  25. Student.ReportAverageAge();
  26. Student.ReportScore();
  27. }
  28. class Student
  29. {
  30. public int Age;
  31. public int Score;
  32. public static int AverageAge;
  33. public static int AverageScore;
  34. public static int Amout;
  35. public Student()
  36. {
  37. Amout++;
  38. }
  39. public static void ReportAmount()
  40. {
  41. Console.WriteLine(Amout);
  42. }
  43. public static void ReportAverageAge()
  44. {
  45. Console.WriteLine(AverageAge);
  46. }
  47. public static void ReportScore()
  48. {
  49. Console.WriteLine(AverageScore);
  50. }
  51. }
  52. }

字段的声明

字段(field)是一种表示与对象或类关联的变量的成员。

  • 对于实例字段,它初始化的时机是在实例创建时
    • 声明实例字段时初始化值与在实例构造器里面初识化实例字段是一样的
  • 对于静态字段,它初始化的时机是在运行环境加载该数据类型时
    • 即静态构造器初始化时
    • 声明静态字段时设置初始化值与在静态构造器里面初始化静态字段其实是一样的

数据类型被运行环境加载时,它的静态构造器将会被调用,且只被调用一次。

静态构造器:

  1. public static int Amout=100;
  2. static Student()
  3. {
  4. Amout = 200;
  5. }

只读字段 readonly

只读字段为实例或类型保存一旦初始化后就不希望再改变的值。

:::info Code Shippet: ctor + 2 * TAB:插入构造函数代码片段。 :::

属性

017 字段、属性、索引器、常量 - 图5

建议:永远使用属性(而不是字段)来暴露数据,即字段永远是 private 或 protected 的。
字段只在类内部使用,类之间交换数据,永远只用属性。

什么是属性

使用 Get/Set 方法对之前:

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. var stu1 = new Student()
  6. {
  7. Age = 20
  8. };
  9. var stu2 = new Student()
  10. {
  11. Age = 20
  12. };
  13. var stu3 = new Student()
  14. {
  15. // 非法值,污染字段
  16. Age = 200
  17. };
  18. var avgAge = (stu1.Age + stu2.Age + stu3.Age) / 3;
  19. Console.WriteLine(avgAge);
  20. }
  21. }
  22. class Student
  23. {
  24. public int Age;
  25. }

使用 Get/Set 后:

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. try
  6. {
  7. var stu1 = new Student();
  8. stu1.SetAge(20);
  9. var stu2 = new Student();
  10. stu2.SetAge(20);
  11. var stu3 = new Student();
  12. stu3.SetAge(200);
  13. var avgAge = (stu1.GetAge() + stu2.GetAge() + stu3.GetAge()) / 3;
  14. Console.WriteLine(avgAge);
  15. }
  16. catch (Exception ex)
  17. {
  18. Console.WriteLine(ex.Message);
  19. }
  20. }
  21. }
  22. class Student
  23. {
  24. private int age;
  25. public int GetAge()
  26. {
  27. return age;
  28. }
  29. public void SetAge(int value)
  30. {
  31. if (value >= 0 && value <= 120)
  32. {
  33. age = value;
  34. }
  35. else
  36. {
  37. throw new Exception("Age value has error.");
  38. }
  39. }
  40. }

使用 Get/Set 来保护字段的方法至今仍在 C++、JAVA 里面流行(即 C++、JAVA 里面是没有属性的)。
因为 Get/Set 写起来冗长,微软应广大程序员请求,给 C# 引入了属性。

引入属性后:

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. try
  6. {
  7. var stu1 = new Student();
  8. stu1.Age=20;
  9. var stu2 = new Student();
  10. stu2.Age = 20;
  11. var stu3 = new Student();
  12. stu3.Age = 200;
  13. var avgAge = (stu1.Age + stu2.Age + stu3.Age) / 3;
  14. Console.WriteLine(avgAge);
  15. }
  16. catch (Exception ex)
  17. {
  18. Console.WriteLine(ex.Message);
  19. }
  20. }
  21. }
  22. class Student
  23. {
  24. private int age;
  25. public int Age
  26. {
  27. get
  28. {
  29. return age;
  30. }
  31. set
  32. {
  33. if (value >= 0 && value <= 120)
  34. {
  35. age = value;
  36. }
  37. else
  38. {
  39. throw new Exception("Age value has error.");
  40. }
  41. }
  42. }
  43. }

引入属性后代码简洁明了许多。

语法糖背后的秘密:
参照 13 14 15 16 表达式、语句详解 用 ildasm 反编译工具,反编译上面的示例。
发现编译器使用语法糖,把背后比较复杂的机制掩蔽了起来,即属性也是一种语法糖。
017 字段、属性、索引器、常量 - 图6

017 字段、属性、索引器、常量 - 图7

属性的声明

017 字段、属性、索引器、常量 - 图8

:::info Code Snippet:

  • prop + 2 * TAB:属性的简略声明
  • propfull + 2 * TAB:属性的完整声明 :::

017 字段、属性、索引器、常量 - 图9

:::info 将字段转换成属性,单击字段变量,Ctrl+R+E ::: **
图片.png

动态计算值的属性

主动计算,每次获取 CanWork 时都计算,适用于 CanWork 属性使用频率低的情况。

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. try
  6. {
  7. var stu1 = new Student();
  8. stu1.Age = 12;
  9. Console.WriteLine(stu1.CanWork);
  10. }
  11. catch (Exception ex)
  12. {
  13. Console.WriteLine(ex.Message);
  14. }
  15. }
  16. }
  17. class Student
  18. {
  19. private int age;
  20. public int Age
  21. {
  22. get { return age; }
  23. set
  24. {
  25. age = value;
  26. }
  27. }
  28. public bool CanWork
  29. {
  30. get
  31. {
  32. return age > 16;
  33. }
  34. }
  35. }

被动计算,只在 Age 赋值时计算一次,适用于 Age 属性使用频率低,CanWork 使用频率高的情况。

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. try
  6. {
  7. var stu1 = new Student();
  8. stu1.Age = 12;
  9. Console.WriteLine(stu1.CanWork);
  10. }
  11. catch (Exception ex)
  12. {
  13. Console.WriteLine(ex.Message);
  14. }
  15. }
  16. }
  17. class Student
  18. {
  19. private int age;
  20. public int Age
  21. {
  22. get { return age; }
  23. set
  24. {
  25. age = value;
  26. CalculateCanWork();
  27. }
  28. }
  29. private bool canWork;
  30. public bool CanWork
  31. {
  32. get { return canWork; }
  33. }
  34. private void CalculateCanWork()
  35. {
  36. canWork = age > 16;
  37. }
  38. }

索引器

017 字段、属性、索引器、常量 - 图11

:::info Code Snippet: index + 2 * TAB:快速声明索引器 :::

索引器一般都是用在集合上面,像如下示例这样用是很少见的(只是为了讲解方便)。

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. var stu = new Student();
  6. stu["Math"] = 90;
  7. stu["Math"] = 100;
  8. var mathScore = stu["Math"];
  9. Console.WriteLine(mathScore);
  10. }
  11. }
  12. class Student
  13. {
  14. private Dictionary<string, int> scoreDictionary = new Dictionary<string, int>();
  15. public int? this[string subject]
  16. {
  17. get
  18. {
  19. if (scoreDictionary.ContainsKey(subject))
  20. {
  21. return scoreDictionary[subject];
  22. }
  23. else
  24. {
  25. return null;
  26. }
  27. }
  28. set
  29. {
  30. if (value.HasValue == false)
  31. {
  32. throw new Exception("Score cannot be null");
  33. }
  34. if (scoreDictionary.ContainsKey(subject))
  35. {
  36. // 可空类型的 Value 属性才是其真实值。
  37. scoreDictionary[subject] = value.Value;
  38. }
  39. else
  40. {
  41. scoreDictionary.Add(subject, value.Value);
  42. }
  43. }
  44. }
  45. }

常量

017 字段、属性、索引器、常量 - 图12

017 字段、属性、索引器、常量 - 图13

各种“只读”的应用场景

  • 常量:隶属于类型,没有所谓的实例常量
  • 只读字段:只有一次初始化机会,就是在声明它时初始化(等价于在构造函数中初始化)
  • 只读属性:对于类使用静态只读属性,对于实例使用实例只读属性
    • 要分清没有 Set,与 private Set 的区别
    • 常量比静态只读属性性能高,因为编译时,编译器将用常量的值代替常量标识符
  • 静态只读字段:字段没有类型局限,但常量只能是简单类型,不能是类/自定义结构体类型