مینی‌مالیسم : در راستای حل مسئله در برنامه نویسی

تولید شده توسط هوش مصنوعی  Midjourney
تولید شده توسط هوش مصنوعی Midjourney

تو این پست قراره نگاهی به منابع و رویکرد های حل مسئله در برنامه نویسی داشته باشیم.


بازگشت به ریپوهای قدیمی

شروع مسیر من تو برنامه نویسی با ناامیدی زیادی همراه بود. اوایل که شروع میکنی ممکنه حس درجا زدن داشته باشی، حس کنی که هیچ پیشرفتی وجود نداره اما وقتی که به خودت زمان بدی و راه درست رو پیدا کنی این احساسات محو می‌شن.

1.ما انسان های عجولی هستیم .

نمیدونم دقیقا بشر از چه زمانی شروع کرد به سریع تر و سریع تر، خواستن و رسیدن. شاید از وقتی که متوجه شد چقدر این دنیا بزرگه و چقدر چیزهای زیادی وجود داره که هیچوقت نمیتونه همه شون رو داشته باشه. این تشنگی و عطش برای خواستن داره نابود مون میکنه. من تو این مسیر بود که یادگرفتم هرچیز خوبی زمان می‌بره و باید مدت خوبی براش تمرکز کنی ، بهش انرژی و توجه بدی و امیدوار باشی که به نتیجه برسی. عجله دشمن یادگیری هست، تمرکز رو از بین می‌بره و ما رو از مسیر اصلی مون دور میکنه . عادت کردن به اینکه تو هر مسیری زود نتیجه بگیریم، فکر نکنم اصلا تو برنامه نویسی جواب بده! به جاش باید از مسیر لذت برد و تو checkpoint های معینی، نتایج رو بررسی کرد.

2. مشورت کن! ولی بعد اینکه تمام تلاشتو کردی.

تصمیم گرفتم این پست رو بنویسم چون به شکل های مختلف تفاوت نگاه آدمی که حل مسئله بلده و آدمی که بلد نیست رو دیدم. بعد از چند ماه به ریپوی قدیمی ام برگشتم و این بار سعی کردم دوباره یادبگیرم. به طرز عجیبی دیگه مثل قبل ناامید نشدم، گیج نشدم و نحوه رسیدن به layout مناسب خیلی کمتر زمان برد، ذهن من یادگرفت سلسله مراتبی به مشکلی که روبروش هست نگاه کنه.

یادمه قبلا چقدر با ساده ترین سوالات مشکل داشتم، دائم میخواستم حل بشه هرطوری که هست، از بقیه بپرسم ، سرچ کنم یا حتی یه مدت بیخیالش بشم. ولی الان بعد چندماه که یادگیری # C رو شروع کردم، دیگه دنبال جواب سریع و آماده نیستم، هر چیزی رو سرچ می‌کنم حتی بیخود ترین سوالات ممکن رو و به عنوان آخرین و تنها ترین راه حل اگه نتونم هیچ جوابی پیدا کنم سراغ Stack Overflow میرم.

پنج تا از بهترین دوستان شما اینهاست : بپرسید چه چیز، چرا، چه جا، چه وقت و چگونه و هنگامی که نیاز به اندرز دارید از هیچکس دیگر چیزی نپرسید.

3. ساده فکر کنید : هر چی کمتر بهتر

LESS IS MORE !
LESS IS MORE !

میخوام باهاتون یکی از مهم ترین اکتشافات اخیرم رو در میون بذارم :) اینکه خیلی از اطرافیانم و خودم، به جای اینکه به ساده ترین راه حل ممکنی که الان برای مشکل وجود داره فکر کنیم، دنبال سخت ترین، دور از دسترس ترین و غیر ممکن ترین راه حل می‌گردیم و امید داریم اینطوری به نتیجه هم برسیم. واقعا چرا اولین راه حل ساده ای که روبرو مون هست رو نمی‌بینیم و دنبال پیچیده کردن همه چی هستیم ؟ مشکل مینی‌مالیست بودن چیه؟ :D تو این مدت که چالش صد روز کد رو شروع کردم، متوجه این موضوع شدم، که همیشه دارم برای حل کردن سوالم دنبال راه سختش میرم وقتی تمرکز کردم روی این چالش، خیلی از جزئیات ریز و مهمی که ازش بی خبر بودم پیداشون شد، صحبت کردن با آدم های حرفه ای باعث شد حتی بیشتر توجه ام به این نکته جلب بشه . چند وقت پیش یه متنی رو خوندم و اینجا هم خلاصه اش رو گذاشتم، با آقای آخی در این مورد صحبت میکردیم که این نوع از خطاهای کامپیوتری که موقع گرد کردن و کار با اعداد تو محدوده خاصی اتفاق می‌افته محدود به کامپیوتر ها نمیشه و از زمانی که اینترنت رشد پیدا کرد به شیوه های جدیدی برای ذخیره سازی اعداد نیاز داشتیم، اما بدلیل استفاده از ابزارهای سطح بالا و نزدیک به زبان انسان این موضوع رو متوجه نمی‌شیم و حتی شارژر های موبایل هم از این قائده مستثنی نیستن :

