操作符重载

有的编程语言允许一个类型定义操作符应该如何操作类型的实例,比如string类型和int类型都重载了(==)和(+)等操作符,当编译器发现两个int类型的实例使用+操作符的时候,编译器会生成把两个整数加到一起的代码。

当编译器发现两个string类型的实例使用+操作符的时候,编译器会生成把两个字符串连接到一起的代码。那么编译器怎么就会知道这样做呢?如何进行操作符重载呢?

下面C#代码展示了一个类中如何进行操作符重载:

  1. namespace DoNet.Seven.ConsoleApplicationTest
  2. {
  3. class Program
  4. {
  5. static void Main(string[] args)
  6. {
  7. rational r1 = new rational();
  8. rational r2 = new rational();
  9. rational r3= r1 + r2;
  10. Console.WriteLine(r3.Value);
  11. Console.ReadKey();
  12.  
  13. }
  14. }
  15. //有理数
  16. public sealed class rational
  17. {
  18. private int _value = ;
  19.  
  20. public int Value
  21. {
  22. get { return _value; }
  23. set { _value = value; }
  24. }
  25. public rational(int value)
  26. {
  27. this._value = value;
  28. }
  29.  
  30. public rational()
  31. {
  32.  
  33. }
  34. public static rational operator+(rational num1,rational num2)
  35. {
  36. rational result = new rational(num1.Value+num2.Value);
  37. return result;
  38. }
  39.  
  40. }
  41. }

运行代码输入结果是15

用IL工具看下编译器生成的代码如下:

1、首先CLR规范要求操作符重载方法必须是public和static方法。另外,C#编译器要求操作符重载方法至少有一个参数的类型与当前定义的这个方法的类型相同。之所以这样做,

是为了是编译器能在合理的时间内找到要绑定的操作符方法。

2、编程语言的编译器看到源码中出现一个+操作符时,会检查是否有一个操作数的类型定义了一个名为op_Addtion的specialname方法,而且该方法的参数兼容于操作数的类型,

如果存在这样的一个方法,编译器就生成调用它的代码。如果不存在这样的一个方法,就生成一个编译错误。

3、对于其它操作符编译之后对应的方法如下表所示(左边是一元操作符,右边是二元操作符)

                                

转换操作符

当设计一个类型时应该考虑到和其它类型之间的转换,这个其实很重要,将对我们的编码有很大的好处,就像每个类型都会有的一个方法Tostring()一样,我们定义一个int类型,可以很方便的用tostring()方法把

int转换为string,当然也可以转换为其它类型。就像上面的rational一样,如果能将一个int或者double转换为一个rational,就会很方便,反之亦然。

  1. //有理数
  2. public sealed class rational
  3. {
  4. private int _value = ;
  5.  
  6. public int Value
  7. {
  8. get { return _value; }
  9. set { _value = value; }
  10. }
  11. public rational(int value)
  12. {
  13. this._value = value;
  14. }
  15.  
  16. public rational(double value)
  17. {
  18. this._value =(int)value;
  19. }
  20. public rational()
  21. {
  22.  
  23. }
  24.  
  25. public int ToInt()
  26. {
  27. return _value;
  28. }
  29. public double ToDouble()
  30. {
  31. return (double)_value;
  32. }
  33. public static rational operator+(rational num1,rational num2)
  34. {
  35. rational result = new rational(num1.Value+num2.Value);
  36. return result;
  37. }
  38.  
  39. }

1、调用这些构造器和方法,开发人员可以很方便的将int和double对象转换成rational对象,这将给编程工作带来很多方便。设计类型时,应该认真考虑类型需要支持的转换构造器和方法。

2、int i=10;long j=i;这样的代码我们经常会看到,那么从int类型到long类型的转换为什么就可以隐士的进行呢?这就涉及到了我们的转换操作符,下面我们也为rational定义几个转换操作符。

  1. namespace DoNet.Seven.ConsoleApplicationTest
  2. {
  3. class Program
  4. {
  5. static void Main(string[] args)
  6. {
  7. int n = ;
  8. rational r1 = n;
  9. double d=(double)r1;
  10.  
  11. Console.WriteLine(r1.Value);
  12. Console.WriteLine(d.ToString());
  13. Console.ReadKey();
  14.  
  15. }
  16. }
  17. //有理数
  18. public sealed class rational
  19. {
  20. private int _value = ;
  21.  
  22. public int Value
  23. {
  24. get { return _value; }
  25. set { _value = value; }
  26. }
  27. public rational(int value)
  28. {
  29. this._value = value;
  30. }
  31.  
  32. public rational(double value)
  33. {
  34. this._value =(int)value;
  35. }
  36. public rational()
  37. {
  38.  
  39. }
  40.  
  41. public int ToInt()
  42. {
  43. return _value;
  44. }
  45. public double ToDouble()
  46. {
  47. return (double)_value;
  48. }
  49. public static rational operator+(rational num1,rational num2)
  50. {
  51. rational result = new rational(num1.Value+num2.Value);
  52. return result;
  53. }
  54.  
  55. public static implicit operator rational(int value)
  56. {
  57. return new rational(value);
  58. }
  59. public static implicit operator rational(double value)
  60. {
  61. return new rational(value);
  62. }
  63. public static explicit operator int(rational value)
  64. {
  65. return value.ToInt();
  66. }
  67. public static explicit operator double(rational value)
  68. {
  69. return value.ToDouble();
  70. }
  71.  
  72. }
  73. }

输出的结果是10、10。 我们可以在rational、int、double之间来回转换,是不是觉得挺方便的,在这个过程中,编译器又帮我们做了什么呢?

在C#中,implicit关键字告诉编译器为了生成代码来调用方法,不需要在源代码中进行显示转换,相反,explicit关键字告诉编译器只有在发现了显示转型时,才调用方法。

