کاهش مصرف حافظه و افزایش سرعت اجرای کد پایتون با Generator – راهنمای کاربردی


اغلب افرادی که با Generator-های پایتون آشنایی ندارند یا به تازگی آشنا شدهاند، در مورد اهمیت این مفهوم به دیده اغماض مینگرند. Generator-ها به طور خاص در زمینه یادگیری ماشین برای نوشتن تابعهای سفارشی نقش بسیار مهمی ایفا میکنند. در این مقاله با روش کاهش مصرف حافظه و افزایش سرعت اجرای کد پایتون با Generator آشنا خواهیم شد.
تابعهای Generator این امکان را فراهم میسازند که تابعی اعلان کنیم که مانند یک تکرارکننده (iterator) عمل میکند. این نوع از تابعها به برنامهنویس امکان میدهند یک تکرارکننده را به روشی سریع، آسان و تمیز ایجاد کند. منظور از تکرارکننده شیئی است که میتواند (در یک حلقه) تکرار شود. از تکرارکننده برای تجرید یک کانتینر دادهها استفاده میشود تا طوری رفتار کند که گویی یک شیء تکرارپذیر است. برخی از موارد رایج از اشیای تکرارپذیر شامل رشتهها، لیستها و دیکشنریها هستند. Generator تا حد زیادی شبیه یک تابع است، اما به جای return از کلیدواژه yield استفاده میکند. برای درک بهتر یک مثال را بررسی میکنیم:
این یک تابع generator است و زمانی که آن را فراخوانی کنیم، یک شیء generator بازگشت میدهد:
نکته مهمی که باید توجه داشته باشیم این است که حالت (State) درون بدنه تابع generator قرار گرفته است. امکان حرکت به صورت گام به گام با استفاده از تابع داخلی next() نیز وجود دارد:
فراخوانی ()next پس از رسیدن به انتها
StopIteration یک نوع استثنای داخلی است که به صورت خودکار در مواردی که generator نتواند yield کند صادر میشود. این نشانهای از این است که حلقه باید متوقف شود.
گزاره yield
وظیفه اصلی گزاره yield کنترل گردش تابع generator به روشی است که مانند گزاره return به نظر رسد. هنگامی که یک تابع generator را فرا میخوانید یا از یک عبارت generator استفاده میکنید، یک تکرارکننده خاص به نام generator دریافت میشود. میتوان این generator را به یک متغیر انتساب داد تا از آن استفاده کند. زمانی که متدهای خاص روی generator از قبیل ()next را فراخوانی میکنید، کد درون تابع تا yield اجرا میشود.
هنگامی که پایتون با گزاره yield مواجه شود، برنامه اجرای تابع را معلق میکند و مقدار yield شده را به فراخوانی کننده بازگشت میدهد. این وضعیت مخالف گزاره return در پایتون است که اجرای تابع را به طور کامل متوقف میکند. زمانی که یک تابع تعلیق شود، حالت تابع ذخیره میشود. اکنون که با generator پایتون آشنا شدید، نوبت آن رسیده است که رویکرد معمولی را با رویکردی که از generator استفاده میکند از نظر مصرف حافظه و زمان مورد نیاز برای اجرای کدهای پایتون مقایسه کنیم.
صورت مسئله
فرض کنید مجبور هستیم روی یک لیست بزرگی از اعداد (مثلاً 100000000 عدد) حلقهای تعریف کنیم و مربع همه این اعداد را در یک لیست جداگانه ذخیره کنیم.
رویکرد معمول
با اجرای کد فوق خروجی زیر به دست میآید:
It took 21.876470000000005 Secs and 1929.703125 Mb to execute this method
رویکرد Generator
با اجرای کد فوق خروجی زیر به دست میآید:
It took 2.9999999995311555e-05 Secs and 0.02656277 Mb to execute this method
برای مقایسه بهتر تفاوت دو رویکرد در جدول زیر نشان داده شده است:
چنان که میبینید، هم مصرف حافظه و هم زمان مورد نیاز برای اجرای کد در زمان استفاده از Generator به میزان زیادی کاهش یافته است. Generator-ها تنها بنا به تقاضا کار میکنند و مشهور است که از طریق ارزیابی تنبل (lazy) عمل میکنند. این بدان معنی است که میتوانند در مصرف CPU، حافظه و دیگر منابع محاسباتی صرفهجویی کنند.
سخن پایانی
در این بخش به جمعبندی مطالب طرح شده در این مقاله میپردازیم. در این راهنما نشان دادیم که generator-های پایتون میتوانند برای کاهش مصرف حافظه و اجرای سریعتر کدها مورد استفاده قرار گیرند. مزیت اصلی آنها در این واقعیت نهفته است که نتایج را در حافظه ذخیره نمیکنند، بلکه آنها را به صورت درجا تولید میکنند و از این رو حافظه تنها زمانی استفاده میشود که نتیجهای را از آنها طلب کنیم. همچنین generator-ها اغلب بخشهای کد از پیش آماده مورد نیاز برای نوشتن تکرارکنندهها را دور میاندازند که به کاهش اندازه کد نیز کمک میکند.
اگر این مطلب برای شما مفید بوده است، آموزشهای زیر نیز به شما پیشنهاد میشوند:
- مجموعه آموزشهای پایتون Python
- مجموعه آموزشهای برنامهنویسی
- گنجینه آموزش های برنامه نویسی پایتون (Python)
- دستورات break و continue در پایتون — به زبان ساده
- ماژول های پایتون (Python Modules) — به زبان ساده
==
بررسی و مقایسه شما به کل اشتباهه
تابع اول لیست رو محاسبه و برمیگردونه
تابع دوم صرفا جنراتور ساخته شده و برگردونده شده چیز معادلی محاسبه نشده که بیایم مقایسه کنیم. الا تازه باید جنراتور رو بندازیم تو فور و مقادیر رو محاسبه کنیم.
سلام و وقت بخیر؛
بهطور کلی، جنریتورها بهعنوان یک ویژگی قدرتمند در پایتون، میتوانند در مصرف حافظه صرفهجویی کنند. چون مقادیر را در لحظه تولید میکنند بهجای اینکه همه آنها را در حافظه ذخیره کنند.
از همراهی شما با مجله فرادرس سپاسگزاریم.
سلام.
خیلی عالی با یک مثال ساده مزیت استفاده از generator را توضیح دادید.
در آموزش پایتون همراه با مثال های عملی هم به این موضوع پرداخته شده است.