زنگ سی شارپ – قسمت سی و چهارم

بررسی operator overloading و جزئیات آن


۴ دیدگاه سی شارپ یکشنبه, ۹ام تیر , ۱۳۹۲ 8235 بازدید

زنگ سی‌شارپ - قسمت سی و چهارم

overload کردن True و False

کلمات کلیدی true و false نیز می‌توانند به‌عنوان unary operators به‌منظور overload کردن مورد استفاده قرار گیرند. نسخه‌ی overload شده‌ی این operator ها با توجه به کلاسی که شما می‌سازید، شخصی‌سازی می‌شود. هنگامی‌که true و false برای یک کلاس overload می‌شوند، می‌توانید از اشیای آن کلاس برای کنترل کردن if، for، while و do-while و همچنین ? استفاده کنید.

Operator های true و false باید باهم overload شوند و نمی‌توانید فقط یکی از آن‌ها را overload کنید. هر دوی آن‌ها unary operator هستند و فرم کلی آن‌ها به‌صورت زیر است:

public static bool operator true(param-type operand)
{
    // return true or false
}
public static bool operator false(param-type operand)
{
    // return true or false
}

دقت کنید که هریک مقدار bool را return می‌کند.

مثال زیر نشان می‌دهد که چگونه true و false می‌توانند در کلاس TwoD اعمال شوند. فرض بر این است که اگر حداقل یکی از فیلدها غیر صفر باشد، شیء true است و اگر همه‌ی فیلدها صفر باشند، شیء false است:

using System;
class TwoD
{
    int X, Y;
    public TwoD()
    {
        X = Y = 0;
    }
    public TwoD(int a, int b)
    {
        X = a;
        Y = b;
    }
    public static bool operator true(TwoD op)
    {
        return (op.X != 0 || op.Y != 0);
    }
    public static bool operator false(TwoD op)
    { 
        return (op.X == 0 && op.Y == 0);
    }
    public static TwoD operator --(TwoD op)
    {
        TwoD result = new TwoD();
        result.X = op.X - 1;
        result.Y = op.Y - 1;
        return result;
    }
    public static TwoD operator ++(TwoD op)
    {
        TwoD result = new TwoD();
        result.X = op.X + 1;
        result.Y = op.Y + 1;
        return result;
    }
    public void Show()
    {
        Console.WriteLine("{0}, {1}", X, Y);
    }
}
class OpOverloadingDemo
{
    static void Main()
    {
        TwoD ob = new TwoD(5, 5);

        if (ob)
            Console.WriteLine("ob is true");
        Console.WriteLine();
        
        for (; ob; ob--)
        {
            ob.Show();
        }

        Console.WriteLine();
        ob = new TwoD(-3, -3);

        while (ob)
        {
            ob.Show();
            ob++;
        }
    }
}

دقت کنید که چگونه از شیء TwoD برای کنترل if و while و for استفاده شده است. به‌عنوان مثال، در مورد if، شیء TwoD توسط true ارزیابی می‌شود. اگر حداقل یکی از فیلدهای این شیء مخالف صفر باشد، دستور if اجرا خواهد شد. در مورد حلقه‌ها، حلقه تا زمانی اجرا می‌شود که شیء true باشد (فیلدهای شیء مخالف صفر باشد) و به‌محض این‌که شیء false شد (همه‌ی فیلدها صفر شدند) برنامه از حلقه خارج می‌شود.

overload کردن عملگرهای منطقی

همان‌طور که می‌دانید، سی‌شارپ شامل عملگرهای منطقی & ، | ، ! و && و || است. البته فقط & و | و ! می‌توانند overload شوند. با این حال با دنبال کردن یک سری قوانین می‌توانید از مزیت‌های && و || بهره ببرید.

اگر قصد استفاده از عملگرهای منطقی short-circuit را نداشته باشید، می‌توانید بسیار ساده & و | را overload کنید که هر کدام از آن‌ها یک مقدار bool را return می‌کند. overload کردن ! نیز مقدار bool را return می‌کند.

