هرم جفت‌شده

از testwiki
پرش به ناوبری پرش به جستجو

یک هرم جفت شده یک نوع داده ساختار هیپ با پیاده‌سازی نسبتاً ساده و عملکرد سرشکن شدهٔ عالی معرفی شده توسط Micheal Fredman، Robert Sedgewick، Daniel Sleator و Robert Tarjan در سال ۱۹۸۶ می‌باشد.[۱] هرمهای جفت شده ساختمان‌های درختی چند راهه می‌باشند که مانند هرم مرتب شده‌اند و می‌توانند هرمهای فیبوناچی ساده شده در نظر گرفته شوند. آن‌ها انتخابات قوی برای پیاده‌سازی الگوریتم‌هایی مانند الگوریتم پریم در نظر گرفته می‌شوند[۲] و از توابع زیر پیروی می‌کنند (با فرض اینکه مین-هیپ باشند):

  • پیدا کردن کمینه: به سادگی به آیتم بالای هرم برمی‌گردیم.
  • ادغام: دو ریشه را با یک دیگر مقایسه می‌کنیم ریشهٔ کوچکتر ریشهٔ اصلی می‌شود و ریشهٔ بزرگتر و زیردرخت آن فرزند این ریشهٔ اصلی می‌شوند.
  • درج: یک هرم برای عنصر درج شده ساخته و آن را با هرم اصلی ادغام می‌کنیم.
  • کاهش کلید (اختیاری): زیرهرمی که ریشه اش این کلید است را حذف می‌کنیم کلید را با یک کلید کوچکتر جایگزین می‌کنیم سپس نتیجه را دوباره با هرم اصلی ادغام می‌کنیم.
  • حذف مینیمم: ریشه را حذف کرده و زیرهرمهایش را با یکدیگر ادغام می‌کنیم. استراتژی‌های مختلفی به کار گرفته می‌شوند.

تحلیل پیچیدگی زمانی هرمهای جفت شده ابتدا از درختان اسپلی الهام گرفته شده بود.[۱] اوردر سرشکن شدهٔ هر حذف کمینه الگو:ریاضی است و عملیات پیدا کردن حداقل، ادغام و درج در سرشکن شدهٔ الگو:ریاضی انجام می‌گیرد.[۳]

تعیین زمان دقیق مجانبی هرمهای جفت شده زمانی که یک عملیات کاهش کلید نیاز است کمی دشوار می‌باشد. در ابتدا به صورت تجربی حدس می‌زدند که زمان پیچیدگی این عملیات الگو:ریاضی باشد[۴] اما Fredman ثابت کرد که زمان سرشکن هر کاهش کلید برای توالی برخی عملیات حداقل Ω(loglogn) است.[۵] با استفاده از متدهای مختلف استدلال استهلاکی Pettie پس از آن ثابت کرد که درج کردن و کاهش-کلید همه در O(22loglogn) سرشکن می‌شوند که برابر o(logn) است.[۶] بعد از مدتی Elmasry یک نوع هرم جفت شده معرفی کرد که کاهش کلید آن به صورت سرشکن در O(loglogn) انجام می‌شود و سایر عملیات مطابق هیپ فیبوناچی است[۷] اما Θ(loglogn) برای داده ساختار اصلی درست نیست.[۳][۶] اما این یک سؤال بی جواب است که سرشکن o(logn) برای کاهش-کلید و O(1) برای درج کردن را می‌توان به طور همزمان داشت یا خیر.[۸]

اگر چه این بدتر از دیگر الگوریتم‌های صف اولویت مانند هیپ فیبوناچی است که انجام کاهش کلیدشان در سرشکن O(1) است ولی در عمل بسیار عالی می‌باشند. Stasko و Vitter[۴] Moret و shapiro[۹] و larkin و sen و Tarjan[۸] روی هرمهای جفت شده و سایر ساختمان‌های دادهٔ هرم آزمایش‌هایی انجام دادند. آنها به این نتیجه رسیدند که هرمهای جفت شده اغلب در عمل سریع تر از هیپ‌های دودویی مبتنی بر آرایه و هیپ دی تایی و تقریباً همیشه عملاً سریع ترند نسبت به دیگر هرمهای مبتنی بر اشاره گر از جمله ساختارهای داده‌ای مثل هیپ فیبوناچی که به لحاظ نظری کارآمد ترند.

ساختار

