方法

作者:追风剑情 发布于:2021-4-25 15:20 分类:C#

示例——无参构造器
public class SomeType {
}

//等价于
public class SomeType {
	public SomeType() : base() { }
}

示例:字段初始化

  1. using System;
  2.  
  3. namespace ConsoleApp2
  4. {
  5. class Program
  6. {
  7. static void Main(string[] args)
  8. {
  9. SomeType st = new SomeType("ctor...");
  10. Console.WriteLine(st.ToString());
  11. Console.Read();
  12. }
  13. }
  14.  
  15. internal sealed class SomeType
  16. {
  17. // 如果显示初始化这些字段的值,
  18. // 编译器会将字段的初始化内嵌到每个构造器函数中,
  19. // 这就会造成代码膨胀
  20.  
  21. // 不要显式初始化下面的字段
  22. private Int32 m_x;
  23. private String m_s;
  24. private Double m_d;
  25. private Byte m_b;
  26.  
  27. // 该构造器将所有字段设为默认值
  28. // 其他所有构造器显式调用该构造器
  29. public SomeType()
  30. {
  31. m_x = 5;
  32. m_s = "Hi there";
  33. m_d = 3.14159;
  34. m_b = 0xff;
  35. }
  36.  
  37. // 该构造器将所有的字段都设为默认值,然后修改m_s
  38. public SomeType(String s) : this()
  39. {
  40. m_s = s;
  41. }
  42.  
  43. // 该构造器首先将所有字段都设为默认值,然后修改m_x和m_s
  44. public SomeType(Int32 x, String s) : this()
  45. {
  46. m_x = x;
  47. m_s = s;
  48. }
  49.  
  50. public override string ToString()
  51. {
  52. return string.Format("m_x={0},m_s={1},m_d={2},m_b={3}",
  53. m_x,m_s,m_d,m_b);
  54. }
  55. }
  56. }

运行测试
11111.png

示例——值类型(struct)构造器
internal struct Point
{
	//值类型的字段总是被编译器初始化为0或null
        //不能为实例字段直接赋值
	public Int32 m_x, m_y;
        //可以为静态字段直接赋值
        public static Int32 m_z = 5;

	//错误: CS0568 结构不能包含显式的无参数构造函数
	//因为编译器不会自动调用无参构造器
	//public Point() { }

	public Point(Int32 x, Int32 y)
	{
		//必须对所有字段完成赋值,否则会报错
		m_x = x;
		m_y = y;
	}

	public Point(Int32 x)
	{
		//在值类型(struct)中this是可读写的,在引用类型(class)中this是只读的
		//看起来很奇怪,但编译没问题,会将所有字段初始化为0或null
		this = new Point();
		//现在可以只对部分字段赋值了
		m_x = x;//用x覆盖m_x的0
	}
}

internal sealed class Rectangle
{
	public Point m_topLeft, m_bottomRight;

	public Rectangle()
	{
		//在C#中,向一个值类型应用关键字new,
		//可以调用构造器来初始化值类型的字段
		m_topLeft = new Point(1, 2);
		m_bottomRight = new Point(100, 200);
	}
}

示例——静态构造器
internal sealed class SomeRefType
{
	//下面四种叫法都指同一种构造器:
	//类型构造器(type constructor)
	//静态构造器(static constructor)
	//类构造器(class constructor)
	//类型初始化器(type initializer)
	static SomeRefType()
	{
		//SomeRefType被首次访问时,执行这里的代码
	}
}

internal struct SomeValType
{
	//C#允许值类型定义无参的类型构造器
	static SomeValType()
	{
		//SomeValType被首次访问时,执行这里的代码
	}
}

类型构造器总是私有;C#自动把它们标记为private。之所以必须私有,是为了防止任何由开发人员写的代码调用它,对它的调用总是由CLR负责。

重要提示   虽然能在值类型中定义类型构造器,但永远不要真的那么做,因为CLR有时不会调用值类型的静态类型构造器。

示例——静态构造器
internal struct SomeValType {
	static SomeValType() {
		Console.WriteLine("这句话永远不会显示");
	}
	public Int32 m_x;
}

public sealed class Program {
	public static void Main() {
		SomeValType[] a = new SomeValType[10];
		a[0].m_x = 123;
		Console.WriteLine(a[0].m_x);//显示123
	}
}

示例——静态构造器
internal struct SomeValType {
	private static Int32 s_x = 5;
}

//等效于下面的写法
internal struct SomeValType {
	private static Int32 s_x;
	static SomeValType() {
		s_x = 5;
	}
	
}

注意   由于CLR保证了一个类型构造器在每个AppDomain中只执行一次,而且(这种执行)是线程安全的,所以非常适合在类型构造器中初始化类型需要的任何单实例(Singleton)对象。

单个线程中的两个静态类型构造器包含相互引用的代码可能出问题