مثال زیر عملگرهای منطقی ! و & و | را overload می‌کند. مانند قبل فرض بر این است که شیء TwoD در صورتی true است که حداقل یکی از فیلدهای آن غیر صفر باشد و در صورتی‌که همه‌ی فیلدهای TwoD صفر باشند، شیء false است:

using System;
class TwoD
{
    int X, Y;
    public TwoD()
    {
        X = Y = 0;
    }
    public TwoD(int a, int b)
    {
        X = a;
        Y = b;
    }

    public static bool operator &(TwoD op1, TwoD op2)
    { 
        return ((op1.X != 0 && op1.Y != 0) && (op2.X != 0 && op2.Y != 0));
    }
    public static bool operator |(TwoD op1, TwoD op2)
    { 
        return ((op1.X != 0 || op1.Y != 0) || (op2.X != 0 || op2.Y != 0));
    }
    public static bool operator !(TwoD op)
    { 
        return (op.X == 0 & op.Y == 0);
    }

    public void Show()
    {
        Console.WriteLine("{0}, {1}", X, Y);
    }
}
class OpOverloadingDemo
{
    static void Main()
    {
        TwoD a = new TwoD(1, 2);
        TwoD b = new TwoD(8, 8);
        TwoD c = new TwoD();

        Console.Write("Here is a: ");
        a.Show();
        Console.Write("Here is b: ");
        b.Show();
        Console.Write("Here is c: ");
        c.Show();
        Console.WriteLine();

        if(!a)
            Console.WriteLine("a is false");
        if(!b)
            Console.WriteLine("b is false");
        if(!c)
            Console.WriteLine("c is false");

        Console.WriteLine();

        if (a & b)
            Console.WriteLine("a & b is true");
        else
            Console.WriteLine("a & b is false");
        if(a & c)
            Console.WriteLine("a & c is true");
        else
            Console.WriteLine("a & c is false");

        Console.WriteLine();

        if(a | b)
            Console.WriteLine("a | b is true");
        else
            Console.WriteLine("a | b i false");
        if(a | c)
            Console.WriteLine("a | c is true");
        else
            Console.WriteLine("a | c is false");
    }
}

خروجی:

در این روش، operator method های & و | و ! هرکدام یک مقدار bool را return می‌کنند. به‌دلیل این‌که از این operator ها در حالت استاندارد خودشان استفاده شود، return کردن مقدار bool ضروری است. روشی که در مثال قبل مشاهده کردید، فقط برای زمانی است که به short-circuit نیاز ندارید.

فعال کردن operator های short-circuit

برای فعال کردن && و || باید از چهار قاعده پیروی کنید. یک، کلاس مورد نظر باید & و | را overload کند. دو، return type متدهای overload شده‌ی & و | باید از نوع همان کلاسی باشد که این operator ها در آن overload می‌شوند. سه، هر پارامتر باید به یک شیء از جنس همان کلاسی که operator در آن overload شده است، رجوع کند. چهار، operator های true و false باید برای کلاس مربوطه overload شود.

برنامه زیر نشان می‌دهد که چگونه & و | را برای کلاس TwoD اجرا کرده تا بتوان از مزیت && و || استفاده کنید:

using System;
class TwoD
{
    int X, Y;
    public TwoD()
    {
        X = Y = 0;
    }
    public TwoD(int a, int b)
    {
        X = a;
        Y = b;
    }

    // Overload & for short-circuit evaluation.
    public static TwoD operator &(TwoD op1, TwoD op2)
    {
        if ((op1.X != 0 && op1.Y != 0) && (op2.X != 0 && op2.Y != 0))
            return new TwoD(1, 1);
        else
            return new TwoD();
    }
    // Overload | for short-circuit evaluation. 
    public static TwoD operator |(TwoD op1, TwoD op2)
    {
        if ((op1.X != 0 || op1.Y != 0) || (op2.X != 0 || op2.Y != 0))
            return new TwoD(1, 1);
        else
            return new TwoD();
    }

