11.5.1 問題的提出
在面向對象的程序設計中,自己定義一個類,就等於創建了一個新類型。類的實例和變量一樣,可以作為參數傳遞,也可以作為返回類型。
在第七章中,我們介紹了系統定義的許多操作符。比如對於兩個整型變量,使用算術操作符可以簡便地進行算術運算:
class A
{
public int x;
public int y;
public int Plus{
return x+y;
}
}
再比如,我們希望將屬於不同類的兩個實例的數據內容相加:
class B
{
public int x;
}
class Test
{
public int z;
public static void Main{
A a=new A();
B b=new B();
z=a.x+b.x;
}
}
使用a.x+b.x這種寫法不夠簡潔,也不夠直觀。更為嚴重的問題是,如果類的成員中聲明時使用的不是publixc修飾符的話,這種訪問就是非法的。
我們知道,在C#中,所有數據要麼屬於某個類,要麼屬於某個類的實例,充分體現了面向對象的思想。因此,為了表達上的方便,人們希望可以重新給已定義的操作符賦予新的含義,在特定的類的實例上進行新的解釋。這就需要通過操作符重載來解決。
11.5.2 使用成員方法重載操作符
C#中,操作符重載總是在類中進行聲明,並且通過調用類的成員方法來實現。
操作符重載聲明的格式為:
type operator operator-name(formal-param-list)
C#中,下列操作符都是可以重載的:
+ - ! ~ ++ -- true false
* / % & | ^ << >> == != > < >= <=
但也有一些操作符是不允許進行重載的,如:
=,&&,||,?:,new,typeof,sizeof,is
一元操作符重載
顧名思義,一元操作符重載時操作符只作用於一個對象,此時參數表為空,當前對象作為操作符的單操作數。
下面我們舉一個角色游戲中經常遇到的例子。扮演的角色具有內力、體力、經驗值、剩余體力、剩余內力五個屬性,每當經驗值達到一定程序時,角色便會升級,體力、內力上升,剩余體力和內力補滿。“升級”我們使用重載操作符“++”來實現。
程序清單11-10:
using System;
class Player
{
public int neili;
public int tili;
public int jingyan;
public int neili_r;
public int tili_r;
public Player()
{
neili=10;
tili=50;
jingyan=0;
neli_r=50;
tili_r=50;
}
public static Player operator ++(Player p){
p.neili=p.neili+50;
p.tili=p.tili+100;
p.neili_r=p.neili;
p.tili_r=p.tili;
return p;
}
public void Show()
{
Console.WriteLine("Tili:{0}",tili);
Console.WriteLine("Neili:{0}",neili);
Console.WriteLine("Tili_full:{0}",tili_r);
Console.WriteLine("Neili_full:{0}",neili_r);
}
}
class Test
{
public static void Main(){
Player man=new Player();
man.Show();
man++;
Console.WriteLine("Now upgrading...:");
man.Show();
}
}
二元操作符重載
大多數情況下我們使用二元操作符重載。這時參數表中有一個參數,當前對象作為該操作符的左操作數,參數作為操作符的右操作數。
下面我們給出二元操作符重載的一個簡單例子,即笛卡兒坐標相加。
程序清單11-11:
using System;
class DKR
{
public int x,y,z;
public DKR(int vx,int vy,int vz){
x=vx;
y=vy;
z=vz;
}
public static DKR operator +(DKR d1,DKR d2)
{
DKR dkr=new DKR(0,0,0);
dkr.x=d1.x+d2.x;
dkr.y=d1.y+d2.y;
dkr.z=d1.z+d2.z;
return dkr;
}
}
class Test
{
publixc statixc void Main(){
DKR d1=new DKR(3,2,1);
DKR d2=new DKR(0,6,5);
DKR d3=d1+d2;
Console.WriteLine("The 3d location of d3 is:{0},{1},{2}",d3.x,d3.y,d3,z);
}
}
試著編譯、運行該程序,看結果是否與預期一致。