Remove ads
From Wikipedia, the free encyclopedia
موضوع تحلیل الگوریتمها تعیین میزان منابعی است که برای اجرای هر الگوریتم لازم است. منابعی مثل زمان، حافظه، پهنای باند ارتباطی، یا سختافزار رایانه در نظر گرفته میشوند. اکثر الگوریتمهای طراحی شده برای کار با ورودیهای با طول اختیاری تولید میشوند. کارایی یا پیچیدگی هر الگوریتم را با تابعی نشان میدهند که تعداد مراحل لازم برای اجرای الگوریتم را برحسب طول داده ورودی، یا میزان محلهای گرفته شدهٔ حافظه را بر حسب طول داده ورودی نشان میدهد. پیچیدگی زمانی، مقدارپیچیدگی محاسباتیای را توصیف میکند که در اجرای یک الگوریتم مصرف میشود. غالباً مشاهده میشود که یک مسئله را با استفاده از چندین تکنیک مختلف میتوان حل نمود ولی فقط یکی از آنها به الگوریتمی منجر میشود که از بقیه سریعتر است.
در تجزیه و تحلیل نظری الگوریتم آنچه مشترک است، برآورد پیچیدگی در معنای تقریبی آن است. به عنوان مثال، برآورد تابع پیچیدگی برای ورودی خودسرانه بزرگ. نمادهای برای این منظور استفاده میشوند. مثلاً گفته میشود، جستجوی دودویی به اجرا در مرحله تناسب دارد.
معمولاً تخمینهای تقریبی استفاده میشوند، چرا که پیادهسازیهای مختلف از همان الگوریتم، ممکن است در کارایی متفاوت باشد. با این حال بازده هر دو، با «منطق» پیادهسازی یک الگوریتم داده شده، ضرب در یک ضریب ثابت به نام ثابت مخفی مرتبط است.
در مورد تحلیل الگوریتم باید در مورد موضوعاتی مانند تابع رشد، روش تحلیل الگوریتمهای ترتیبی و بازگشتی، حل رابطههای بازگشتی ساده، همگن و نا همگن و همچنین تحلیل سرشکنی صحبت کرد.
در نظریه پیچیدگی محاسباتی، الگوریتمها از حیث کارایی در استفاده از منابعی مانند زمان و فضا (حافظه) مورد بررسی قرار میگیرند. در این بررسی آنچه اهمیت دارد میزان زمان و فضای مورد نیاز برای حل یک مسئلهٔ خاص نیست بلکه چگونگی افزایش میزان زمان و فضای مورد نیاز با بزرگ شدن ورودی الگوریتم مورد توجه است.
مثلاً فرض کنید دو الگوریتم الف و ب برای حل مسئلهٔ زیر پیشنهاد شدهاند. در این مسئله نقشهٔ یک منطقه شامل شهرهای قرار گرفته در آن منطقه، جادههای بین آنها و طول هر جاده و نیز نام دو شهر مبدأ و مقصد داده شدهاست. هدف پیدا کردن کوتاهترین مسیر از مبدأ به مقصد است.
فرض کنید به صورت تجربی مشاهده کردهایم که اگر تنها یک شهر به تعداد شهرهای نقشه اضافه شود الگوریتم الف به دو برابر زمان برای حل مسئله نیاز دارد در حالی که زمان مورد نیاز برای الگوریتم ب وقتی دو برابر میشود که تعداد شهرهای نقشه دو برابر شده باشد. بدیهی است که در چنین شرایطی الگوریتم ب را کاراتر از الگوریتم الف میدانیم.
در نظریه پیچیدگی محاسباتی نیز همین روش را پیمیگیریم ولی به جای روشهای تجربی از روشهای ریاضی استفاده میکنیم. به این منظور ابتدا برای ورودی مسئله یک اندازه تعریف میکنیم. این اندازه میتواند میزان حافظهٔ مورد نیاز برای ذخیره کردن ورودی مسئله باشد. سپس زمان (یا فضای) مورد نیاز برای حل مسئله توسط یک الگوریتم به صورت تابعی از اندازهٔ مسئله محاسبه میشود. به محاسبه یا تقریب زدن این چنین تابعی تحلیل الگوریتم گفته میشود.
در تحلیل الگوریتمها بهترین، بدترین و متوسط زمان اجرا بررسی میشود. هرچند معمولاً بهترین زمان اجرای آن اهمیت ندارد؛ زیرا اساساً بهازای ورودیهای غیر محتمل این حالت رخ میدهد. لیکن لازم است الگوریتمها را در بدترین حالت و در صورت امکان در حالت میانگین تحلیل و بررسی کنیم. ممکن است پیچیدگی الگوریتم در یک حالت میانگین بهتر از پیچیدگیاش دربدترین حالت باشد و این اطلاعات مفیدی در مورد آن الگوریتم است.[1]
الگوریتمها به دو دسته تقسیم میشوند، بازگشتی و ترتیبی. الگوریتمهای ترتیبی را معمولاً با شمارش دفعات اجرای دستوری که بر اساس اندازه دادهٔ ورودی، بیشترین بار اجرا میشود، (در صورتی که زمان اجرای دستور از یک مقدار ثابت تبعیت کند) تحلیل میکنیم. اما زمان اجرا و الگوریتم بازگشتی با رابطههای بازگشتی بیان میشوند. روشهای مختلفی برای حل این نوع رابطهها داریم. یک روش خوب برای حل رابطههای بازگشتی، استفاده از درخت بازگشتی است. این روش نحوه جایگذاری یک عبارت بازگشتی و نیز مقدار ثابتی را که در هر سطح از آن عبارت به دست میآید، نشان میدهد. حاصل جمع مقادیر ثابت تمام سطرها جواب حل بازگشتی است.[2][3]
برای بررسی خوب بودن یک الگوریتم، باید به آهنگ رشد منحنی زمان اجرا-اندازه ورودی یا میزان حافظه مصرفی-اندازه ورودی توجه کنیم. برای بررسی دقیقتر به بررسی شاخصهای تحلیل الگوریتم و تعریف توابع رشد میپردازیم.[4]
همانطور که در بالا گفته شد، تحلیل الگوریتم به معنای تخمین میزان منابعی است که الگوریتم به آن نیاز دارد. فرض میکنیم با یک تک پردازندهٔ عمومی پیادهسازی شده با حافظه دسترسی تصادفی (RAM) طرف هستیم و الگوریتم پیادهسازی شدهٔ ما توسط برنامهٔ کامپیوتری، عملیاتها را همزمان انجام نمیدهد. در مدل RAM دستورات تعریف شدهای وجود دارند که اجرا کردنشان از یک مقدار ثابتی تبعیت میکند، دستورهایی که در ساختار کامیپوتر تعبیه شدهاند: مانند عملگرهای حسابی (جمع و تقسیم و ضرب و غیره)، علمیاتهای انتقال اطلاعات (ذخیره کردن، رونویسی کردن حافظه، بازخوانی حافظه و غیره) و عملیاتهای کنترل (فراخوانی توابع، بازگشتناز دستور و غیره).[5]
شبه کد زیر که الگوریتم مرتبسازی حبابی را پیادهسازی میکند در نظربگیرید[6]:
for i=1 to A.length
for j=1 to A.length - i
if A[j]>A[j+1]
swap A[j],A[j+1]
زمان اجرای الگوریتم، جمع زمان اجرای هر دستوری است که اجرا میشود. اگر زمان اجرای هر خط را (که گفتیم مقداری ثابت است) با نشان دهیم و اگر را زمان کل دستورات خط ۱ تا ۴ در نظر بگیریم، زمان اجرای کل برنامه برابر جمع زمان کل اجرای دستورات هر خط میشود.
برای خط اول:
برای خط دوم:
برای خط سوم: و خط چهارم هم در بدترین حالت (اگر هر دفعه وارد if شود) است. پس زمان اجرای کل برنامه در بدترین حالت:
است.
طبق تعریفی که در قسمت تابع خواهیم گفت، با وجود داشتن و درست بودن رابطه برای هر خواهیم دید که الگوریتم بالا است.[7]
بنا به تعریف، هر الگوریتمی که نوسان رفتاری کمتری از خود نشان دهد، اصطلاحاً از نظر محاسباتی باثباتتر است. نظر به این تعریف، دو موضوع تعیینکنندهٔ میزان ثبات محاسباتی الگوریتم خواهند بود:
برای سنجش میزان ثبات زمانی، میتوان از شاخص (و یا ) استفاده کرد. مسلماً الگوریتمی از نظر محاسباتی باثباتتر است که واریانس ها کمتر باشد. روش دیگری که میتوان میزان ثبات زمانی دو یا چند الگوریتم را مقایسه کرد، آن است که با استفاده از آزمون نیکویی بَرازش بررسی کنیم که مقادیر ها مربوط به کدام الگوریتم، در بازهٔ ، به توزیع یکنواخت نزدیکتر است.
در این باب چند نکته مدنظر است:
بهطور مشابه، برای سنجش میزان ثبات الگوریتم در کیفیت جواب نهایی میتوان واریانس کیفیت جوابهای الگوریتم را بعد از چند بار اجرا، اندازه گرفت؛ بنابراین، میتوان قواعد نکتهٔ قبل را برای این نوع ثبات هم بیان کرد. واضح است که بهترین عملکرید برای ثبات کیفیت جواب نهایی را الگوریتمهای دقیق دارند.[8]
مسلماً هر الگوریتمی که سریعتر همگرا شود، مطلوبتر است. این نکته دربارهٔ الگوریتمها دقیق کاملاً صادق است اما دربارهٔ الگوریتمهای تقریبی باید نرخ همگرایی با احتیاط بیشتری بررسی شود. زیرا ممکن است الگوریتم A در مدت زمان کوتاهتری در مقایسه با الگوریتم B به یک نقطهٔ بهینهٔ محلی همگرا شود، اما در عوض الگوریتم B به یک جواب بهینهٔ سراسری همگرا شود. برای بررسی همگرایی که به تنهایی خود یک مقولهٔ جداست به تبیین تعاریف زیر میپردازیم:
اگر الگوریتمی بدینصورت باشد که اگر با هر نقطهٔ آغازین دلخواهی شروع شود و در نهایت به یک جواب مشخص برسد، آن را همگراشوندهٔ سراسری مینامند. از این نوع الگوریتمها میتوان به الگوریتم غیر مرکب (سیمپلکس) اشاره کرد.
فرض کنید الگوریتم A با شروع از جواب اولیهٔ ، دنبالهٔ جوابهای ، ، .... را تا همگرا شدن به جواب تولید کند که میتواند جواب بهینهٔ محلی یا سراسری باشد. به عبارت دیگر، میتوانیم الگوریتم را معادل یک تابع در نظر بگیریم که:
در این صورت مرتبهٔ همگرایی بزرگترین عدد حقیقی نامنفی مانند است که بهازای آن حاصل حد زیر عددی متناهی و نامنفی شود:
که در این حال، را نرخ همگرایی مینامیم.
عدهای معتقدند که هرچقدر مرتبهٔ همگرایی الگوریتم بزرگتر باشد، با الگوریتم بهتری سروکار داریم؛ زیرا فاصله از جواب حدی حداقل در قسمت انتهایی دنبالهٔ جوابها، به اندازهٔ توان اُم یک قدم، کاسته میشود. اما میتوان به این نظر زمانی که عملکرد الگوریتم از ابتدا تا میانهش دچار تغییرات محسوسی شود ایراداتی وارد کرد.
در توضیح بالا اگر و ، اصطلاحاً گفته میشود که همگرایی الگوریتم از مرتبهٔ خطی و با نرخ همگرایی است. اکنون فرض کنید دنباله جوابهای الگوریتم بهصورت خطی است و با نرخ همگرایی به جواب همگرا میشود. در اینصورت، میتوان گفت که ضریب ثابتی مانند وجود دارد که نقاط، حداقل با سرعت یک دنبالهٔ هندسی همچون به جواب همگرا میشوند. از این رو در بعضی منابع همگرایی خطی را همگرایی هندسی نیز مینامند.
ثبات نرخ همگرایی یکی از شاخصهاییاست که با استفاده از واریانس دادههای ستون شیب همگرایی برای هر الگوریتمی قابل محاسبهاست. هر الگوریتمی که دارای ثبات در مدت زمان اجرا و ثبات در کیفیت جواب نهایی باشد، دارای ثبات در نرخ همگرایی خواهد بود.[9]
یکی از شاخصهای تحلیل الگوریتمها بحث هوشمندی الگوریتم است. شاید این مقوله تا حد زیاد کیفی به نظر برسد اما بررسی آن خالی از لطف نیست.
این بحث کمی ریشهدار تر از زمانی است که مفهوم الگوریتم و تحلیل آن صورت بندی شد و اساساً این مسئلهٔ اساسی مطرح بود که کمال هوشمندی یک الگوریتم چیست؟ آیا قرابت آن به فکر و استدلال نشانهٔ هوشمندی آن است یا برعکس؛ چون اینطور مینماید که انسانها معمولاً کارها را بهینه انجام نمیدهند. بنا به تعریفی یک الگوریتم را هوشمندانه میگویند اگر:
همانطور که محسوس است، این تعریف نیز چندان از ذات کیفی بودن خود شاخص دور نشدهاست. شاید تعریف زیر بتواند قدری به کمٓیسازی این ویژگی کمک کند:
الگوریتم A از الگوریتم B هوشمندتر است اگر بهطور همزمان:
چند نکته درمورد مقولهٔ هوشمندی مطرح میشود:
۱. برای تابع تابع را به صورت مجموعه توابع زیر تعریف میکنیم:
و میگوییم تابع کران بسته مجانبی برای است و معمولاً به کار میبریم: [11]
۲. برای تابع تابع را به صورت مجموعه توابع زیر تعریف میکنیم:
ُیعنی آهنگ رشد تابع برای مقادیر بزرگ n، بیشتر یا مساوی آهنگ رشد تابع است. در این صورت میگوییم تابع کران بالای مجانبی برای است و معمولاً به کار میبریم: [12]
۳. برای تابع تابع را به صورت مجموعه توابع زیر تعریف میکنیم:
ُیعنی آهنگ رشد تابع برای مقادیر بزرگ n، کندتر یا مساوی آهنگ رشد تابع است. در این صورت میگوییم تابع کران پایین مجانبی برای است و معمولاً به کار میبریم: [13]
نکته: شرط لازم و کافی برای اینکه این است که و
در الگوریتمهای بازگشتی که اندازهٔ ورودی n است؛ زمان اجرای الگوریتم از رابطه زیر پیروی میکند:
متغیر: را به صورت تعریف میکنیم؛ و پیچیدگی زمانی الگوریتم را به صورت زیر محاسبه میکنیم:
Case | حالت بندی روی درمقایسه با , یعنی | توضیحات | توصیف با تابع رشد | مثال |
---|---|---|---|---|
۱ | اگر در شرایطی که | آهنگ رشد بر برگها غلبه میکند. | جواب به صورت زیر است :
|
اگر و , آنگاه . |
۲ | وقتی برای هر | آهنگ رشد توابع بازگشتی با تقریباً یکسان است. | جواب به صورت زیر است : | اگر و , آنگاه .
اگر و , آنگاه . |
۳ | وقتی در حالی که | آهنگ رشد توابع بازگشتی ایجاد شده زیاد بوده و تعداد برگها به غلبه میکند. | جواب به صورت زیر است:
|
اگر و آنگاه . |
گاهی با دنبالهای از اعمال روی دادهساختاری سروکار داریم که هزینهشان بالا ولی تعدادشان پایین است و یا برعکس؛ یا اینکه یک عمل که هزینه زیادی دارد موجب میشود تا همان عمل در مراحل بعد با هزینهٔ کمتری انجام شود. در این حالت هزینهٔ یک عمل در بدترین حالتش زیاد است، اما اگر میانگین مجموع هزینهها را برای همهٔ اعمال موجود در دنباله حساب کنیم؛ هزینهٔ معقولتری نسبت به هزینه در بدترین حالت را به دست آوردهایم؛ که به آن هزینهٔ سرشکنی و به روش محاسبه آن تحلیل سرشکنی میگویند.[15]
در این روش مطابق تعریف، جمع هزینههای اعمال محسابه میشود و بر تعداد آنها تقسیم میشود تا هزینهٔ سرشکن شده بهدست آید.
این روش یک مدلسازی با سازوکار خزانهای است. در این روش به ازای هر عمل، مقداری پرداختی وجود دارد. فرض کنید که مبلغی صرف انجام خود عمل میشود و مازاد آن در مخزن ذخیره میشود. در هر عملیات اگر هزینه بیشتر از پرداختی آن نوبت بود از خزانه برای جبران آن استفاده میشود. درصورتی که خزانه هیچگاه کمبود نداشته باشد، میزان پول پرداختی برای هر عمل با هزینهٔ سرشکن شدهٔ آن عمل متناسب است.
این روش تعمیمیافتهٔ همان روش حسابداری است که با فرمولبندی ریاضی بیان و تحلیل میشود.[16]
فرض کنید دادهساختار اولیه و دادهساختار ما بعد از عمل ام باشد. همچنین فرض کنید که هزینهٔ واقعی عمل ام است. در این صورت تابع پتانسیل را تعریف میکنیم که را به یک عدد حقیقی نگاشت میکند. همچنین را مطابق فرمول زیر تعریف میکنیم:
و آن را هزینهٔ سرشکن شدهٔ عمل ام مینامیم.
در این صورت داریم:
اگر ، داریم:
یعنی کران بالایی برای است که میخواهیم بدست آوریم.
با نیمنگاهی به روش حسابداری میتوان چنین تناظری میان آن و روش تابع پتانسیل برقرار کرد:
یک شمارندهٔ بیتی درنظر بگیرید که در ابتدا صفر است. با هر مرحله افزایش حداکثر یک بیت آن از ۰ به ۱ تغییر میکند و تعدادی از بیتها نیز از ۱ به ۰ تغییر خواهند داشت. هزینهٔ دقیق این عملیات، متناسب با تعداد تغییر بیتها در شمارنده خواهد بود. فرض کنید بیتها به ترتیب ارزش از تا شماره گذاری شده باشند. در اینصورت خواهیم داشت:
بیت ۰: با هر افزایش تغییر میکند یعنی در مجموع بار.
بیت ۱: یک در میان تغییر میکند یعنی در مجموع بار.
بیت : در مجوع دقیقاً بار تغییر میکند.
تعداد تغییرات بیتها:
پس هزینهٔ سرشکن شدهٔ هر عمل افزایش، است.[18]
در این روش را برابر تعداد عناصر موجود در پشته میگیریم. بدیهتاً در ابتدا این مقدار صفر بوده تا پایان مجموع عملیاتها مقداری مثبت خواهد داشت.
اکنون هزینهٔ سرشکن شده را بدست میآوریم:
Push: چون یک عنصر اضافه میشود داریم:
از طرفی چون میدانیم که هزینهٔ درج واقعی است پس:
Pop: چون یک عنصر کم میشود داریم:
هزینهٔ واقعی حذف هم است پس
Multi-Pop: چون تعدادی عمل Pop است پس هزینهٔ آن نیز صفر میباشد.
بنابراین چون
هزینهٔ سرشکن شده است.[19]
آنچه که در بالا مطرح شد، شاخصهایی بود برای تحلیل عادی الگوریتمها به کار میرفت؛ لیکن الگوریتمهایی هستند دوگانه (هیبریدی) که حل آنها ابتکاری یا فرا ابتکاری محسوب میشود و رسیدن به آنها از چارچوب یک تحلیل ریاضیاتی ساده و بهکار بردن الگوریتمهای آشنا خارج است که میتوان به نمونههایی از آنها اشاره کرد:
Seamless Wikipedia browsing. On steroids.
Every time you click a link to Wikipedia, Wiktionary or Wikiquote in your browser's search results, it will show the modern Wikiwand interface.
Wikiwand extension is a five stars, simple, with minimum permission required to keep your browsing private, safe and transparent.