C#重载运算符详解
作者:shichen2014 发布时间:2023-10-27 13:35:18
本文较为详细的描述了重载运算符的方法。一般来说,重载运算符在实际的项目开发中会经常的用到,但如果某些自定义类型通过简短几行代码重载一些常用的运算符(如:+-*/),就能让编程工作带来方便;重载运算符就是告诉编译器+-*/等运算符对于自定义类型进行什么样的操作,在代码中需要注意几点。
一、尽可能的不要改变运算符本身的含义
二、所有的运算符重载都必须声明为public和static
三、不同于扩展方法,所重载的方法必须是在被重载的类型内部,且用关键字operator
C#中的两个字符串相加,实际上是连接两个字符串,假如有两个EmployeeDetail类型相加得到一个EmployeeCollection集合,如:
EmployeeDetail a,b;
....
EmployeeCollection collection = a+b;
当编译器遇到上面的代码时就会自动调用EmployeeDetail类上标有operator +的静态方法,并将两个操作数a和b作为参数传递给对于的方法,该方法需要方法一个值赋给collection,假设EmployeeDetail类有三个属性分别是FirstName,MiddleName,LastName,还重写了ToString方法返回一个连接这三个名称的字符串,代码如:
[Serializable]
public class EmployeeDetail
{
public string FirstName { get; set; }
public string MiddleName { get; set; }
public string LastName { set;get; }
public override string ToString()
{
return string.Format("{0}{1}{2}{3}{4}", FirstName, string.IsNullOrWhiteSpace(MiddleName) ? null : "."
, MiddleName
, string.IsNullOrWhiteSpace(LastName) ? null : ".",
LastName).Trim();
}
}
下面的代码为“+”运算符提供支持的运算符重载:
public static EmployeeCollection operator +(EmployeeDetail a, EmployeeDetail b)
{
return new EmployeeCollection() { a, b };
}
OK,给EmployeeDetail类加上这样的一个方法之后,我们就可以像下面那个写代码了:
EmployeeCollection collection = new EmployeeDetail(){FirstName="Jackson",LastName="Bruce"} + new EmployeeDetail(){FirstName="Michael",LastName="Jackson"} ;
但是这样还不够完美,假设a,b,c都是EmployeeDetail类型,下面的代码会抛出一个编译错误:
EmployeeCollection collection = a + b + c;
为什么编译不通过呢?大家都知道除了赋值运算符外表达式是从左到右执行的,a+b返回的是EmployeeCollection类型,EmployeeCollection类型并没有重载“+”运算符,编译器不知道要执行什么操作,所以我们还有下面的两个方法:
public static EmployeeCollection operator +(EmployeeCollection collection, EmployeeDetail a)
{
collection.Add(a);
return collection;
}
public static EmployeeCollection operator +(EmployeeDetail a, EmployeeCollection collection)
{
return collection + a;
}
这看起来似乎已经很完美了,但我们还可以做得更好一些,比如要将字符串“Jackson.Bruce”直接隐式转换为EmployeeDetail类型,也就是说可以将“Jackson.Bruce"这种格式的字符串直接赋给EmployeeDetail类型的对象,如:EmployeeDetail employee= “Jackson.Bruce",那么就需要重载隐式类型转换运算符了,代码如下:
/// <summary>
/// 隐式类型转换
/// </summary>
/// <param name="name"></param>
/// <returns></returns>
public static implicit operator EmployeeDetail(string name)
{
/// 其实在这里可以写一个正则表达式检查name的字符串格式是否合法,如果不合法就抛出异常
///
string[] arr;
return string.IsNullOrWhiteSpace(name) ? null :
new EmployeeDetail()
{
FirstName = (arr = name.Trim().Split('.'))[0]
,
LastName = arr.Length > 1 ? arr[arr.Length > 2 ? 2 : 1] : null,
MiddleName = arr.Length > 2 ? arr[1] : null
};
}
public static EmployeeCollection operator +(EmployeeDetail a, string b)
{
return new EmployeeCollection() { a, b };
}
看到这里您是不是迫不及待地想试试看,OK写个控制台程序来测试一下:
static void Main(string[] args)
{
EmployeeDetail employee = "Jackson.Bruce";
Console.WriteLine("FirstName={0},MiddleNam={1},LastName={2}", employee.FirstName, employee.MiddleName, employee.LastName);
Console.WriteLine("toString={0}", employee);
Console.WriteLine();
EmployeeCollection collection = "Michael.Jackson" + employee;
collection += "Bruce.Lee";
foreach (var e in collection)
{
Console.WriteLine(e);
}
Console.WriteLine();
collection -= employee;
foreach (var e in collection)
{
Console.WriteLine(e);
}
Console.WriteLine("===end===");
Console.Read();
}
运行结果如下图所示:
全部代码,里面还包含其他运算符的重载,这里就不再介绍了,赶紧动手测试一下吧:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
namespace 重载运算符
{
[Serializable]
public class EmployeeDetail
{
public string FirstName { get; set; }
public string MiddleName { get; set; }
public string LastName { set;get; }
public static EmployeeCollection operator +(EmployeeDetail a, EmployeeDetail b)
{
return new EmployeeCollection() { a, b };
}
public static EmployeeCollection operator +(EmployeeCollection collection, EmployeeDetail a)
{
collection.Add(a);
return collection;
}
public static EmployeeCollection operator +(EmployeeDetail a, EmployeeCollection collection)
{
return collection + a;
}
/// <summary>
/// 隐式类型转换
/// </summary>
/// <param name="name"></param>
/// <returns></returns>
public static implicit operator EmployeeDetail(string name)
{
string[] arr;
return string.IsNullOrWhiteSpace(name) ? null :
new EmployeeDetail()
{
FirstName = (arr = name.Trim().Split('.'))[0]
,
LastName = arr.Length > 1 ? arr[arr.Length > 2 ? 2 : 1] : null,
MiddleName = arr.Length > 2 ? arr[1] : null
};
}
public static EmployeeCollection operator +(EmployeeDetail a, string b)
{
return new EmployeeCollection() { a, b };
}
public override string ToString()
{
return string.Format("{0}{1}{2}{3}{4}", FirstName, string.IsNullOrWhiteSpace(MiddleName) ? null : "."
, MiddleName
, string.IsNullOrWhiteSpace(LastName) ? null : ".",
LastName).Trim();
}
}
public class EmployeeCollection : List<EmployeeDetail>
{
public static EmployeeCollection operator +(EmployeeCollection a, string b)
{
a.Add(b);
return a;
}
public static EmployeeCollection operator +(string b, EmployeeCollection a)
{
return a + b;
}
public static EmployeeCollection operator -(EmployeeCollection a, EmployeeDetail b)
{
a.Remove(b);
return a;
}
}
class Program
{
static void Main(string[] args)
{
EmployeeDetail employee = "Jackson.Bruce";
Console.WriteLine("FirstName={0},MiddleNam={1},LastName={2}", employee.FirstName, employee.MiddleName, employee.LastName);
Console.WriteLine("toString={0}", employee);
Console.WriteLine();
EmployeeCollection collection = "Michael.Jackson" + employee;
collection += "Bruce.Lee";
foreach (var e in collection)
{
Console.WriteLine(e);
}
Console.WriteLine();
collection -= employee;
foreach (var e in collection)
{
Console.WriteLine(e);
}
Console.WriteLine("===end===");
Console.Read();
}
}
}