یک هرم جفت شده یا یک هرم خالی است یا یک جفت متشکل از یک ریشه و احتمالاً یک لیست خالی از هرمهای جفت شده. برای ویژگی ترتیب هرمها نیاز است که ریشه هیچ‌کدام از زیرهرمهای این هرم کوچکتر از ریشهٔ خود هرم نباشند. این توصیف صرفاً یک هرم تابعی را می‌دهد که از کاهش کلید پشتیبانی نمی‌کند. الگو:چپ‌چین

type PairingHeap[Elem] = Empty | Heap(elem: Elem, subheaps: List[PairingHeap[Elem]])

الگو:پایان چپ‌چین یک پیاده‌سازی بر اساس اشاره گر برای ماشین‌های رم می‌توان ارائه داد که از کاهش کلید پشتیبانی می‌کند. این پیاده‌سازی در هر گره سه اشاره گر دارد: یکی به اولین فرزندش، یکی به برادرش و یکی به پدرش. همچنین اشاره به پدر می‌تواند حذف شود اگر برگ‌ها به جای فرزند به ریشه اشاره کنند و یک پرچم برای برگ‌ها گذاشته شود تا تشخیص داده شوند؛ که این یک ساختار جمع و جور تر است با همان سربار قبلی.[۱]

عملیات‌ها

پیدا کردن مینیمم

تابع پیدا کردن مینیمم فقط ریشهٔ هرم را برمی‌گرداند:

الگو:چپ‌چین

function find-min(heap)

 if heap == Empty
 error
 else
 return heap.elem

الگو:پایان چپ‌چین

ادغام

ادغام با یک هرم خالی هرم اول را برمی‌گرداند در غیر این صورت یک هرم جدید برگردانده می‌شود که ریشهٔ کوچکتر ریشهٔ آن بوده و هرم با ریشهٔ بزرگتر به فرزندان آن اضافه می‌شود:

الگو:چپ‌چین

function merge(heap1, heap2)

 if heap1 == Empty
 return heap2
 elsif heap2 == Empty
 return heap1
 elsif heap1.elem < heap2.elem
 return Heap(heap1.elem, heap2 :: heap1.subheaps)
 else
 return Heap(heap2.elem, heap1 :: heap2.subheaps)

الگو:پایان چپ‌چین

درج

ساده‌ترین راه برای درج یک آیتم در یک هرم این است که هرم اصلی را با یک هرم که فقط عنصر درج شده در آن قرار دارد ادغام کنیم:

الگو:چپ‌چین

function insert(elem, heap)

 return merge(Heap(elem, []), heap)

الگو:پایان چپ‌چین

حذف

تنها عمل اساس غیر بدیهی در هرم حذف عنصر مینیمم از هرم می‌باشد. استراتژی استاندارد اول زیردرخت‌ها را ادغام می‌کند (نام گذاری این ساختمان داده به خاطر این مرحله می‌باشد) از چپ به راست و سپس لیست هیپ‌های نتیجه شده را از راست به چپ ادغام می‌کند:

الگو:چپ‌چین

function delete-min(heap)

 if heap == Empty
 error
 else
 return merge-pairs(heap.subheaps)

الگو:پایان چپ‌چین که این از تابع کمکی merge-pairs استفاده می‌کند:

الگو:چپ‌چین

function merge-pairs(l)

 if length(l) == ۰
 return Empty
 elsif length(l) == ۱
 return l[0]
 else
 return merge(merge(l[0], l[1]), merge-pairs(l[2.. ]))

الگو:پایان چپ‌چین که این در واقع ادغام دو طرفه چپ به راست و راست به چپ را پیاده‌سازی می‌کند:

 الگو:چپ‌چین

merge-pairs([H1, H2, H3, H4, H5, H6, H7])

=> merge(merge(H1, H2), merge-pairs([H3, H4, H5, H6, H7]))
 # merge H1 and H2 to H12, then the rest of the list
=> merge(H12, merge(merge(H3, H4), merge-pairs([H5, H6, H7])))
 # merge H3 and H4 to H34, then the rest of the list
=> merge(H12, merge(H34, merge(merge(H5, H6), merge-pairs([H7]))))
 # merge H5 and H6 to H56, then the rest of the list
=> merge(H12, merge(H34, merge(H56, H7)))
 # switch direction, merge the last two resulting heaps, giving H567
=> merge(H12, merge(H34, H567))
 # merge the last two resulting heaps, giving H34567
=> merge(H12, H34567)
 # finally, merge the first merged pair with the result of merging the rest
=> H1234567

الگو:پایان چپ‌چین

منابع

الگو:پانویس

پیوند به بیرون