مقدمهای بر عملگر Walrus در Python
۲ تیر ۱۴۰۰
عملگر Walrus با سینتکس NAME := expr
یک Assignment Expression یا عبارتی برای مقداردهی است که در Python 3.8 معرفی شده و توانسته توجه بسیاری از توسعهدهندگان را به خود جلب کند. طبق PEP 572، کدنویسی کمتر و پیشگیری از تکرار کدها که درنهایت باعث سریعتر شدن برنامه Python میشوند، منطق اصلی پشت Assignment Expression است بنابراین در ادامهی مقاله با ما همراه باشید تا موارد استفاده و کاربردهای این عملگر جدید را با جزئیات بیشتری بررسی کنیم.
موارد استفاده از عملگر Walrus
جلوگیری از تکرار کدها
با استفاده از عملگر Walrus، بسیار راحتتر از قبل میتوانید به اصل DRY پایبند باشید و دفعات تکرار کدها را کاهش دهید. فرض کنید میخواهید برنامهای بنویسید که در آن طول لیست my_long_list
بررسی شده و درصورتی که لیست طولانی بود، یک خطا چاپ شود:
my_long_list = list(range(1000))
# You get the length twice!
if len(my_long_list) > 10:
print(f"List is too long to consume (length={len(my_long_list)}, max=10)")
ممکن است در نگاه اول همه چیز خوب بهنظر برسد اما اگر دقت کرده باشید، len(my_long_list)
دو بار تکرار شده و درصورتی که my_long_list
یک لیست طولانی باشد، شاهد عملکرد کند برنامه خواهیم بود. حال میتوان با استفاده از عملگر Walrus بهراحتی از تکرار len(my_long_list)
جلوگیری کرد:
my_long_list = list(range(1000))
# Much better :)
if (count := len(my_long_list)) > 10:
print(f"List is too long to consume (length={count}, max=10)")
خواناتر شدن کدها
یکی از الگوهای معمول در برنامهنویسی، مقداردهی نتیجه به یک متغیر و بررسی نتیجه است:
result = parse_field_from(my_data)
if result:
print("Success")
در چنین مواردی میتوان این نوع بلوکهای کد را با استفاده از عملگر Walrus به کدهای تمیزتری تبدیل کرد:
if result := parse_field_from(my_data):
print("Success")
همچنین بهکمک عملگر Walrus میتوان پیچیدگیهای statementهای تودرتو را کاهش داد. فرض کنید که یک دیکشنری از اطلاعات دانشآموزهای مدرسهای در اختیار شما قرار داده شده و باید شناسه یا تاریخ فارغالتحصیلی هر دانشآموز را درصورت وجود، چاپ کنید:
sample_data = [
{"student_id": 200, "name": "Sally West", "graduation_date": "2019-05-01"},
{"student_id": 404, "name": "Zahara Durham", "graduation_date": None},
{"student_id": 555, "name": "Connie Coles", "graduation_date": "2020-01-15"},
{"student_id": None, "name": "Jared Hampton", "graduation_date": None},
]
for student in sample_data:
graduation_date = student["graduation_date"]
if graduation_date:
print(f'{student["name"]} graduated on {graduation_date}')
else:
# This nesting can be confusing!
student_id = student["student_id"]
if student_id:
print(
f'{student["name"]} is currently enrolled with ID {student_id}')
else:
print(f'{student["name"]} has no data')
حال statementهای تودرتو فوق را میتوان بهکمک عملگر Walrus به یک زنجیره از if
، elif
و else
تبدیل کرد:
sample_data = [
{"student_id": 200, "name": "Sally West", "graduation_date": "2019-05-01"},
{"student_id": 404, "name": "Zahara Durham", "graduation_date": None},
{"student_id": 555, "name": "Connie Coles", "graduation_date": "2020-01-15"},
{"student_id": None, "name": "Jared Hampton", "graduation_date": None},
]
for student in sample_data:
# Much cleaner
if graduation_date := student["graduation_date"]:
print(f'{student["name"]} graduated on {graduation_date}')
elif student_id := student["student_id"]:
print(f'{student["name"]} is currently enrolled with ID {student_id}')
else:
print(f'{student["name"]} has no data')
استفاده مجدد از متغیرها
استفاده از Regex در Python یکی از متداولترین مواردی است که هر توسعهدهنده با آن سروکار دارد. برای مثال فرض کنید باید کد منطقهی یک لیست از شمارههای تلفن را با استفاده از Regex تشخیص داده و آنها را چاپ کنید:
import re
phone_numbers = [
"(317) 555-5555",
"431-2973",
"(111) 222-3344",
"(710) 982-3811",
"290-2918",
"711-7712",
]
for number in phone_numbers:
# The regular expression "\(([0-9]{3})\)" checks for a substring
# with the pattern "(###)", where # is a 0-9 digit
if match := re.match("\(([0-9]{3})\)", number):
print(f"Area code: {match.group(1)}")
else:
print("No area code")
زمانی که میخواهید یک فایل را بخوانید
فرض کنید که میخواهید یک خط از فایلی را خوانده و در حلقهی while
، تغییراتی بر روی آن اعمال کنید:
f = open("walrus.txt", "r")
line = f.readline()
while line:
do_something(line)
line = f.readline()
همانطور که در کدهای بالا مشاهده میکنید، f.readline()
دو بار تکرار شده اما بهکمک عملگر Walrus میتوان کدهای فوق را بهصورت زیر خلاصه کرد:
f = open("walrus.txt", "r")
while line := f.readline():
do_something(line)
مواردی که استفاده از عملگر Walrus توصیه نمیشود
استفاده از Assignment expression در برخی موارد خاص توصیه نمیشود زیرا ممکن است باعث ایجاد ابهام شوند:
y := f(x) # INVALID
(y := f(x)) # Valid, though not recommended
y0 = y1 := f(x) # INVALID
y0 = (y1 := f(x)) # Valid, though discouraged
foo(x = y := f(x)) # INVALID
foo(x=(y := f(x))) # Valid, though probably confusing
توسعهدهندگان دربارهی ما چه میگویند
تجربه کار باliara_cloud@امروز خیلی خوب بود. یکی از سرویس هام رو منتقل کردم روش و راضیم. انقد سریع و جذاب کارم راه افتادم اصن باورم نمیشد! برعکس سرویس های PaaS دیگه با اون همه پیچیدگیشون. دمتون گرم
...
MohammadReza
keikaavousi
بعد از بسته شدن @fandoghpaas و ناراحتی همهمون از اینکه یه سرویس خوب و صادق نمیتونه از پس هزینهها بر بیاد، سرویسم رو منتقل کردم به پاس لیارا (https://liara.ir @liara_cloud) . تجربه راحت و خوب. تفاوتهایی داشت که کمی کار میخواست ولی تا الان کاملا راضی.
jadi
jadi
یه خسته نباشید باید به تصمیمliara_cloud@بگم،
بعد از چندین روز سرکله زدن با سرویس های مشابه بالاخره تصمیم گرفتم لیارا رو امتحان کنم و باور نمیشه ۱۰ دقیقه بیشتر وقت نبرد،
دمتون گرم.
Arch
EbadiDev
واسه سرویس PaaS با اختلاف لیارا بهترین رابط کاربری داره و یکی از مزیتهای سرویس دیتابیسشون اینه که خودشون به صورت دورهای بکآپ میگیرن.
...
Ali Najafi
me_ali_najafi
یکی از کارهای خوبی که جدیداً میکنم اینه که یه دیتابیس روی لیارا میسازم و به پروژه وصل میکنم اینطوری هم خونه و هم محل کار دیتابیس بروز رو دارم و راحت میتونم ادامه بدم کار روliara_cloud@
Navid
1navid
عاشقliara_cloud@شدم درسته در حد AWS نیست ولی خب تجربه خوبی واسه پروژه های داخل ایران ارائه میده، میتونم رو CD هم اجراش کنم
Amir H Shekari
vanenshi