خیلی پیش پا افتاده‌تر فکر کن .همین اپل و اندروید اتحادیه اروپا اپل رو مجبور کرد که اون هم بیاد از تایپ‌سی استفاده کنه که توی اندروید وجود داره. اپل اومده یه شارژر تایپ سی طراحی کرده که به تمام گوشی‌های تایپ سی دنیا می‌خوره اما هیچ شارژر تایپ سی دیگه‌ای روی اپل کار نمی‌کنه??‍♂️?

حل مسئله

مراحل حل مسئله
مراحل حل مسئله


Divide and Conquer: make it to small pieces.

یه ویدئو از یوتیوب چند وقت پیش می‌دیدم، درمورد این که چطور سوالات Leetcode رو حل کنیم، نکته ای گفت که خیلی مهم بود : همه تازه کار ها وقتی با یه سوال جدید روبرو میشن، سریع دست به کار میشن و شروع به نوشتن کد میکنن، امتحان میکنن و سعی میکنن از طریق تغیر دادن و تست کردن به جواب برسن. من همیشه همین کار رو انجام می‌دادم و خیلی وقت ها طول میکشید که به جواب برسم ولی حرفه ای ها، چند دقیقه اول رو فقط صرف فکر کردن به سوال میکنن، درک سوال و فهمیدن اینکه چه چیزی ازمون میخواد نصف بیشتر مسیر هست . تازه مرحله دوم هم کد زدن نیست. حل کردن سوال بدون نوشتن کد هست، وقتی سوال رو متوجه شدی پس الان حتما یه راه حل براش تو ذهنت داری، و باید بتونی این مراحل رو برای خودت واضح کنی تا جایی که به حل سوال برسی.
مرحله آخر میشه عملی کردن راه حلی که تو ذهن مون داریم به شکل کد. به عبارت دیگه یه کار رو اون قدر به مراحل کوچیک تقسیم کن که راحت بتونی انجامش بدی. کاری که من انجام میدادم این بود که سعی داشتم یه مسئله سخت رو بدون داشتن قدم های کوچیک حل کنم. این شکستن مسئله به اجزا کوچیک تر و انجام دادن هر بخش کوچیک، مفهوم اصلی یکی از مهم ترین الگوریتم ها یعنی Divide and Conquer هست .

دوست خوبم مهدیس دانشجوی کارشناسی ریاضی شهید بهشتی، کتاب چگونه مسئله را حل کنیم نوشته جورج پولیا رو بهم معرفی کرده بود.

قسمت هایی از کتاب رو در راستای مراحل حل سوال براتون میذارم :

فهمیدن مسئله : اول باید مسئله را بفهمید

مجهول چیست ؟ داده ها کدام است ؟ شرط چیست ؟ شکلی رسم کنید. علامت های مناسب رو به کار ببرید.

طرح نقشه : ارتباط میان داده ها و مجهول را پیدا کنید. ممکنه در صورت پیدا نکردن ارتباط مستقیم بین داده ها و مجهول، مسئله های کمکی در نظر بگیرید. باید سرانجام یک نقشه برای مسئله طرح کنید.

آیا آن را پیشتر دیده بودید ؟ آیا همین مسئله را به صورت دیگر دیده اید ؟ ( وصل کردن اطلاعات گذشته به وضعیت فعلی باعث تحریک نورون ها و یادگیری بهتر میشه ). به مجهول نگاه کنید! و بکوشید تا درباره مسئله ای بیاندیشید که همین مجهول یا شبیه آن را داشته باشد. در این جا مسئله ای وابسته به مسئله شما وجود دارد که پیشتر حل شده است، آیا می‌توانید آن را به کار ببرید ؟ آیا می‌توانید روش آن را به کار ببرید؟ آیا باید یک عنصر کمکی را وارد کنید تا به کار بردن آن را ممکن سازد ؟ آیا می‌توانید صورت مسئله را دوباره بیان کنید ؟ آیا می‌توانید آن را به صورتی دیگر بیان کنید ؟ به تعریف ها رجوع کنید. اگر نمی‌توانید مسئله طرح شده را حل کنید، نخست به حل کردن مسئله ای وابسته به آن بپردازید. آیا می‌توانید مسئله وابسته ای را که بیشتر در دسترس باشد تخیل کنید ؟

سوم : اجرای نقشه

در ضمن اجرای نقشه حل مسئله، هر گام را که برمی‌دارید وارسی و امتحان کنید . آیا می‌توانید آشکارا ببینید که گام برداشته شده درست بوده است ؟ آیا می‌توانید درست بودن آن را ثابت کنید ؟