    // Overload true.
    public static bool operator true(TwoD op)
    {
        return (op.X != 0 || op.Y != 0); // at least one field is non-zero
    }
    // Overload false.
    public static bool operator false(TwoD op)
    {
        return (op.X == 0 & op.Y == 0); // all fields are zero
    }

    // Overload !.
    public static bool operator !(TwoD op)
    {
        return (op.X == 0 & op.Y == 0);
    }

    public void Show()
    {
        Console.WriteLine("{0}, {1}", X, Y);
    }
}
class OpOverloadingDemo
{
    static void Main()
    {
        TwoD a = new TwoD(5, 6);
        TwoD b = new TwoD(10, 10);
        TwoD c = new TwoD();

        Console.Write("Here is a: ");
        a.Show();
        Console.Write("Here is b: ");
        b.Show();
        Console.Write("Here is c: ");
        c.Show();
        Console.WriteLine();

        if (!a)
            Console.WriteLine("a is false");
        if (!b)
            Console.WriteLine("b is false");
        if (!c)
            Console.WriteLine("c is false");

        Console.WriteLine();

        if (a & b)
            Console.WriteLine("a & b is true");
        else
            Console.WriteLine("a & b is false");
        if (a & c)
            Console.WriteLine("a & c is true");
        else
            Console.WriteLine("a & c is false");

        Console.WriteLine();

        if (a | b)
            Console.WriteLine("a | b is true");
        else
            Console.WriteLine("a | b i false");
        if (a | c)
            Console.WriteLine("a | c is true");
        else
            Console.WriteLine("a | c is false");

        Console.WriteLine();

        // Now use short-circuit ops.
        Console.WriteLine("Use short-circuit && and ||");
        if (a && b) 
            Console.WriteLine("a && b is true.");
        else 
            Console.WriteLine("a && b is false.");

        if (a && c) 
            Console.WriteLine("a && c is true.");
        else 
            Console.WriteLine("a && c is false.");

        if (a || b)
            Console.WriteLine("a || b is true.");
        else
            Console.WriteLine("a || b is false.");

        if (a || c)
            Console.WriteLine("a || c is true.");
        else
            Console.WriteLine("a || c is false.");
    }
}

خروجی:

در این‌جا نگاهی دقیق‌تر به برنامه‌ی بالا می‌اندازیم و می‌بینیم که چطور & و | اجرا شده‌اند. به این قسمت از کد بالا توجه کنید:

// Overload & for short-circuit evaluation.
public static TwoD operator &(TwoD op1, TwoD op2)
{
    if ((op1.X != 0 && op1.Y != 0) && (op2.X != 0 && op2.Y != 0))
        return new TwoD(1, 1);
    else
        return new TwoD(); 
}
// Overload | for short-circuit evaluation. 
public static TwoD operator |(TwoD op1, TwoD op2)
{
    if ((op1.X != 0 || op1.Y != 0) || (op2.X != 0 || op2.Y != 0))
        return new TwoD(1, 1);
    else
        return new TwoD();
}

دقت کنید که هر دو یک شیء از جنس TwoD را return می‌کنند. توجه داشته باشید که چگونه این شیء به‌وجود آمده است. اگر خروجی عملیاتی که انجام شده true باشد، بنابراین یک شیء TwoD که true است (شیء‌ای که حداقل یکی از فیلدهای آن غیر صفر است) return می‌شود. اگر خروجی false باشد، یک شیء false ساخته شده و سپس return می‌شود.

بنابراین در عباراتی به شکل زیر:

if (a & b)
        Console.WriteLine("a & b is true");
else
        Console.WriteLine("a & b is false");

خروجی a & b یک شیء TwoD است که در این مورد یک شیء true است. از آن‌جا که operator های true و false  در این کلاس تعریف شده‌اند، شیء‍‌ای که return شده است مربوط به true operator بوده و در نهایت یک مقدار بولین return می‌شود که در این‌جا این شیء true است و در نتیجه مقدار true نیز return می‌شود و محتویات دستور if اجرا خواهد شد.

