سربارگذاری عملگرها در برنامه نویسی پایتون
سربارگذاری عملگرها در برنامه نویسی پایتون
در این درس از مجموعه آموزش برنامه نویسی سایت سورس باران، شما یاد خواهید گرفت که چگونه سربارگذاری عملگرها در برنامه نویسی پایتون (Python Operator Overloading) استفاده کنید
پیشنهاد ویژه : پکیج آموزش طراحی وب سایت با پایتون
سربارگذاری عملگرها در برنامه نویسی پایتون
اپراتورهای پایتون برای کلاس های توکار، کار می کنند. اما همان اپراتور با انواع مختلف رفتارهای متفاوتی دارد. به عنوان مثال، عملگر + جمع حساب را روی دو عدد انجام می دهد، دو لیست را ادغام می کند یا دو رشته را بهم می پیوندد.
به این ویژگی در پایتون که اجازه می دهد همان عملگر با توجه به زمینه معنای متفاوتی داشته باشد، سربارگذاری عملگرها در برنامه نویسی پایتون (Python Operator Overloading) نامیده می شود.
بنابراین چه اتفاقی می افتد که ما از آنها با اشیای یک کلاس تعریف شده توسط کاربر استفاده کنیم؟ بیایید کلاس زیر را در نظر بگیریم که سعی در شبیه سازی یک نقطه در سیستم مختصات 2-D دارد.
1 2 3 4 5 6 7 8 9 |
class Point: def __init__(self, x=0, y=0): self.x = x self.y = y p1 = Point(1, 2) p2 = Point(2, 3) print(p1+p2) |
خروجی
1 2 3 4 |
Traceback (most recent call last): File "<string>", line 9, in <module> print(p1+p2) TypeError: unsupported operand type(s) for +: 'Point' and 'Point' |
در اینجا، می توانیم ببینیم که یک TypeError مطرح شده است ، زیرا پایتون نمی دانست چگونه دو شی را به یکدیگر اضافه کند.
با این حال ، ما می توانیم از طریق سربارگذاری عملگرها به این کار در پایتون برسیم. اما در ابتدا، بیایید اطلاعاتی درباره توابع خاص بدست آوریم.
توابع ویژه پایتون
توابع کلاس را که با زیر خط دوتایی __ شروع می شوند، در پایتون توابع ویژه می نامند.
این توابع توابع معمولی نیستند که برای یک کلاس تعریف کنیم. تابع ()__init __ که در بالا تعریف کردیم یکی از آنهاست. هر وقت شی a جدیدی از آن کلاس ایجاد کنیم فراخوانی می شود.
توابع ویژه بیشمار دیگری نیز در پایتون وجود دارد.
با استفاده از توابع ویژه، می توانیم کلاس خود را با توابع توکار سازگار کنیم.
1 2 3 |
>>> p1 = Point(2,3) >>> print(p1) <__main__.Point object at 0x00000000031F8CC0> |
فرض کنید می خواهیم تابع ()print به جای آنچه که به دست آوردیم مختصات شی Point را چاپ کند. ما می توانیم در کلاس خود یک روش()__str __ تعریف کنیم که نحوه چاپ شی را کنترل کند. بیایید ببینیم چگونه می توانیم به این هدف برسیم:
1 2 3 4 5 6 7 |
class Point: def __init__(self, x = 0, y = 0): self.x = x self.y = y def __str__(self): return "({0},{1})".format(self.x,self.y) |
حالا بیایید دوباره تابع ()print را امتحان کنیم.
1 2 3 4 5 6 7 8 9 10 11 |
class Point: def __init__(self, x=0, y=0): self.x = x self.y = y def __str__(self): return "({0}, {1})".format(self.x, self.y) p1 = Point(2, 3) print(p1) |
خروجی
1 |
(2, 3) |
معلوم است که وقتی از تابع ()str یا ()format استفاده می کنیم، همین روش فراخوانی می شود.
1 2 3 4 5 |
>>> str(p1) '(2,3)' >>> format(p1) '(2,3)' |
1 |
' |
بنابراین ، وقتی از (str (p1 یا (format (p1 استفاده می کنید، پایتون به صورت داخلی روش ()p1 .__ str __ را فراخوانی می کند. از این رو نام آن، توابع ویژه است.
سربارگذاری عملگر + در پایتون
برای سربارگذاری عملگر + در برنامه نویسی پایتون، باید عملکرد() __add __ را در کلاس پیاده سازی کنیم. در داخل این تابع می توانیم هر کاری را که دوست داریم انجام دهیم. اما منطقی تر است که یک شی از مقدار مختصات را برگردانیم
1 2 3 4 5 6 7 8 9 10 11 12 |
class Point: def __init__(self, x=0, y=0): self.x = x self.y = y def __str__(self): return "({0},{1})".format(self.x, self.y) def __add__(self, other): x = self.x + other.x y = self.y + other.y return Point(x, y) |
حالا بیایید دوباره عمل جمع را امتحان کنیم:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
class Point: def __init__(self, x=0, y=0): self.x = x self.y = y def __str__(self): return "({0},{1})".format(self.x, self.y) def __add__(self, other): x = self.x + other.x y = self.y + other.y return Point(x, y) p1 = Point(1, 2) p2 = Point(2, 3) print(p1+p2) |
خروجی
1 |
(3,5) |
آنچه در واقع اتفاق می افتد این است که ، وقتی از p1 + p2 استفاده می کنید، پایتون p1 را فراخوانی می کند که به نوبه خود(Point.__add__(p1,p2) است. پس از این، عملیات جمع به روشی که مشخص کردیم انجام می شود.
به همین ترتیب، می توانیم عملگرهای دیگر را نیز سربارگیری کنیم. نابع ویژه ای که باید پیاده سازی کنیم در زیر آورده شده است.
عملگر | اصطلاح | داخلی |
---|---|---|
جمع | p1 + p2 |
(p1.__add__(p2 |
تفریق | p1 - p2 |
(p1.__sub__(p2 |
ضرب | p1 * p2 |
(p1.__mul__(p2 |
توان | p1 ** p2 |
(p1.__pow__(p2 |
تقسیم | p1 / p2 |
(p1.__truediv__(p2 |
جزء صحیح | p1 // p2 |
(p1.__floordiv__(p2 |
باقیمانده | p1 % p2 |
(p1.__mod__(p2 |
Bitwise Left Shift | p1 << p2 |
(p1.__lshift__(p2 |
Bitwise Right Shift | p1 >> p2 |
(p1.__rshift__(p2 |
Bitwise AND | p1 & p2 |
(p1.__and__(p2 |
Bitwise OR | p1 | p2 |
(p1.__or__(p2 |
Bitwise XOR | p1 ^ p2 |
(p1.__xor__(p2 |
Bitwise NOT | ~p1 |
(p1.__invert__() |
سربارگذاری عملگرها مقایسه
پایتون سربارگذاری عملگرها را فقط به عملگرهای حساب محدود نمی کند. ما می توانیم عملگرهای مقایسه را نیز اضافه بار دهیم.
فرض کنید می خواستیم نماد کمتر از نماد <را در کلاس Point خود پیاده کنیم.
اجازه دهید مقدار این نقاط را از مبدا مقایسه کنیم و نتیجه را برای این منظور برگردانیم.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
# overloading the less than operator class Point: def __init__(self, x=0, y=0): self.x = x self.y = y def __str__(self): return "({0},{1})".format(self.x, self.y) def __lt__(self, other): self_mag = (self.x ** 2) + (self.y ** 2) other_mag = (other.x ** 2) + (other.y ** 2) return self_mag < other_mag p1 = Point(1,1) p2 = Point(-2,-3) p3 = Point(1,-1) # use less than print(p1<p2) print(p2<p3) print(p1<p3) |
خروجی
1 2 3 |
True False False |
به همین ترتیب، توابع ویژه ای که برای سربارگذاری عملگر مقایسه، نیاز به پیاده سازی داریم، در زیر آورده شده است.
عملگر | اصطلاح | داخلی |
---|---|---|
کوچکتر | p1 < p2 |
(p1.__lt__(p2 |
کوچکتر مساوی | p1 <= p2 |
(p1.__le__(p2 |
برابر است یا | p1 == p2 |
(p1.__eq__(p2 |
برابر نیست | p1 != p2 |
(p1.__ne__(p2 |
بزرگتر | p1 > p2 |
(p1.__gt__(p2 |
بزرگتر مساوی | p1 >= p2 |
(p1.__ge__(p2 |
لیست جلسات قبل آموزش برنامه نویسی پایتون
- آموزش نصب و اجرای برنامه نویسی پایتون
- کلیدواژه ها و شناسه های برنامه نویسی پایتون
- دستورات، تورفتگی ها و کامنت ها در برنامه نویسی پایتون
- متغیرها، ثابت ها و لیترال ها در برنامه نویسی پایتون
- انواع داده ها در برنامه نویسی پایتون
- تبدیل نوع در برنامه نویسی پایتون
- ورودی، خروجی و وارد کردن در برنامه نویسی پایتون
- عملگرها در برنامه نویسی پایتون
- نام و دامنه در برنامه نویسی پایتون
- دستور شرطی if…else در برنامه نویسی پایتون
- حلقه for در برنامه نویسی پایتون
- حلقه while در برنامه نویسی پایتون
- دستورات break و continue در برنامه نویسی پایتون
- دستور pass در برنامه نویسی پایتون
- توابع در برنامه نویسی پایتون
- آرگومان تابع در برنامه نویسی پایتون
- تابع بازگشتی در برنامه نویسی پایتون
- تابع بی نام/ تابع لامبدا در برنامه نویسی پایتون
- متغیرهای سراسری، محلی و غیر محلی در برنامه نویسی پایتون
- کلیدواژه global در برنامه نویسی پایتون
- ماژول های برنامه نویسی پایتون
- پکیج ها در برنامه نویسی پایتون
- اعداد و تبدیل نوع داده در برنامه نویسی پایتون
- لیست در برنامه نویسی پایتون
- تاپل در برنامه نویسی پایتون
- رشته ها در برنامه نویسی پایتون
- مجموعه ها در برنامه نویسی پایتون
- دیکشنری در برنامه نویسی پایتون
- عملیات ورودی/خروجی در برنامه نویسی پایتون
- دایرکتوری و مدیریت فایل ها در برنامه نویسی پایتون
- خطاها و استثناهای توکار در برنامه نویسی پایتون
- مدیریت استثناها در برنامه نویسی پایتون
- استثناهای تعریف شده توسط کاربر در برنامه نویسی پایتون
- برنامه نویسی شی گرا در پایتون
- اشیا و کلاس ها در برنامه نویسی پایتون
- وراثت در برنامه نویسی پایتون
- وراثت چندگانه در برنامه نویسی پایتون
دیدگاه شما