چهارم : امتحان کردن جوابی که به دست آمده، یعنی به عقب نگاه کردن .

آیا می‌توانید نتیجه را بررسی کنید ؟ آیا میتوانید نتیجه را از راهی دیگر بدست آورید ؟ آیا می‌توانید نتیجه یا روش را در مسئله ای دیگر به کار ببرید ؟

حل کردن مسائل، همچون مثلا شناوری، یک مهارت تمرینی و عملی است. هر مهارت عملی را از راه تقلید و تمرین بدست می‌آوریم. برای آموختن شنا از کاری که دیگران با دست و پای خود انجام می‌دهند تا سر خود را بالای آب نگاه دارند، تقلید می‌کنید و سرانجام با تمرین کافی شایستگی شنا کردن بهره شما می‌شود. برای آموختن فن حل مسائل باید به ملاحظه و مشاهده آنچه دیگران برای حل مسئله می‌کنند و تقلید از ایشان بپردازید و سرانجام خود می‌آموزید که چگونه در ساختن مسائل توفیق حاصل کنید.


گشتن در پیرامون یک مانع، همان کاری است که ما در حل هرگونه مسئله انجام می‌دهیم.


مثال

با یه سوال ساده از بحثLINQ های #C شروع کنیم .

فرض کنید، اطلاعات دانش آموز ها رو به این شکل داخل فایل StudentDatabase.cs ذخیره کردیم :

public static IQueryable<Student> GetStudentsFromDb()
{
return new[] {
new Student() {StudentId = 1, StudentName = &quotJohn Bross&quot, Score = 82, StudentCity = &quotNYC&quot, StudentActvity = true},
new Student() {StudentId = 2, StudentName = &quotJasmine Curly&quot, Score = 84, StudentCity = &quotNYC&quot, StudentActvity = false},
new Student() {StudentId = 4, StudentName = &quotBen Wild&quot, Score = 65, StudentCity = &quotCA&quot, StudentActvity = true},
new Student() {StudentId = 6, StudentName = &quotSara Ride&quot, Score = 81, StudentCity = &quotCA&quot, StudentActvity = true},
new Student() {StudentId = 7, StudentName = &quotBonnie Brooklyn&quot, Score = 80, StudentCity = &quotLA&quot, StudentActvity = false},
new Student() {StudentId = 8, StudentName = &quotLoren Gibson&quot, Score = 88, StudentCity = &quotCA&quot, StudentActvity = false},
}.AsQueryable();
}

حالا می‌خوایم اسم دانش آموزانی رو داشته باشیم که score بالای 80 دارن. چطور انجامش می‌دین ؟

var studentsList = GetStudentsList();
studentsScoreMoreThan80 = studentsList.Where(x => x.Score > 80).Select(x => x.StudentName);
foreach (var item in studentsScoreMoreThan80)
{ 
Console.WriteLine(item);
}

حالا اگه همین سوال رو یه کم سخت تر کنیم، با توجه به مراحلی که گفتیم باید بتونید با توجه به سوال قبلی، مقایسه کردن، سرچ کردن و بررسی کد تون، حلش کنید.

این بار اسم دانش آموزهایی که فعال score بالای 80 و شهر نیویورک هستن رو پیدا کنید.

با توجه به سوال قبلی، باید مطمئن بشیم که Lambda Expressions و مباحث دیگه رو بلد هستیم، پس اگه یادتون رفته مثلا این عبارت #Lambda Expressions in C رو سرچ کنید تا چند نمونه و توضیح ببینید.


//Access to the class metho
 var studentList = StudentDatabase.GetStudentsFromDb();
var studentsName1 = studentList.Where(x => x.StudentActvity && x.Score > 80 && x.StudentCity == &quotNYC&quot).Select(x => x.StudentName);
foreach (var item1 in studentsName1)
{
Console.WriteLine($&quotStudents with true activity, score > 80 and NY city : {item1}&quot);
}

می‌تونید این کد رو بهتر کنید ؟ اگه راه حل کوتاه تر و بهتری دارین پس سعی کنید اونا رو هم امتحان کنید مثلا من پرانتز های اضافی و && های اضافی رو حذف کردم. و همچنین میتونید ریپوزیتوری این برنامه رو از اینجا ببینید.

پایان

همونطور که جورج پولیا گفته :

حل کردن مسائل یکی از فعالیت های بنیادی انسان است. بعضی از اشخاص در رسیدن به هدف های خود که حل کردن مسائل است کامیابی بیشتری دارند و بعضی کمتر. باید میل به حل کردن مسئله در ما وجود داشته باشد. بدون داشتن میلی شدید به حل مسئله دشوار، بخت در رسیدن به جواب آن یار ما نخواهد شد.


هر نظر، پبشنهاد و انتقادی دارین خوشحال میشم بدونم :) ممنونم که خوندین ^_^