在implicit或explicit关键字之后,要指定operator关键字告诉编译器该方法是一个转换操作符。在operator之后,指定对象要转换成什么类型。在圆括号内,则指定要从什么类型转换。

C#编译器检测到代码中的转型,并内部生成IL代码来调用rational类型定义的转换操作符方法,如果用反编译器看的话可以发现,转换操作符方法会生成下面这样的代码:

结论

不论是操作符重载还是转换操作符,都是在设计类型是考虑到我们编码方便而设计的,下面我们看下C#中decimal类型中的定义。

C#中如何利用操作符重载和转换操作符的更多相关文章

  1. C#中如何利用操作符重载和转换操作符 (转载)

    操作符重载 有的编程语言允许一个类型定义操作符应该如何操作类型的实例,比如string类型和int类型都重载了(==)和(+)等操作符,当编译器发现两个int类型的实例使用+操作符的时候,编译器会生成 ...

  2. C#关于操作符重载与转换

    随便写写 首先,假设我们有一个Person类型 其类型定义如下 class Person { public string Name { get; set; } = "Person" ...

  3. (二) operator、explicit与implicit 操作符重载

      有的编程语言允许一个类型定义操作符应该如何操作类型的实例,比如string类型和int类型都重载了(==)和(+)等操作符,当编译器发现两个int类型的实例使用+操作符的时候,编译器会生成把两个整 ...

  4. [置顶] operator overloading(操作符重载,运算符重载)运算符重载,浅拷贝(logical copy) ,vs, 深拷贝(physical copy)

    operator overloading(操作符重载,运算符重载) 所谓重载就是重新赋予新的意义,之前我们已经学过函数重载,函数重载的要求是函数名相同,函数的参数列表不同(个数或者参数类型).操作符重 ...

  5. C++解析(17):操作符重载

    0.目录 1.操作符重载 2.完善的复数类 3.小结 1.操作符重载 下面的复数解决方案是否可行? 示例1--原有的解决方案: #include <stdio.h> class Compl ...

  6. c++ 操作符重载和友元

    操作符重载(operator overloading)是C++中的一种多态,C++允许用户自定义函数名称相同但参数列表不同的函数,这被称为函数重载或函数多态.操作符重载函数的格式一般为: operat ...

  7. C++一些注意点之操作符重载

    重载操作符需要注意 (1)重载操作符必须具有一个类类型操作数.不能重载内建类型的操作符. operator +(int,int);//这个是错误的,都为内建类型 operator +(int,clas ...

  8. 15.C++-操作符重载

    首先回忆下以前学的函数重载 函数重载 函数重载的本质为相互独立的不同函数 通过函数名和函数参数来确定函数调用 无法直接通过函数名得到重载函数的入口地址 函数重载必然发生在同一个作用域中 类中的函数重载 ...

  9. 15.C++-操作符重载、并实现复数类

    首先回忆下以前学的函数重载 函数重载 函数重载的本质为相互独立的不同函数 通过函数名和函数参数来确定函数调用 无法直接通过函数名得到重载函数的入口地址 函数重载必然发生在同一个作用域中 类中的函数重载 ...

随机推荐

  1. sessionStorage html5客户端本地存储之sessionStorage及storage事件

    可以看一下<JavaScript本地存储实践(html5的localStorage和ie的userData)>sessionStorage和上文中提到的localStorage非常相识,方 ...

  2. c++库大全

    1.C++各大有名库的介绍——C++标准库 2.C++各大有名库的介绍——准标准库Boost 3.C++各大有名库的介绍——GUI 4.C++各大有名库的介绍——网络通信 5.C++各大有名库的介绍— ...

  3. MySQL中的数据类型

    文本 CHAR(*):最多255个字节的定长字符串,它的长度必须在创建时指定 VARCHAR(*):最多255个字节的可变长度字符串,它的长度必须在创建时指定 TEXT:最大长度为64K字符的变长文本 ...

  4. 音乐社交APP源码 V1.1

    1.关于音乐曲库,对接的是百度音乐,会自动随搜索链接百度曲库2.便捷聊天,采用xmpp基本架构.3.加入和整理了群聊天.4.分布式聊天,喜欢该专辑直接进入聊天,喜欢该音乐的进入聊天.5.采用兴趣社交和 ...

  5. crontab的使用说明

    网上瞎转载的,仅供参考 名称 : crontab 使用权限 : 所有使用者 使用方式 : crontab file [-u user]-用指定的文件替代目前的crontab. crontab-[-u ...

  6. html5面向对象做一个贪吃蛇小游戏

    canvas加面向对象方式的贪吃蛇 2016-08-25 这个小游戏可以增加对面向对象的理解,可以加强js逻辑能力,总之认真自己敲一两遍收获还是不少啊!!适合刚学canvas的同学练习!! 废话不多说 ...

  7. 网页热力图 heatmap js

    HBuilder +js 实现网页热力图 废话不多说,上代码 <!DOCTYPE html> <html> <head> <title>111</ ...

  8. win php nginx 配置小细节

    win下配置php Nginx 首先 下载 php-Windows版本.下载Nginx Windows 版本 1> php.ini-production 修改为 php.ini 让其成为php的 ...

  9. QQ分组实现,可收缩---ExpandableListView

    activity: package com.zzw.qqgroup; import java.util.ArrayList; import java.util.HashMap; import java ...

  10. 年薪10W和100w的人差距在哪?

    12年前,我直升了硕士,在家闲得慌,去一家香港的婴幼儿杂志全职实习,每天早上8点上班,下午5点下班,一个月我负责20p左右的内容,实习工资800元. 公司很小,没有办公室政治,大家都很松散,上班打打游 ...