猜你喜欢
- 本文实例讲述了C#不重复输出一个数组中所有元素的方法。分享给大家供大家参考。具体如下:1.算法描述0)输入合法性校验1)建立临时数组:与原数
- minio 注册成windows 服务的工具开发using System;using System.Collections.Generic;
- eclipse导入appcompat项目报错解决办法我们在eclipse导入开源项目后,经常会发现找不到类似Theme.AppCompat.
- 异常处理是每个项目中都绕不开的话题,那么如何优雅的处理异常,是本文的话题。本文将结合SpringBoot框架一起和大家探讨下。要思考的问题在
- TextView加载字体包在 Android 中,若需要使得某个TextView加载字体包,使用以下方式即可: Typeface typeF
- 本文实例讲述了C#时间戳基本用法。分享给大家供大家参考。具体如下:一、C#如何生成一个时间戳/// <summary> ///
- Android 消息分发使用EventBus的实例详解1. AndroidStudio使用dependencies {//最新版本 &nbs
- 像javascript中有eval()来执行动态代码,c#中是没有的,于是自己动手丰衣足食,先来代码using System;using S
- java 二叉查找树实例代码1.左边<中间<右边2.前序遍历 左中右3.中序遍历 中左右4.后序遍历 左右中public cla
- 前言在日常的开发中StringBuilder大家肯定都有用过,甚至用的很多。毕竟大家都知道一个不成文的规范,当需要高频的大量的构建字符串的时
- 使用Aspose.Cells创建和读取Excel文件,供大家参考,具体内容如下1. 创建ExcelAspose.Cells.License
- 前言工作中你可能会遇到很多这样的场景,一个接口,要从其他几个service调用查询方法,分别获取到需要的值之后再封装数据返回。还可能在微服务
- 大家可能在做app的时候,或多或少需要使用联系人,而根据google提供的api,你需要编写大量的代码,例如首先需要查询数据库,涉及到数据库
- 首先struts上传最大大小由两个地方决定. · struts.multipart.maxSize决定整个post的form最大是
- 本文主要是分析Spring bean的循环依赖,以及Spring的解决方式。 通过这种解决方式,我们可以应用在我们实际开发项目中。1. 什么
- MVC注解式开发即处理器基于注解的类开发, 对于每一个定义的处理器, 无需在xml中注册.只需在代码中通过对类与方法的注解, 即可完成注册.
- 使用JDBC时,我们都会很自然得使用下列语句:Class.forName("com.mysql.jdbc.Driver"
- 我们在构建C# Form窗口的时候经常需要到弹出新的窗口,那么接着就会如何弹出窗口的疑问。这里介绍最常见的两种弹窗方法show()和show
- 概述不知道大家在平时的开发过程中或者源码里是否留意过内部类,那有思考过为什么要有内部类,内部类都有哪几种形式,静态内部类和普通内部类有什么区
- 0x01 新建SpringBoot项目1. 新建maven工程ps:在上一教程的基础上操作,就不用新建项目了,请参考文章:SpringBoo