۵ قابلیت پیشرفته پایتون و روش های استفاده از آن ها — راهنمای کاربردی
پایتون یک زبان زیبا است و در عین این که استفاده از آن ساده محسوب میشود ساختار کاملاً گویایی دارد. اما آیا با همه امکانات این زبان برنامهنویسی آشنا هستید؟ قابلیتهای پیشرفته هر زبان برنامهنویسی عموماً از طریق به کارگیری گسترده و طولانیمدت آن زبان به دست میآیند. برای نمونه، شاید تاکنون در زمان کدنویسی یک پروژه پیچیده مجبور به جستجوی چیزی در وبسایت stackoverflow شده باشید. در این موارد ممکن است به راهحل زیبایی برای مسئله خود دست یافته باشید که در آن از یک قابلیت پیشرفته پایتون استفاده شده است و شما نیز تا به حال هرگز آن را مورد استفاده قرار ندادهاید.
این روش کشف از طریق کاوش و تصادف، مسلماً یکی از جالبترین روشهای یادگیری محسوب میشود.
در این مقاله 5 قابلیت پیشرفته پایتون را معرفی میکنیم که کاملاً مفید هستند و مهمتر از آن این است که روش استفاده از آنها را نیز مورد بررسی قرار خواهیم داد.
1. تابعهای لامبدا
تابع «لامبدا» (Lambda) یک تابع ناشناس یا بینام کوچک است. منظور از بینام در اینجا آن است که عملاً هیچ نامی برای تابع تعیین نشده است.
تابعهای پایتون به طور معمول با سبک زیر تعریف میشوند:
1def a_function_name()
اما تابعهای لامبدا هیچ نامی نمیگیرند. دلیل این کار آن است که مقصود از تعریف تابعهای لامبدا، اجرای نوعی عبارت یا عملیات ساده است که نیازی به تعریف یک تابع کامل وجود ندارد.
تابع لامبدا میتواند هر چند آرگومان که لازم است بگیرد؛ اما در اغلب موارد تنها یک عبارت دارد:
1x = lambda a, b : a * b
2print(x(5, 6)) # prints '30'
3
4x = lambda a : a*3 + 3
5print(x(3)) # prints '12'
همان طور که میبینید تعریف این تابع بسیار آسان است. در کد فوق یک محاسبه ریاضی مقدماتی را بدون نیاز به تعریف کامل تابع اجرا کردهایم. این یکی از قابلیتهای فراوان پایتون است که موجب میشود بتوانیم به روشی تمیز و سادهسازی شده از این زبان برنامهنویسی بهره بگیریم.
2. نگاشت
()Map یک تابع توکار پایتون است که برای اِعمال تابعی روی یک دنباله از عناصر مانند لیست یا دیکشنری استفاده میشود. این تابع کاملاً سرراست است و مهمتر از آن، این است که چنین عملیاتی را به روشی «مطمئن» اجرا میکند.
1def square_it_func(a):
2 return a * a
3
4x = map(square_it_func, [1, 4, 7])
5print(x) # prints '[1, 16, 49]'
6
7def multiplier_func(a, b):
8 return a * b
9
10x = map(multiplier_func, [1, 4, 7], [2, 5, 8])
11print(x) # prints '[2, 20, 56]'
مثال فوق را بررسی کنید. ما میتوانیم تابع خود را روی یک لیست منفرد یا لیستهای چندگانه به کار بگیریم. در واقع، میتوان از یک map روی هر نوع تابع پایتون که فکرش را بکنید استفاده کرد. تنها شرط این کار آن است که آن تابع با دنباله عناصری که عملیات روی آنها اجرا میشود سازگار باشد.
3. فیلتر کردن
تابع توکار filter در پایتون کاملاً مشابه تابع Map است، چون یک تابع را روی یک دنباله (لیست، چندتایی، دیکشنری) اعمال میکند. تفاوت کلیدی این است که ()filter تنها عناصری را بازمیگرداند که تابع اعمالشده مقادیر true برای آنها بازگشت دهد.
برای توضیح بیشتر کد مثال زیر را بررسی کنید:
1# Our numbers
2numbers = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15]
3
4# Function that filters out all numbers which are odd
5def filter_odd_numbers(num):
6
7 if num % 2 == 0:
8 return True
9 else:
10 return False
11
12filtered_numbers = filter(filter_odd_numbers, numbers)
13
14print(filtered_numbers)
15# filte
ما در کد فوق نهتنها درست یا نادرست بودن هر عنصر لیست را بررسی کردهایم؛ بلکه تابع ()filter از این نکته نیز اطمینان حاصل کرده است که تنها عناصری بازگشت مییابند که به صورت درست ارزیابی شدهاند. این تابع برای مدیریت آسان دو مرحلهی بررسی یک عبارت و ساخت یک لیست بازگشتی کاملاً مناسب است.
4. Itertools
ماژول Itertools پایتون، مجموعهای از ابزارها برای مدیریت تکرارکنندهها است. منظور از تکرارکننده (iterator) نوع دادهای است که میتواند در یک حلقه استفاده شود و شامل لیست، چندتایی و دیکشنری است.
استفاده از تابع در ماژول Itertools امکان اجرای عملیات مختلف تکرارکننده را به ما میدهد که به طور معمول نیازمند تابعهای چندخطی و «خلاصهسازی لیست» (list comprehension) پیچیده بودند. کد مثال زیر را برای درک بهتر ویژگیهای فوقالعاده Itertools ملاحظه کنید:
1from itertools import *
2
3# Easy joining of two lists into a list of tuples
4for i in izip([1, 2, 3], ['a', 'b', 'c']):
5 print i
6# ('a', 1)
7# ('b', 2)
8# ('c', 3)
9
10# The count() function returns an interator that
11# produces consecutive integers, forever. This
12# one is great for adding indices next to your list
13# elements for readability and convenience
14for i in izip(count(1), ['Bob', 'Emily', 'Joe']):
15 print i
16# (1, 'Bob')
17# (2, 'Emily')
18# (3, 'Joe')
19
20# The dropwhile() function returns an iterator that returns
21# all the elements of the input which come after a certain
22# condition becomes false for the first time.
23def check_for_drop(x):
24 print 'Checking: ', x
25 return (x > 5)
26
27for i in dropwhile(should_drop, [2, 4, 6, 8, 10, 12]):
28 print 'Result: ', i
29
30# Checking: 2
31# Checking: 4
32# Result: 6
33# Result: 8
34# Result: 10
35# Result: 12
36
37
38# The groupby() function is great for retrieving bunches
39# of iterator elements which are the same or have similar
40# properties
41
42a = sorted([1, 2, 1, 3, 2, 1, 2, 3, 4, 5])
43for key, value in groupby(a):
44 print(key, value), end=' ')
45
46# (1, [1, 1, 1])
47# (2, [2, 2, 2])
48# (3, [3, 3])
49# (4, [4])
50# (5, [5])
5. Generator
تابعهای Generator امکان اعلان یک تابع را به ما میدهند که مانند یک تکرارکننده عمل میکند، یعنی میتواند در یک حلقه استفاده شود. این وضعیت موجب میشود که کد تا حد زیادی سادهتر شود و کارایی حافظه آن نسبت به حلقههای for ساده بسیار بالاتر است. برای نمونه تصور کنید میخواهیم همه اعداد 1 تا 1000 را با هم جمع کنیم. بخش نخست کد زیر شیوه انجام این کار را با استفاده از حلقه for نشان میدهد:
1# (1) Using a for loop
2numbers = list()
3
4for i in range(1000):
5 numbers.append(i+1)
6
7total = sum(numbers)
8
9# (2) Using a generator
10 def generate_numbers(n):
11 num = 0
12 while num < n:
13 yield num
14 num += 1
15 total = sum(generate_numbers(1000))
16
17 # (3) range() vs xrange()
18 total = sum(range(1000 + 1))
19 total = sum(xrange(1000 + 1))
این کد در مواردی که لیست کوچک است، مثلاً در صورت وجود 1000 عدد عملکرد مناسبی دارد. مشکل زمانی بروز پیدا میکند که بخواهیم لیست بسیار بزرگی، برای مثال 1 میلیارد عدد اعشاری را با هم جمع کنیم. در این حالت اگر از یک حلقه for استفاده کنیم، حجم بالایی از حافظه از سوی لیست ایجاد شده در حافظه اشغال میشود. همه افراد به RAM نامحدود دسترسی ندارند تا چنین چیزی را ذخیره کنند. تابع ()range در پایتون همین کار را انجام میدهد و لیست را در حافظه میسازد.
بخش دوم کد فوق روش جمع کردن لیستی از اعداد را با استفاده از generator پایتون نمایش میدهد. یک generator تنها زمانی که لازم باشد، عناصر را میسازد و آنها را در حافظه ذخیره میکند و این کار را به صورت یک به یک انجام میدهد. این بدان معنی است که اگر مجبور باشید 1 میلیارد عدد اعشاری ایجاد کنید، در هر زمان تنها یکی از آنها در حافظه ذخیره خواهند شد. تابع ()xrange در پایتون از generator-ها برای ساخت لیست استفاده میکند.
نکته مهم
نکته مهم اینجا است که اگر بازه بزرگی داشته باشید که بخواهید یک لیست برای آن بسازید باید از یک generator برای تابع xrange استفاده کنید. این وضعیت به طور خاص در مواردی مفید خواهد بود که بخواهید یک سیستم واقعاً حساس از نظر حافظه مانند اپلیکیشن موبایل یا محاسبات پیشرفته بسازید.
این بدان معنی است که اگر بخواهید روی یک لیست چندین بار چرخهای را تکرار کنید و این لیست آن قدر کوچک باشد که در حافظه جای بگیرد، بهتر است از حلقههای for و تابع range استفاده کنید. دلیل این امر آن است که generator-ها و xrange مقادیر لیست را هر بار که به آنها دسترسی پیدا میکنید، از نو میسازند در حالی که range یک لیست استاتیک است و مقادیر صحیح برای دسترس سریع در حافظه ایجاد میشوند.
بدین ترتیب به پایان این مطلب با عنوان بررسی 5 قابلیت پیشرفته پایتون و روش استفاده از آنها میرسیم. هر گونه دیدگاه یا پیشنهاد خود را در بخش نظرات این نوشته با ما و دیگر خوانندگان مجله فرادرس در میان بگذارید.
اگر این مطلب برای شما مفید بوده است، آموزشهای زیر نیز به شما پیشنهاد میشوند:
- مجموعه آموزشهای برنامهنویسی پایتون Python
- مجموعه آموزشهای برنامهنویسی
- زبان برنامه نویسی پایتون (Python) — از صفر تا صد
- ترفندهای پایتون که باید آنها را بدانید — راهنمای کاربردی
- آموزش پایتون: ساخت اپلیکیشن دیکشنری — از صفر تا صد
==