به‌دلیل این‌که قوانین مورد نیاز رعایت شده‌اند، operator های short-circuit‌ برای استفاده از اشیای TwoD نیز فعال هستند و به‌این صورت کار می‌کند که، اولین operand توسط operator true (برای ||) یا operator false (برای &&) بررسی می‌شود. اگر خروجی عملیات قابل تشخیص باشد، بنابراین & یا | مربوطه مورد ارزیابی قرار نمی‌گیرد. در غیر این‌صورت، از & یا | مربوطه برای یافتن خروجی استفاده می‌شود. بنابراین، استفاده از && یا || موجب می‌شود & یا | مربوطه تنها زمانی فراخوانی شود که operand اول نتواند خروجی را مشخص کند. برای نمونه، به کد زیر توجه کنید:

if (a || c) Console.WriteLine("a || c is true.");

در ابتدا true operator روی a اعمال می‌شود و از آن‌جا که در این مورد a برابر با true است، دیگر نیازی به استفاده از | operator نیست. اما در کد زیر:

if (c || a) Console.WriteLine(“a || c is true.”);

ابتدا true operator‌ روی c اعمال می‌شود که در این مورد c برابر با false است. بنابراین، | operator فراخوانی خواهد شد تا بررسی شود که آیا a برابر با true است یا خیر (که در این مورد، a برابر با true است). شاید در ابتدا فکر کنید که تکنیک استفاده شده برای فعال کردن short-circuit مقداری پیچیده و دشوار باشد اما اگر اندکی در مورد آن فکر کنید متوجه خواهید شد که کارهای انجام شده منطقی به‌نظر می‌رسد. در حالت کلی، بهتر است از همین روش استفاده کنید.

Conversion Operators

در برخی از مواقع، می‌خواهید از شیء یک کلاس در عبارتی استفاده کنید که شامل data type های دیگری نیز است. در بعضی موارد، overload کردن یک یا چند operator می‌تواند این کار را برای شما انجام دهد اما گاهی چیزی که شما نیاز دارید یک تبدیل ساده از نوع کلاس به نوع مورد نظرتان است. برای انجام این دسته از موارد، سی‌شارپ به شما اجازه می‌دهد نوع خاصی از operator method را بسازید که conversion operator نامیده می‌شود. Conversion operator‌ یک شیء از کلاس شما را به نوع دیگری که مد نظرتان است تبدیل می‌کند.

دو حالت از conversion operator موجود است: implicit و explicit که فرم کلی آن‌ها به شکل زیر است:

public static operator implicit target-type(source-type v) { return value; }
public static operator explicit target-type(source-type v) { return value; }

در این‌جا، target-type مشخص کننده‌ی نوعی است که قصد دارید source-type را به آن تبدیل کنید و value مقدار کلاس، بعد از تبدیل است. Conversion operator اطلاعات را مطابق با target-type باز می‌گرداند (return می‌کند).

اگر conversion operator به‌طور implicit مشخص شود، بنابراین conversion به‌صورت اتوماتیک انجام خواهد شد، مثل حالتی که شیء در یک عبارت همراه با یک data type دیگری از نوع target-type در تعامل است. هنگامی‌که conversion به‌صورت explicit تعریف شده باشد، بنابراین هنگامی که cast مورد نیاز است conversion فراخوانی می‌شود. توجه کنید که نمی‌توانید برای یک source-type و target-type هم implicit و explicit را تعریف کنید.

برای نشان دادن conversion operator آن را در کلاس TwoD اعمال می‌کنیم. با فرض اینکه می‌خواهید یک شیء TwoD را به یک int تبدیل کنید و در یک عبارت از جنس int از آن استفاده نمایید:

