青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

C++ Programmer's Cookbook

{C++ 基礎} {C++ 高級} {C#界面,C++核心算法} {設計模式} {C#基礎}

c#運算符重載和技巧 (equal()函數的實現使用)

Overview

Occasionally this question pops up in newsgroups and forums : Why does C# insist on operator overloads being static? The person raising the question also usually complains how this prevents him or her from implementing virtual overloaded operators.

This article explains why operator overloads have to be static in C# (or any other MSIL compiler), and also shows how you can simulate virtual overloaded operators very easily. It's not a universe-shattering theory (nor a very original one for that matter) and uses a very simple pattern though it's this very simplicity that makes it interesting.

So, why do they have to be static?

In C#, GC'd objects are heap-allocated (the managed CLR heap, not the CRT heap) and thus GC'd objects are always used as references (into the CLR heap). You cannot have stack-based GC'd objects in C# and what this means that you never know when an object variable is null. So, you can imagine what happens if operator overloads were instance methods and you tried to use an operator on a null object! Traditional C++ (as opposed to the CLI version) never faced this problem because the operators were always applied on stack objects; and if at all pointers were used, since pointers followed their own set of operational behavior - you never face a situation where an overloaded op-overload method is invoked on an invalid object.

Traditional C++ example

See below some C++ code that uses virtual operator overloads :-

class Base
{
public:
    Base(int x):x_value(x){}
    virtual bool operator ==(const Base& b)
    {
        return x_value == b.x_value;
    }
protected:
    int x_value;
};

class Derived : public Base
{
public:
    Derived(int x, int y): Base(x), y_value(y){}    
    virtual bool operator ==(const Base& b)
    {        
        Derived* pD = (Derived*)&b;
        return (pD->y_value == y_value) && (pD->x_value == x_value);
    }
private:
    int y_value;
};

int _tmain(int argc, _TCHAR* argv[])
{
    Base* b1 = new Derived(2,11);
    Base* b2 = new Derived(2,11);

    cout << (*b1==*b2) << endl;

    return 0;
}

When (*b1==*b2) is evaluated the == operator overload for class Derived is invoked which can be easily verified by trying out different values for the Base constructors for b1 and b2 or by setting a breakpoint inside the operator overload.

A port to C#

Taking what we know of C# let's attempt a straight-forward port of the above example to C#.

public class Base
{
    int x;

    public Base( int x )
    {
        this.x = x;
    }

    public static bool operator==( Base l, Base r )
    {
        if( object.ReferenceEquals( l, r ) )
            return true;
        else if( object.ReferenceEquals( l, null ) || 
                 object.ReferenceEquals( r, null ) )
            return false;
            
        return l.x == r.x;
    }

    public static bool operator!=( Base l, Base r )
    {
        return !(l == r);
    }

    public int X { get { return x; } }
}

public class Derived : Base
{
    int y;

    public Derived( int x, int y ) : base( x )
    {
        this.y = y;
    }

    public static bool operator==( Derived l, Derived r )
    {
        if( object.ReferenceEquals( l, r ) )
            return true;
        else if( object.ReferenceEquals( l, null ) || 
                 object.ReferenceEquals( r, null ) )
            return false;
        
        return (l.y == r.y) && (l.X == r.X);
    }

    public static bool operator!=( Derived l, Derived r )
    {
        return !(l == r);
    }

    public int Y { get { return y; } }
}

class Program
{
    static void Main()
    {
        Derived d1 = new Derived( 2, 11 );
        Derived d2 = new Derived( 2, 11 );

        Console.WriteLine( d1 == d2 );
        Console.ReadLine();
    }
}

If we run the program as it is above everything will work like the C++ version, but if we introduce a slight change to the program things begin to deviate greatly.

class Program
{
    static void Main()
    {
        Base d1 = new Derived( 2, 11 );
        Base d2 = new Derived( 2, 12 );

        Console.WriteLine( d1 == d2 );
        Console.ReadLine();
    }
}

What's going on here? As simple debugging will show us the despite the objects being compared being instances of the Derived class the Base class == operator is being called. This is because C# (and thus, most other languages) figure out which == operator method to call based on the known (i.e. compile-time) type of the object on the left hand side of the operation.

There are ways around this as you'll see below.

Simulating operator polymorphism with C#

Here, we see how to simulate this in C# :-

class Base
{
    protected int x_value = 0;    

    public Base(int x)
    {
        x_value = x;
    }

    public static bool operator==(Base b1, Base b2)
    {
        if( object.ReferenceEquals( b1, b2 ) )
        {
            return true;
        }
        else if( object.ReferenceEquals( b1, null ) || 
                 object.ReferenceEquals( b2, null ) )
        {
            return false;
        }
        
        return b1.Equals(b2);            
    }

    public static bool operator !=(Base b1, Base b2)
    {
        return !(b1 == b2);
    }

    public override bool Equals(object obj)
    {
        if( obj == null )
            return false;
        
        Base o = obj as Base;
        
        if( o != null )    
            return x_value == o.x_value;
        return false;
    }


    public override int GetHashCode()
    {
        return x_value.GetHashCode();
    }
}

class Derived : Base
{
    protected int y_value = 0;
    

    public Derived(int x, int y) : base(x)
    {
        y_value = y;
    }

    public override bool Equals(object obj)
    {
        if( !base.Equals( obj ) )
          return false; 
          
        Derived o = obj as Derived;
        
        if( o == null )
            return false;
        
        return y_value == o.y_value;
    }

    public override int GetHashCode()
    {
        return x_value.GetHashCode() ^ y_value.GetHashCode() + x_value;
    }
}

class Program
{
    static void Main(string[] args)
    {
        Base b1 = new Derived(10, 12);
        Base b2 = new Derived(10, 11);
        
        Console.WriteLine(b1 == b2);

        b2 = null;

        Console.WriteLine(b1 == b2);

        Console.ReadKey(true);
    }
}

Rather than rely on the == operator overload to do all of the heavy lifting we push all of the work onto the virtual Equals method, from there we let polymorphism work its magic.

Points to note

  • The operator overload has to be static, so we have virtual instance methods that implement the logic for us and we invoke these virtual methods from the static operators

  • In our example, the method Equals corresponds to == and Equals is a virtual method (inherited from System.Object)

  • Within the static overload we need to check for null (to avoid null-reference exceptions)

  • Within each derived class's corresponding operator-logic method (Equals in our case), we cast the System.Object argument to the type of the class (e.g. - In the Derived class we cast to Derived)

  • While Equals and == already exist in System.Object, we can implement similar methods for any operator in our class hierarchies. Say, we need to implement the ++ operator, we then add a PlusPlus (or Increment) virtual method to the root base class in our object hierarchy and in the ++ overload we invoke the Increment method on the passed-in object.

  • In our example, we check for null and return true or false depending on whether the objects are both null or not. But, if you are implementing an operator like ++ then you might want to check for null and throw an ArgumentNullException (to override the inappropriate NullReferenceException that'd otherwise get thrown).

History

  • Apr 19, 2005 : Article first published
  • Apr 20, 2005 : Fixed a bug in Derived.Equals and also changed the GetHashCode implementations for Base and Derived (thanks to Jeffrey Sax for pointing this out)

posted on 2006-03-14 12:35 夢在天涯 閱讀(2003) 評論(0)  編輯 收藏 引用 所屬分類: C#/.NET

公告

EMail:itech001#126.com

導航

統計

  • 隨筆 - 461
  • 文章 - 4
  • 評論 - 746
  • 引用 - 0

常用鏈接

隨筆分類

隨筆檔案

收藏夾

Blogs

c#(csharp)

C++(cpp)

Enlish

Forums(bbs)

My self

Often go

Useful Webs

Xml/Uml/html

搜索

  •  

積分與排名

  • 積分 - 1816419
  • 排名 - 5

最新評論

閱讀排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
      <noscript id="pjuwb"></noscript>
            <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
              <dd id="pjuwb"></dd>
              <abbr id="pjuwb"></abbr>
              欧美一区二区国产| 久久精品国产免费看久久精品| 欧美精品高清视频| 在线观看亚洲精品视频| 久久综合色天天久久综合图片| 亚洲天堂成人| 国产视频久久久久| 久久资源av| 老色鬼精品视频在线观看播放| 91久久黄色| 99视频一区二区| 国产欧美韩国高清| 免费不卡中文字幕视频| 欧美成人精品在线观看| 99精品国产热久久91蜜凸| 日韩亚洲欧美一区| 国产精品性做久久久久久| 久久人人97超碰人人澡爱香蕉| 久久久久久久久久久久久女国产乱| 永久免费视频成人| 亚洲国产精品日韩| 国产精品成人国产乱一区| 亚洲综合国产| 国产欧美视频一区二区| 久久香蕉国产线看观看网| 免费国产一区二区| 欧美一级视频免费在线观看| 久久超碰97人人做人人爱| 亚洲破处大片| 亚洲欧美日韩一区二区在线| 亚洲国产精品va在线看黑人动漫 | 亚洲视频第一页| 欧美亚洲日本网站| 日韩一级裸体免费视频| 欧美一区二区三区四区夜夜大片 | 依依成人综合视频| 在线中文字幕日韩| 亚洲激情视频在线观看| 亚洲欧美日韩精品久久奇米色影视| 在线电影国产精品| 亚洲一区二区三区中文字幕| 亚洲国产天堂久久综合网| 午夜精品免费视频| 一区二区三区www| 美女脱光内衣内裤视频久久网站| 亚洲欧美日韩综合aⅴ视频| 欧美h视频在线| 久久综合中文色婷婷| 国产精品裸体一区二区三区| 欧美高清在线| 国产综合精品| 欧美一级成年大片在线观看| 亚洲午夜国产一区99re久久| 欧美大片国产精品| 欧美激情精品久久久久| 国外成人性视频| 午夜精品久久久久99热蜜桃导演| 亚洲小少妇裸体bbw| 免费一级欧美片在线播放| 久久精品五月| 国产亚洲观看| 亚洲欧美偷拍卡通变态| 亚洲一区三区视频在线观看| 欧美精品免费在线| 亚洲国产精品久久久久婷婷老年| 经典三级久久| 久久久久综合| 欧美xx69| 91久久精品美女高潮| 久久久一区二区三区| 另类尿喷潮videofree| 国内在线观看一区二区三区| 久久成人国产精品| 玖玖视频精品| 亚洲欧洲另类| 欧美片第1页综合| 这里只有精品电影| 久久av免费一区| 韩国福利一区| 美女图片一区二区| 亚洲区一区二区三区| 日韩视频一区二区三区在线播放免费观看 | 国语自产在线不卡| 久久精品一区蜜桃臀影院| 久久综合一区二区| 亚洲国产精品欧美一二99| 免费成人激情视频| 亚洲免费观看高清完整版在线观看熊| 一区二区日本视频| 国产精品日韩在线一区| 久久爱www.| 亚洲国产毛片完整版| 一区二区激情视频| 国产精品青草久久久久福利99| 亚洲在线观看| 蜜臀a∨国产成人精品| 亚洲风情亚aⅴ在线发布| 欧美激情视频在线免费观看 欧美视频免费一| 亚洲高清在线观看一区| 亚洲在线电影| 国产精品主播| 欧美gay视频激情| 亚洲影院色无极综合| 欧美不卡三区| 亚洲摸下面视频| 在线观看91精品国产入口| 欧美日韩国产天堂| 久久精品国产亚洲a| 日韩视频在线永久播放| 久久嫩草精品久久久精品| 亚洲激情网站免费观看| 国产精品国产三级国产专播精品人 | 久久久久久久久久久久久久一区 | 亚洲电影下载| 国产精品综合av一区二区国产馆| 久久网站免费| 亚洲女性裸体视频| 亚洲精品乱码久久久久久久久| 先锋影院在线亚洲| 日韩视频中午一区| 在线成人激情| 国产精品视频yy9099| 欧美黄色aa电影| 久久精品首页| 午夜精品在线视频| 在线视频欧美日韩| 亚洲大片一区二区三区| 久久久综合激的五月天| 亚洲欧美日韩区| 亚洲午夜精品一区二区| 亚洲精品久久久久久下一站| 国产午夜精品美女视频明星a级| 欧美大片在线影院| 美女日韩在线中文字幕| 久久不射中文字幕| 欧美在线免费视屏| 午夜亚洲精品| 亚洲婷婷综合久久一本伊一区| 亚洲国产精品第一区二区| 免费日韩av片| 欧美成人精品| 欧美激情片在线观看| 免费成人黄色片| 麻豆精品视频| 欧美黑人国产人伦爽爽爽| 蜜桃av一区| 欧美电影免费观看高清完整版| 猛干欧美女孩| 欧美国产日本高清在线| 亚洲国产精品黑人久久久| 欧美激情亚洲另类| 亚洲欧洲精品一区二区| 亚洲精品免费观看| 99re热这里只有精品视频| 99在线精品视频在线观看| 一道本一区二区| 99re8这里有精品热视频免费| 99re6热只有精品免费观看| 亚洲精品一区二区三区不| 一区二区三区久久| 午夜日韩电影| 另类亚洲自拍| 欧美日韩免费在线观看| 国产精品―色哟哟| 尤物精品在线| 亚洲毛片在线看| 亚洲综合第一| 你懂的成人av| 99精品欧美一区二区三区综合在线 | 99国产精品久久久久久久| 亚洲美女视频| 性久久久久久久久久久久| 久久夜色精品国产| 欧美国产日韩一区二区三区| 欧美激情视频在线免费观看 欧美视频免费一 | 国产欧美精品| 亚洲欧洲另类| 午夜视频一区在线观看| 米奇777超碰欧美日韩亚洲| 亚洲欧洲日产国产网站| 亚洲综合99| 老司机精品导航| 国产精品大片wwwwww| 激情六月婷婷久久| 亚洲一本视频| 欧美freesex交免费视频| 一区二区三区产品免费精品久久75| 亚洲欧美变态国产另类| 欧美成在线观看| 国产日韩欧美在线观看| 亚洲精品国产无天堂网2021| 亚洲欧美日韩在线| 亚洲国产精品久久久久秋霞蜜臀| 亚洲香蕉网站| 蜜臀va亚洲va欧美va天堂| 国产日韩一级二级三级| 一区二区三区国产精华| 欧美夫妇交换俱乐部在线观看| 亚洲一区二区网站| 欧美日韩一区国产|