操作符重载方法

C#允许重载一元和二元操作符,以及由编译器生成的对应的CLS(Common Language Specification,公共语言规范)方法名。

C#的一元操作符及其相容于CLS的方法名
C#操作符 特殊方法名 推荐的相容于CLS的方法名
+ op_UnaryPlus Plus
- op_UnaryNegation Negate
! op_LogicalNot Not
~ op_OnesComplement OnesComplement
++ op_Increment Increment
-- op_Decrement Decrement
(无) op_True IsTrue{get;}
(无) op_False IsFalse{get;}

C#的二元操作符及其相容于CLS的方法名
C#操作符 特殊方法名 推荐的相容于CLS的方法名
+     op_Addition Add
- op_Subtraction Subtract
* op_Multiply Multiply
/ op_Division Divide
% op_Modulus Mod
& op_BitwiseAnd BitwiseAnd
| op_BitwiseOr BitwiseOr
^ op_ExclusiveOr Xor
<< op_LeftShift
LeftShift
>> op_RightShift RightShift
== op_Equality Equals
!= op_Inequality Equals
< op_LessThan Compare
> op_GreaterThan Compare
<= op_LessThanOrEqual Compare
>= op_GreaterThanOrEqual Compare

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading.Tasks;
  6.  
  7. namespace ConsoleApp4
  8. {
  9. class Program
  10. {
  11. static void Main(string[] args)
  12. {
  13. //隐式转型
  14. Rational r1 = 5;//Int32转Rational
  15. Rational r2 = 2.5F;//Single转Rational
  16.  
  17. //显式转型
  18. Int32 x = (Int32)r1;//Ration转Int32
  19. Single s = (Single)r2;//Ration转Single
  20. }
  21. }
  22.  
  23. public sealed class Rational
  24. {
  25. private Int32 i;
  26. private Single si;
  27.  
  28. // 由一个Int32构造一个Rational
  29. public Rational(Int32 num) { i = num; }
  30. // 由一个Single构造一个Rational
  31. public Rational(Single num) { si = num; }
  32. // 将一个Rational转换成一个Int32
  33. public Int32 ToInt32() { return i; }
  34. // 将一个Rational转换成一个Single
  35. public Single ToSingle() { return si; }
  36. // 由一个Int32隐式构造并返回一个Rational
  37. // 被编译成:
  38. // public static Rational op_Implicit(Int32 num)
  39. public static implicit operator Rational(Int32 num)
  40. {
  41. return new Rational(num);
  42. }
  43. // 由一个Single隐式构造并返回一个Rational
  44. // 被编译成:
  45. // public static Rational op_Implicit(Single num)
  46. public static implicit operator Rational(Single num)
  47. {
  48. return new Rational(num);
  49. }
  50. // 由一个Rational显式返回一个Int32
  51. // 被编译成:
  52. // public static Int32 op_Explicit(Rational r)
  53. public static explicit operator Int32(Rational r)
  54. {
  55. return r.ToInt32();
  56. }
  57. // 由一个Rational显式返回一个Single
  58. // 被编译成:
  59. // public static Single op_Explicit(Rational r)
  60. public static explicit operator Single(Rational r)
  61. {
  62. return r.ToSingle();
  63. }
  64. }
  65. }
注意   只有在转换不损失精度或数量级的前提下才能定义隐式转换操作符。反之,则应该定义显式转换操作符。显式转换失败,应该让显式转换操作符抛出OverflowException或者InvalidOperationException异常。

为了真正理解操作符重载方法和转换操作符方法,强烈建议将 System.Decimal 类型作为典型来研究。

扩展方法

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading.Tasks;
  6.  
  7. namespace ConsoleApp5
  8. {
  9. class Program
  10. {
  11. static void Main(string[] args)
  12. {
  13. StringBuilder sb = new StringBuilder("Hello. My name is Jeff.");
  14. Int32 index = sb.Replace('.', '!').IndexOf('!');
  15. }
  16. }
  17.  
  18. public static class StringBuilderExtensions
  19. {
  20. // 为StringBuilder定义扩展方法IndexOf
  21. public static Int32 IndexOf(this StringBuilder sb, Char value)
  22. {
  23. if (null == sb)//注意: sb可能为null
  24. return -1;
  25. for (Int32 index = 0; index < sb.Length; index++)
  26. if (sb[index] == value) return index;
  27. return -1;
  28. }
  29. }
  30. }


规则和原则:
● C#只支持扩展方法,不支持扩展发展、扩展事件、扩展操作符等。
● 扩展方法必须在非泛型的静态类中声明。
● 扩展方法类不能为嵌套类
● 如果扩展方法类在某个命名空间中,程序员必须导入(using 扩展类所在命名空间)才能使用
● 派生类也会得到扩展方法
● 如果微软在类中增加了和扩展方法相同的方法,我们定义的扩展方法将失效