using System;
class TwoD
{
    int X, Y;
    public TwoD()
    {
        X = Y = 0;
    }
    public TwoD(int a, int b)
    {
        X = a;
        Y = b;
    }
    public static implicit operator int(TwoD op)
    {
        return op.X * op.Y;
    }
}
class OpOvDemo
{
    static void Main()
    {
        TwoD ob1 = new TwoD(2, 2);
        TwoD ob2 = new TwoD(1, 1);

        int i = ob1 * 3;
        Console.WriteLine(i);
        i = ob1 - 3;
        Console.WriteLine(i);
        i = ob1 + ob2;
        Console.WriteLine(i);
        i = ob1;
        Console.WriteLine(i);
    }
}

همان‌طور که برنامه بالا نشان می‌دهد، هنگامی‌که از شیء TwoD در یک عبارت integer (مثل i = ob1 – ۳) استفاده شده عملیات convert به‌طور اتوماتیک اعمال می‌شود.

ادامه‌ی مبحث conversion operators را در قسمت بعد دنبال کنید.



نویسنده / مترجم : مسعود درویشیان

علاقه مند به موسیقی و برنامه نویسی بازی


۴ دیدگاه برای این نوشته ثبت شده است


  1. داوود
    ۱۷ تیر ۱۳۹۲

    سلام مسعود جان؛ وقتت بخیر
    من برگشتم و می بینم که دو تا درس جدید گذاشتی عزیزم.
    اگه میشه یکم کندتر کن روند گذاشتن درس رو تا ما هم بهتون برسیم. البته اگه بقیه دوستان موافقند.
    ==
    مسعود جان یک سوال خارج از درس هم داشتم؛ میشه یه توضیح کوچیک راجع به API بدی؟ و اینکه با داشتن API مثلا یک سرویس SMS میشه چه کارهایی رو باهاش انجام داد؟
    آیا ما بعد از مطالعه زنگ سی شارپ قادر به کار با API های آماده هستیم؟




    • مسعود درویشیان
      ۲۵ تیر ۱۳۹۲

      این لینک رو ببینید.
      اگه مشکلی بود بگید تا خودم براتون توضیح بدم.
      آره می‌تونید با API کار کنید، ولی باید با پلتفرمی که می‌خواهید کار کنید اول آشنایی داشته باشید.




      • نوید
        ۲۵ تیر ۱۳۹۲

        سلام؛
        ممنونم مسعود جان از اینکه پاسخ دادید. چشم نگاه می کنم و در صورتی که متوجه نشدم به زحمتتون میندازم.




  2. فرهاد
    ۷ مرداد ۱۳۹۴

    سلام و خسته نباشید
    میخواستم بدونم overload کردن عمل گرهای منطقی چه موقع به کار میان و چرا باید به این شکل استفاده شون کنیم آخه مثالی که برای overloadکردن عملگرها زدید زیاد پیچیده نیست ولی چه موقعی کاربرد داشته باشه خیلی مهمه و ما چه موقعی ازشون استفاده کنیم؟با تشکر



دیدگاه خود را بنویسید





نشانی ایمیل شما منتشر نخواهد شد.

کامنت های شما بعد از تأیید توسط نویسنده وبلاگ، منتشر خواهند شد.

لطفا دیدگاهتان تا حد امکان مربوط به پست بالا باشد. اگر حرف دیگری دارید و یا قصد تماس با من را دارید، از فرم تماس استفاده کنید.

شما میتوانید با مراجعه به سایت گراواتار یک آواتار اختصاصی برای خود تعریف کنید، تا در کنار نام شما نمایش داده شود

برای قرار دادن کدهای نمونه می توانید از تگ های [php] ، [html] ، [css] و [js] استفاده کنید.
به عنوان مثال کدهای php را می توان به صورت زیر قرار داد:
[php] var $whoLoveIranians = "WebTarget!"; [/php]



کلیه حقوق مادی و معنوی برای وب سایت وب تارگت محفوظ است ©2017 webtarget.ir

استفاده از مطالب وب سایت در سایر وب سایت‌ها و نشریات چاپی با ذکر منبع آزاد است.