هرم کمینه-بیشینه

از ویکی‌پدیا، دانشنامهٔ آزاد

در علوم رایانه، هرم کمینه بیشینه یک درخت دودویی کامل است که فواید هرم کمینه و هرم بیشینه را با هم ترکیب کرده است. این ساختمان داده به ما این امکان را می دهد تا عنصر بیشینه و کمینه را

در زمان ثابت ((1)O) بازگرداند یا در زمان لگاریتمی((logn)O) آن ها را حذف کنیم.[۱] این قابلیت هرم کمینه بیشینه را به یک ساختمان داده مفید برای پیاده سازی یک صف اولویت دو طرفه تبدیل می کند.

همانند هرم کمینه و هرم بیشینه دودویی، هرم کمینه بیشینه عملیات های درج و حذف را در زمان لگاریتمی انجام می دهد و در زمان خطی ساخته می شود.[۲] هرم کمینه بیشینه معمولا در آرایه ساخته می شود [۳] به همین دلیل به عنوان یک ساختمان داده ضمنی از آن یاد می شود.

 مشخصه ی یک هرم کمینه بیشینه :

هر گره در سطح زوج در درخت دودویی متناظر با هرم از تمام گره های موجود در زیردرخت خود کوچکتر است درحالی که هر گره در سطح فرد در درخت متناظر با هرم از تمام گره های زیردرخت خود بزرگتر است.[۳]