为接口类型定义扩展方法

  1. using System;
  2. using System.Collections.Generic;
  3.  
  4. namespace ConsoleApp5
  5. {
  6. class Program
  7. {
  8. static void Main(string[] args)
  9. {
  10. //每个Char在控制台上单独显示一行
  11. "Grant".ShowItems();
  12. //每个String在控制台上单独显示一行
  13. new[] { "Jeff", "Kristin" }.ShowItems();
  14. //每个Int32在控制台上单独显示一行
  15. new List<Int32>() { 1, 2, 3 }.ShowItems();
  16. }
  17. }
  18.  
  19. public static class IEnumerableExtensions
  20. {
  21. //任何表达式,只要它最终的类型实现了IEnumerable<T>接口,就能调用此扩展方法
  22. public static void ShowItems<T>(this IEnumerable<T> collection)
  23. {
  24. foreach (var item in collection)
  25. Console.WriteLine(item);
  26. }
  27. }
  28. }


为委托类型定义扩展方法

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading.Tasks;
  6.  
  7. namespace ConsoleApp6
  8. {
  9. class Program
  10. {
  11. static void Main(string[] args)
  12. {
  13. //抛出NullReferenceException
  14. Action<Object> action = o => Console.WriteLine(o.GetType());
  15. //吞噬NullReferenceException
  16. action.InvokeAndCatch<NullReferenceException>(null);
  17.  
  18. //C#编译器允许创建委托来引用一个对象上的扩展方法
  19. Action a = "Jeff".ShowItems;
  20. a();
  21. }
  22. }
  23.  
  24. public static class DelegateExtensions
  25. {
  26. public static void InvokeAndCatch<TException>(this Action<Object> d, Object o)
  27. where TException : Exception
  28. {
  29. try { d(o); }
  30. catch (TException) { }
  31. }
  32. }
  33.  
  34. public static class IEnumerableExtensions
  35. {
  36. //任何表达式,只要它最终的类型实现了IEnumerable<T>接口,就能调用此扩展方法
  37. public static void ShowItems<T>(this IEnumerable<T> collection)
  38. {
  39. foreach (var item in collection)
  40. Console.WriteLine(item);
  41. }
  42. }
  43. }


给枚举添加扩展方法

C#编译器会对定义了扩展方法的静态类应用 ExtensionAttribute 特性。

分部方法

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Threading.Tasks;
  6.  
  7. namespace ConsoleApp6
  8. {
  9. class Program
  10. {
  11. static void Main(string[] args)
  12. {
  13. }
  14. }
  15.  
  16. //工具生成的代码,存储在某个源代码文件中
  17. //这个技术可用于密封类、静态类以及值类型
  18. internal sealed partial class Base
  19. {
  20. private String m_name;
  21.  
  22. //这是分部方法的声明
  23. partial void OnNameChanging(String value);
  24.  
  25. public String Name
  26. {
  27. get { return m_name; }
  28. set
  29. {
  30. //如果OnNameChanging分部方法没有实现主体,这句调用不会产生性能开销
  31. OnNameChanging(value.ToUpper());
  32. m_name = value;
  33. }
  34. }
  35. }
  36.  
  37. //开发人员生成的代码,存储在另一个源代码文件中
  38. internal sealed partial class Base
  39. {
  40. //这是分部方法的实现,会在m_name更改前调用
  41. partial void OnNameChanging(string value)
  42. {
  43. if (String.IsNullOrEmpty(value))
  44. throw new ArgumentNullException("value");
  45. }
  46. }
  47. }


注意   分部方法的工作方式类似于 System.Diagnostics.ConditionalAttribute 特性。然而,分部方法只能在单个类型中使用,而 ConditionalAttribute 能用于对另一个类型中定义的方法进行有选择的调用。
参考:
C#中使用System.Diagnostics.ConditionalAttribute移除无用函数调用
ConditionalAttribute

规则和原则
● 它们只能在分部类或结构中声明。
● 分部方法的返回类型始终是 void,任何参数都不能用 out 修饰符来标记。之所以有这两个限制,是因为方法在运行时可能不存在,所以不能将变量初始化为方法也许会返回的东西。 类似地,不允许 out 参数是因为方法必须初始化它,而方法可能不存在。分部方法可以有 ref 参数,可以是泛型方法,可以是实例或静态方法,而且可以标记为 unsafe
● 当然,分部方法的声明和实现必须具有完全一致的签名。如果两者都应用了定制特性,编译器会合并两个方法的特性。应用于参数的任何特性也会合并。
● 如果没有对应的实现部分,便不能在代码中创建一个委托来引用这个分部方法。这同样是由于方法在运行时不存在。 ● 分部方法总是被视为 private 方法,但C#编译器禁止在分部方法声明之前添加 private 关键字。

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号