اولین قدمها برای یادگیری برنامهنویسی فانکشنال مهمترین قدمها هستند و در برخی اوقات سختترین قدمها. البته اگر از راه درست وارد شوید چندان هم سخت نخواهد بود.
ترکیب توابع
به عنوان برنامهنویس ما آدمهای تنبلی هستیم، ما دوست نداریم بارها و بارها یک کد تکراری بنویسیم. ما همیشه دنبال راهی هستیم که از کدی که قبلا نوشتهایم در پروژههای آتی استفاده کنیم.
استفاده مجدد از کد در نگاه اول چیز فوقالعادهای به نظر میرسد ولی در عمل کار سختی است زیرا اگر کدتان برای یک پروژه زیادی اختصاصی باشد دیگر نمیتوانید در پروژههای بعدی از آن استفاده کنید، و اگر یک کد را زیادی عمومی بنویسید به کار گیری آن در پروژه جاری دشوار است.
پس آنچه که ما نیاز داریم یک موازنه بین این دو حالت است، یعنی راهی برای ساختن قطعات کوچکتر و با قابلیت استفاده مجدد از کد که بتوان از این قطعات برای ساخت پروژههای پیچیدهتر استفاده کرد.
در برنامهنویسی تابعگرا قطعات ساخت ما توابع هستند. ابتدا توابع را برای انجام عملیات اختصاصی مینویسیم، سپس این توابع را مانند قطعات لِگو سر هم میکنیم.
این عمل ترکیب توابع نامیده میشود.
به عنوان مثال دو تابع زیر در جاوا اسکریپت را در نظر بگیرید:
var add10 = function(value) { return value + 10; }; var mult5 = function(value) { return value * 5; };
کد بالا زیادی شلوغ است، اجازه دهید آن را با استاندارد جدید ES6 برای جاوا اسکریپت و با استفاده از fat arrow بازنویسی کنیم:
var add10 = value => value + 10;var mult5 = value => value * 5;
حالا بهتر شد! حالا اجازه دهید تصور کنیم که ما یک تابع میخواهیم که یک عدد رابگیرد عدد ۱۰ را به آن اضافه کند و نتیجه را در ۵ ضرب کند، میتوانیم این تابع را اینجوری بنویسیم:
var mult5AfterAdd10 = value => 5 * (value + 10)
حتی با وجودی که این یک مثال ساده است باز ما دلمان نمیخواهد مجبور باشیم آن را از صفر بنویسم. نخست به این دلیل که ممکن است ما مرتکب اشتباه شویم مثلا گذاشتن پرانتزها را فراموش کنیم، دوم به این دلیل که ما همین الان یک تابع داریم که عدد ۱۰ را به پارامتر ورودی اضافه میکند و یک تابع دیگر داریم که پارامتر ورودی را ضرب در ۵ میکند، یعنی ما این کد را قبلا نوشتهایم.
پس به جای نوشتن این تابع بگذارید از توابع add10 و mult5 برای ساختن تابع جدیدمان استفاده کنیم:
var mult5AfterAdd10 = value => mult5(add10(value));
ما از توابع موجود برای ساخت تابع mult5AfterAdd10 استفاده کردهایم. اما راه بهتری هم وجود دارد.
در ریاضی f ∘ g به معنای ترکیب توابع است وخوانده میشود: « f با g ترکیب شده است» یا به عبارت رایجتر: « f بعد از g». پس (f ∘ g)(x) برابر است با صدا زدن f بعد از g بر روی متغیر x یا به صورت ساده تر:
f(g(x))
در مثال بالا ما داریم mult5 ∘ add10 یا mult5 بعد از add10، به همین دلیل نام تابع را mult5AfterAdd10 گذاشتهایم.
و این دقیقا چیزی است که انجام دادهایم، ما تابع mult5 را بعد از تابع add10 بر روی متغیر value صدا زدهایم یا به عبارت سادهتر:
mult5(add10(value))
از آنجا که جاوا اسکریپت به صورت داخلی از ترکیب توابع پشتیبانی نمیکند، اجازه دهید مثالمان را با زبان Elm بازنویسی کنیم:
add10 value = value + 10 mult5 value = value * 5 mult5AfterAdd10 value = (mult5 << add10) value
علامت >> عملگر جا دهی است که عملیات ترکیب توابع را در زبان Elm انجام میدهد. این علامت کمک میکند که از جریان دادهها یک درک تصویری داشته باشیم: ابتدا متغیر value به تابع add10 پاس داده میشود سپس نتیجه به تابع mult5 فرستاده میشود.
به پرانتزها در تابع بالا دقت کنید، دلیل وجود آنها این است که مطمئن باشیم که ابتدا توابع ترکیب میشوند و سپس مقدار متغیر value اعمال میشود.
شما از این طریق میتوانید هر تعداد تابع که دوست داشته باشید ترکیب کنید:
f x = (g << h << s << r << t) x
در کد بالا متغیر x به تابع t فرستاده میشود، سپس نتیجه به تابع r فرستاده میشود، سپس نتیجه به تابع s فرستاده میشود، و به همین منوال...
اگر شما این کار را در جاوا اسکریپت انجام میدادید نتیجه این شکلی میشد:
g(h(s(r(t(x)))))
یک کابوس وحشتناک از پراتنزها!
علامتگذاری بدون پارامتر (Point-Free Notation)
یک روش در نوشتن توابع وجود دارد که در آن پارامترها را نمینویسم، این روش علامتگذاری بدون پارامتر نام دارد. در نگاه اول این روش کمی عجیب و غریب به نظر میرسد ولی اگر شما به استفاده از آن ادامه دهید به سودمندی آن پی میبرید.
مترجم: مثل اینکه این غربیهای کافر! به ورودیهای یک تابع علاوه بر پارامتر، پوینت (point) هم میگن. یعنی پوینت در اینجا همون پارامتره.در تابع mult5AfterAdd10 میتوانید ببنید که پارامتر value دو بار عنوان شده است، یک بار در لیست پارامترهای ورودی و یک بار هم در جایی که استفاده شده است:
-- This is a function that expects 1 parameter mult5AfterAdd10 value = (mult5 << add10) value
اما این پارامتر غیر ضروری است زیرا تابع add10 (سمت راستترین تابع از ترکیب توابع بالا) هم همین پارامتر را میپذیرد. کد زیر نمونه بدون پارامتر کد بالا است:
-- This is also a function that expects 1 parameter mult5AfterAdd10 = (mult5 << add10)
استفاده از نمونه بدون پارامتر سودمندیهای بسیاری دارد:
نخست، مجبور نیستیم پارامترهای زائد را عنوان کنیم، و از آنجا که مجبور نیستیم آنها را عنوان کنیم، مجبور نیستیم برای آنها دنبال اسم مناسب بگردیم.
دوم، این مدل نوشتن برای خواندن راحتتر است و کدمان کمتر شلوغ میشود.
مثال بالا یک مثال ساده است، تابعی را تصور کنید که پارامترهای بیشتری میگیرد.
دردسر در بهشت
تا اینجا دیدیم که ترکیب توابع چگونه عمل میکند و همچنین روش علامتگذاری بدون پارامتر را آموختیم. حالا اجازه دهید از این چیزهایی که آموختهایم در یک مثال متفاوت استفاده کنیم تا ببینیم که این شگردها چگونه عمل میکنند. تصور کنید که تابع add10 را با تابع add زیر جایگزین کردهایم:
add x y = x + y mult5 value = value * 5
حالا تابع mult5After10 را با این دو تابع جدید چگونه مینویسیم؟
قبل از اینکه به خواندن ادامه دهید کمی در مورد آن فکر کنید. نه جداً! در موردش فکر کنید و سعی کنید حلش کنید.
بسیارخوب، اگر شما واقعا وقت گذاشته باشید و در موردش فکر کرده باشید ممکن است به چیزی شبیه به این راهحل رسیده باشید:
-- This is wrong !!!! mult5AfterAdd10 = (mult5 << add) 10
اما این راهحل غلط است زیرا تابع add دو تا پارامتر میگیرد.
اگر این مسئله برایتان با زبان Elm واضح نیست سعی کنید آن را با جاوا اسکریپت بنویسید:
var mult5AfterAdd10 = mult5(add(10)); // this doesn't work
این کد اشتباه است ولی چرا؟
برای اینکه اینجا تابع add فقط یکی از دو پارامتر ورودیاش را میگیرد در نتیجه حاصلش اشتباه است و همین حاصل اشتباه به تابع mult5 فرستاده میشود. و همین باعث میشود نتیجه نهایی اشتباه باشد.
در واقع در زبان Elm کامپایلر اجازه نمیدهد همچین کد اشتباهی بنویسید که این یکی از ویژگی های عالی زبان Elm است.
بگذارید دوباره سعی کنیم:
var mult5AfterAdd10 = y => mult5(add(10, y)); // not point-free
کد بالا بدون پارامتر نیست ولی به هر حال میتوان با آن کنار آمد، اما توجه کنید که ما فقط توابع را ترکیب نکردهایم بلکه یک تابع جدید نیز نوشتهایم. و یک مسئله دیگر اینکه اگر این کار پیچیدهتر شود مثلا بخواهیم تابع mult5AfterAdd10 را با تابع دیگری ترکیب کنیم کارمان واقعا سخت میشود.
پس میشود نتیجه گرفت که ترکیب توابع سودمندی محدودی دارد زیرا ما نمیتوانیم این دو تا تابع را ترکیب کنیم. این باعث تاسف است زیرا ترکیب توابع واقعا یک ابزار قدرتمند است.
چگونه میتوانیم این مشکل را حل کنیم؟ چه چیزی نیاز داریم که این مشکل را از سر راه برداریم؟
خب خیلی عالی میشد اگر ما راهی داشتیم که میتوانستیم فقط یکی از پارامترهای وروردی تابع add را از قبل برایش ارسال کنیم و پارامتر دوم را بعدا یعنی زمانی که تابع mult5AfterAdd10 صدا زده شد برایش ارسال کنیم.
خب یک راه حل پیدا کردیم، این راهحل اصطلاحا Currying نامیده میشود.
ادامه دارد...
منبع:+