این ساختار همچنین می تواند به گونه ای تعمیم داده شود تا عملیات هایی مثل پیدا کردن میانه(find-median) و (delete-median) حذف کردن میانه[۱] ، پیدا کردن kامین کوچکترین مقدار در ساختار((find(k) و عملیات حذف کردن kامین کوچکترین مقدار در ساختار((delete(k) را به ازای یک مقدار مشخص یا مجموعه ای از مقادیر k انجام دهد. این داده ساختار دو عملیات انتهایی را میتواند به ترتیب در زمان های خطی و لگاریتمی انجام دهد. مفهوم ترتیب کمینه بیشینه میتواند به ساختارهای دیگری که بر پایه ترتیب بیشینه یا کمینه هستند گسترش پیدا کند مانند درخت چپ گرا (leftist trees) که منجر به ایجاد یک دسته جدید و قدرتمند از داده ساختارها می شود.[۳] هرم کمینه بیشینه همچنین برای پیاده سازی مرتب سازی سریع خارجی نیز مفید است. [5]

توضیحات[ویرایش]

معرفی[ویرایش]

  • هرم کمینه بیشنه یک درخت دودویی کامل است که شامل سطوح کمینه (زوج) و بیشینه (فرد) است. سطوح زوج 0،2،4،... و سطوح فرد 1،3،5،... . همچنین فرض می کنیم عنصر ریشه در سطح اول (0) قرار دارد.
مثالی از هرم کمینه-بیشینه
  • عنصر ریشه کوچکترین عنصر موجود در هرم کمینه بیشینه است.
  • یکی از دو گره موجود در سطح دوم که سطح بیشینه (فرد) است، عنصر بیشینه در هرم کمینه بیشینه است.
  • فرض کنید x یک گره دلخواه در هرم کمینه بیشینه است. در این صورت :
    • اگر x در سطح کمینه (یا زوج) قرار داشته باشد در اینصورت کلید آن از کلید همه عناصر موجود در زیردرختی که ریشه آن x است کمتر است.
    • اگرx  در سطح بیشینه (یا فرد) قرار داشته باشد در اینصورت کلید آن از کلید همه عناصر موجود در زیردرختی که ریشه آن x است بیشتر است.

هرم بیشینه کمینه به صورت معکوس تعریف می شود. در این هرم مقدار بیشینه در ریشه ذخیره می شود و مقدار کمینه در یکی از دو گره فرزند ریشه ذخیره می شود.

عملیات[ویرایش]

در عملیات های زیر فرض می کنیم که هرم کمینه بیشینه در یک آرایه قرار دارد. خانه ی iام آرایه به گرهی اشاره میکند که در سطح قرار دارد.

ساخت هرم[ویرایش]

ساختن یک هرم کمینه بیشینه با انجام تغییراتی در الگوریتم ساخت هرم Floyd در زمان خطی انجام می شود که روشی از پایین به بالا است.[۳] الگوریتم ساخت هرم Floyd[۴] به صورت زیر است :

function FLOYD-BUILD-HEAP(h):

    for each index i from down to 1 do:
        push-down(h, i)
    return h

در این تابع، h آرایه اولیه است که لزوما عناصر آن خواص هرم کمینه بیشینه را ندارد. عملیات push-down (که گاهی heapify گفته می شود) برای یک هرم کمینه بیشینه در قسمت بعد توضیح داده شده است.

پایین بردن مقدار در هرم (push-down)[ویرایش]

الگوریتم push-down (یا trickle-down همانطور که در [۳] گفته شده) به صورت زیر است :

function PUSH-DOWN(h, i):
    if i is on a min level then:
        PUSH-DOWN-MIN(h, i)
    else:
        PUSH-DOWN-MAX(h, i)
    endif

پایین بردن مقدار کمینه (push down min)[ویرایش]

function PUSH-DOWN-MIN(h, i):
    if i has children then:
        m := index of the smallest child or grandchild of i
        if m is a grandchild of i then:
            if h[m] < h[i] then:
                swap h[m] and h[i]
                if h[m] > h[parent(m)] then:
                    swap h[m] and h[parent(m)]
                endif
                PUSH-DOWN-MIN(h, m)
            endif
        else if h[m] < h[i] then:
            swap h[m] and h[i]
        endif 
    endif

پایین بردن مقدار بیشینه (push down max)[ویرایش]

الگوریتم مربوط به push-down-max دقیقا مشابه الگوریتم push-down-min است با این تفاوت که جهت عملگرهای مقایسه را برعکس می کنیم.

function PUSH-DOWN-MAX(h, i):
    if i has children then:
        m := index of the largest child or grandchild of i
        if m is a grandchild of i then:
            if h[m] > h[i] then:
                swap h[m] and h[i]
                if h[m] < h[parent(m)] then:
                    swap h[m] and h[parent(m)]
                endif
                PUSH-DOWN-MAX(h, m)
            endif
        else if h[m] > h[i] then:
            swap h[m] and h[i]
        endif 
    endif

فرم پیمایشی[ویرایش]

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

function PUSH-DOWN-MIN-ITER(h, m):
    while m has children then:
        i := m
        m := index of the smallest child or grandchild of i
        if m is a grandchild of i then:
            if h[m] < h[i] then:
                swap h[m] and h[i]
                if h[m] > h[parent(m)] then:
                    swap h[m] and h[parent(m)]
                endif
            endif
        else if h[m] < h[i] then:
            swap h[m] and h[i]
        endif 
    endwhile

درج یک عنصر[ویرایش]

برای درج یک عنصر در هرم کمینه-بیشینه باید اعمال زیر را انجام دهیم:

  1. کلید عنصر را در انتهای آرایه ای که هرم کمینه بیشینه در آن نگهداری می شود قرار می دهیم. این کار به احتمال زیاد ویژگی هرم کمینه بیشینه را در آرایه از بین می برد، به همین دلیل آرایه را باید دوباره سازگار کنیم.
  2. کلید این عنصر را با پدرش مقایسه می‌کنیم :
    1. اگر کوچکتر (بزرگتر) بود مطمئناً از کلید تمام گره‌های سطح بیشینه (کمینه) که از مکان فعلی کلید تا ریشه هرم قرار دارد کوچکتر است. حال کافی است تا با گره‌های سطح کمینه (بیشینه) مقایسه شوند.
    2. مسیر از عنصر جدید تا ریشه (با در نظر گرفتن تنها سطوح کمینه (بیشینه)) باید به صورت نزولی (صعودی) به همان صورتی که قبل از درج قرار داشت، باشد. پس باید با استفاده از درج دودویی عنصر جدید را در رشته قرار دهیم. روش ساده تر این است که عنصر جدید را تا زمانی که از پدرش کوچکتر (بزرگتر) است با پدرش جابه جا کنیم.

بالابردن مقدار (Push up)[ویرایش]

الگوریتم push-up (یا Bubble-up همانطور که در [۵] گفته شده است) به صورت زیر است.

function PUSH-UP(h, i):
    if i is not the root then:
        if i is on a min level then:
            if h[i] > h[parent(i)] then:
                swap h[i] and h[parent(i)]
                PUSH-UP-MAX(h, parent(i))
            else:
                PUSH-UP-MIN(h, i)
            endif
        else:
            if h[i] < h[parent(i)] then:
                swap h[i] and h[parent(i)]
                PUSH-UP-MIN(h, parent(i))
            else:
                PUSH-UP-MAX(h, i)
            endif
        endif
    endif
بالابردن مقدار کمینه (push up min)[ویرایش]

function PUSH-UP-MIN(h, i):

    if i has a grandparent and h[i] < h[grandparent(i)] then:
        swap h[i] and h[grandparent(i)]
        PUSH-UP-MIN(h, grandparent(i))
    endif
بالابردن مقدار بیشینه (push up max)[ویرایش]

الگوریتم push-up-max همانند الگورریتم push-up-min است با این تفاوت که جهت عملگر های مقایسه را برعکس می کنیم.

function PUSH-UP-MAX(h, i):
    if i has a grandparent and h[i] > h[grandparent(i)] then:
        swap h[i] and h[grandparent(i)]
        PUSH-UP-MAX(h, grandparent(i))
    endif
فرم پیمایشی[ویرایش]

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

function PUSH-UP-MIN-ITER(h, i):
    while i has a grandparent and h[i] < h[grandparent(i)] then:
        swap h[i] and h[grandparent(i)]
        i := grandparent(i)
    endwhile

مثالی از درج یک عنصر در هرم کمینه-بیشینه.[ویرایش]

ما هرم کمینه-بیشینه زیر را داریم و می‌خواهیم عنصری با مقدار ۶ را در آن درج کنیم.

Example of Min-max heap

در ابتدا عنصر ۶ را در مکان مشخص شده با j قرار می‌دهیم. عنصر ۶ از پدرش کوچکتر است لذا از تمام عناصر سطوح بیشینه کمتر است. بابراین عنصر ۶ در مکان ریشه درخت قرار می‌گیرد و عنصر قبلی ریشه یک گام به پایین نقل مکان می‌کند.

اگر بخواهیم یک عنصر جدید با مقدار ۸۱ را در هرم مفروض قرار دهیم، مانند قبل جلو می‌رویم. در ابتدا عنصر ۸۱ را در مکان مشخص شده با j قرار می‌دهیم. از آنجا که عنصر ۸۱ از پدرش و تمام عناصر سطوح کمینه بیشتر است کافی است گره‌های سطح بیشینه بررسی شوند.

حذف کوچکترین کلید[ویرایش]

برای حذف کوچکترین کلید در هرم کمینه-بیشینه باید اعمال زیر را انجام دهیم.

گره درخت کوچکترین عنصر است.

  1. گره ریشه را حذف می‌کنیم و فرض می‌کنیم x گره‌ای باشد که در انتهای هرم قرار دارد.
  2. حال x را در هرم کمینه-بیشینه دوباره درج می‌کنیم.

برای دوباره درج کردن دو وضعیت به وجود می‌آید.

اگر ریشه هیچ فرزندی نداشته باشد، پس x می‌تواند در ریشه قرار گیرد.

فرض کنید ریشه حداقل یک فرزند داشته باشد. مقدار کمینه را پیدا می‌کنیم و این عنصر را m می‌نامیم. m یکی از فرزندان یا نوادگان ریشه است. حال شرایط زیر باید در نظر گرفته شود:

  1. اگر x.key <= h[m].key آنگاه x باید در مکان ریشه قرار گیرد.
  2. اگر x.key> h[m].key و m فرزند ریشه باشد چون m در سطح بیشینه است، هیچ فرزندی ندارد بنابراین عنصر [h[m باید در مکان ریشه و عنصر x در گره m قرار گیرد.
  3. اگر x.key> h[m].key و m از نوادگان فرزند ریشه باشد آنگاه باید عنصر [h[m در مکان ریشه قرار گیرد. حال فرض کنیم p پدر m باشد، اگر x.key> h[p].key آنگاه جایگاه [h[p و x باید عوض شود.

کد زیر حذف عنصر با کوچکترین کلید را پیاده‌سازی می‌کند.

element del_min(element heap[], int *s) { // *s: capacity of the heap
    int i, last, m, parent;
    element temp, x;

    if ((*s) == 0) {
        heap[0].key = INT_MAX;
    
        return(heap[0]);
    }

    heap[0] = heap[1];
    x = heap[(*s)--];

    /* find place to insert x */
    for (i = 1, last = (*s) / 2; i <= last;  ) {
        m = min_child_grandchild(i, *s);
    
        if (x.key <= heap[m].key) // case 1
            break;
    
        heap[i] = heap[m]; // case 2 or 3
    
        if (m <= (2 * i) + 1) { // case 2
            i = m;
        
            break;
        }
    
        /* case 3 */
        parent = m / 2;
    
        if (x.key > heap[parent].key)
            SWAP(heap[parent], x, temp);
    
        i = m;
    }

    heap[i] = x;

    return heap[0];
}

منابع[ویرایش]

  1. ۱٫۰ ۱٫۱ Atkinson, M. D.; Sack, J.-R.; Santoro, B.; Strothotte, T. (1986-10-01). "Min-max heaps and generalized priority queues". Communications of the ACM. 29 (10): 996–1000. doi:10.1145/6617.6621.
  2. Atkinson, M. D.; Sack, J.-R.; Santoro, B.; Strothotte, T. (1986-10-01). "Min-max heaps and generalized priority queues". Communications of the ACM. 29 (10): 996–1000. doi:10.1145/6617.6621.
  3. ۳٫۰ ۳٫۱ ۳٫۲ ۳٫۳ ۳٫۴ Atkinson, M. D.; Sack, J.-R.; Santoro, B.; Strothotte, T. (1986-10-01). "Min-max heaps and generalized priority queues". Communications of the ACM. 29 (10): 996–1000. doi:10.1145/6617.6621.
  4. یادکرد خالی (کمک)
  5. Atkinson, M. D.; Sack, J.-R.; Santoro, B.; Strothotte, T. (1986-10-01). "Min-max heaps and generalized priority queues". Communications of the ACM. 29 (10): 996–1000. doi:10.1145/6617.6621.

Loyola University Chicago , MIN-MAX HEAP AND DEAP DATA STRUCTURE A Research Report on MIN MAX HEAP AND DEAP DATA STRUCUTRE, February, 2014, MIN MAX HEAP AND DEAP DATA STRUCUTRE