לדלג לתוכן

בעיית הסכומים החלקיים

מתוך ויקיפדיה, האנציקלופדיה החופשית

במדעי המחשב, בעיית הסכום החלקי (Subset Sum Problem) היא בעיה חשובה בתורת הסיבוכיות ובקריפטוגרפיה. הבעיה היא כזו: בהינתן קבוצה של מספרים שלמים, האם קיימת תת-קבוצה לא ריקה שלה שסכום איבריה הוא אפס? לדוגמה, בהינתן הקבוצה {7-,3-,2-,8,5} כקלט לבעיה, התשובה תהיה חיובית, והקבוצה שסכומה אפס תהיה {2-,3-,5}. בעיה זו היא NP-שלמה.

בעיה שקולה לבעיה זו היא כדלקמן: בהינתן קבוצת מספרים שלמים I, ומספר נוסף s, האם קיימת תת-קבוצה לא ריקה של I שסכום איבריה הוא s?

ניתן גם לחשוב על בעיית הסכום החלקי כעל מקרה מיוחד של בעיית תרמיל הגב.[1] מקרה מיוחד של בעיה זו היא בעיית החלוקה(אנ'), בה נשאלת השאלה האם קיימת תת-קבוצה שסכום איבריה שווה לסכום כל האיברים שאינם בתת-קבוצה זו. הבעיה נכללה ב-21 הבעיות ה-NP שלמות של קארפ, שם היא נקראת "בעיית תרמיל הגב" (כיום, כאמור, שם זה משמש לבעיה אחרת).

אלגוריתמים לפתרון הבעיה

[עריכת קוד מקור | עריכה]

בעיה זו היא NP-שלמה, לכן עדיין לא נמצא אלגוריתם הפותר אותה בזמן יעיל. לעומת זאת, ניתן לפתור את הבעיה באמצעות אלגוריתם פסאודו פולינומי, ובאמצעות אלגוריתם פולינומי מקרב.

אלגוריתם לפתרון בזמן אקספוננציאלי

[עריכת קוד מקור | עריכה]

יש מספר דרכים לפתור את בעיית הסכום החלקי בזמן אקספוננציאלי בגודל הקבוצה. האלגוריתם הנאיבי ביותר יבדוק את כל תתי הקבוצות האפשריות, ויבדוק האם סכומן מתאים להגדרת הבעיה. זמן הריצה של אלגוריתם כזה יהיה , שכן ישנן תתי קבוצות שהאלגוריתם יבדוק, ובכל קבוצה כזו יש לכל היותר איברים שהאלגוריתם יסכום.

קיים גם אלגוריתם טוב יותר שרץ בזמן שהוצג לראשונה על ידי הורוביץ וסהני(אנ') בשנת 1972. האלגוריתם פועל באופן הבא:

  1. נחלק את הקלט לקבוצות T ו-U בגודל כל אחת.
  2. נחשב את הקבוצות A ו-B כאשר A היא קבוצת כל הזוגות הסדורים כך ש- הוא קידוד של בחירת איברים ב-T שסכומם הוא ובאותו אופן B היא קבוצת הזוגות עבור U.
  3. נמיין את A ו-B בסדר עולה.
  4. נחפש זוג ו- כך שמתקיים .

אלגוריתם זה דורש מקום מאחר ש-A ו-B הן בגודל כזה כל אחת (הן מייצגות את קבוצות החזקה של קבוצות בגודל ). כדי להראות את הטענה לגבי זמן הריצה, צריך להראות שניתן לבנות את הקבוצות באופן ממוין בזמן ולחפש זוג מתאים בזמן דומה. הזמן של המיון נובע מהדרך בה בונים את A ו-B. את החיפוש ניתן לבצע בזמן הדרוש באופן הבא:

  1. נסמן
  2. אם , נסמן
  3. אחרת, אם , נסמן
  4. אחרת, נחזיר את איחוד הקבוצות המקודדות על ידי ו-

אלגוריתם לפתרון בזמן פסאודו פולינומי

[עריכת קוד מקור | עריכה]

פתרון בזמן פסאודו פולינומי אומר שעבור בעיה זהה בה הקלט הוא בבסיס אונרי קיים פתרון הרץ בזמן פולינומי.

הפתרון לבעיה הזו משתמש בתכנון דינמי. נניח שהקלט הוא מהצורה , ועלינו לבדוק האם קיימת תת-קבוצה שסכום איבריה הוא אפס. יהא הסכום של כל האיברים השליליים בקבוצה, ויהא הסכום של כל האיברים החיוביים בה.

נגדיר את הפעולה הבוליאנית להיות התשובה לשאלה: "האם קיימת תת-קבוצה של שסכום איבריה הוא  ?". ברור כעת כי פתרון הבעיה הוא .

נבחין כי עבור מתקיים . כעת, ניצור טבלה בגודל , כאשר הערך בנקודה יהיה , ונמלא את הטבלה מלמטה למעלה ומשמאל לימין באמצעות הרקורסיה הבאה:

  • מקרה הבסיס:
  • עבור :

מילוי כל תא לוקח זמן כי ערכי הטבלה שהוא משתמש בהן בפסוק הלוגי ידועים כבר מחישובים קודמים, ולכן זמן מילוי הטבלה יהיה .

הפתרון לא נחשב פולינומי בתורת הסיבוכיות כיוון שהערך אינו פולינומי בגודל הבעיה, שהוא מספר הביטים שדרוש כדי לייצג אותם. האלגוריתם הוא פולינומי בערכים של ושל , שהם אקספוננציאליים במספר הביטים שלהם.

אלגוריתם קירוב בזמן פולינומי

[עריכת קוד מקור | עריכה]

אלגוריתם קירוב לבעיית הסכום החלקי הוא כדלהלן: בהינתן קבוצת מספרים , מספר s, ומספר c>0 כלשהו, נרצה שהפלט יהיה:

  • "לא", אם לא קיימת תת-קבוצה שמסתכמת למספר בין ל.
  • "כן", אם לא קיימת תת-קבוצה שמסתכמת ל.
  • "כן" \ "לא", אחרת.

אם כל המספרים הם לא שליליים, ניתן לפתור זאת בזמן פולינומי ב- וב-.

נשים לב שבבעיה המקורית, אם כל סכום אפשרי של מספרים יכול להיכתב באמצעות P ביטים, אם נפתור את הבעיה המקורית באלגוריתם הקירוב עם , נפתור את הבעיה בדיוק. לכן אלגוריתם זה הוא פתרון מדויק של הבעיה כשהוא רץ בזמן פולינומי ב- וב- (כלומר, אקספוננציאלי ב-).

האלגוריתם המקורב לבעיה הוא:

initialize a list S to contain one element 0.
 for each i from 1 to n do
 let T be a list consisting of xi + y, for all y in S
 let U be the union of T and S
 sort U
 make S empty
 let y be the smallest element of U
 add y to S
 for each element z of U in increasing order do
 //trim the list by eliminating numbers close to one another
 //and throw out elements greater than s
 if y + cs/n < z ≤ s, set y = z and add z to S
 if S contains a number between (1 − c)s and s, output yes, otherwise no

האלגוריתם רץ בזמן פולינומי כיוון שהרשימות S, T, U תמיד בגודל פולינומי ב-n וב-, ולכן גם כל הפעולות שנעשות עליהן מתבצעות בזמן פולינומי. גודל הרשימות אכן נשאר פולינומי היות שאנו מוסיפים עצמים ל-S אם הם גדולים מהעצם הקודם ב- וקטנים מ-. הצעד שמוודא שההפרש בין כל שני עצמים שמתווספים הוא לפחות מוודא שברשימה יש לכל היותר עצמים.

האלגוריתם נכון כיוון שבכל צעד מתווספת שגיאה של לכל היותר , ובמהלך שלבים מתווספת שגיאה של לכל היותר .

לקריאה נוספת

[עריכת קוד מקור | עריכה]

קישורים חיצוניים

[עריכת קוד מקור | עריכה]

הערות שוליים

[עריכת קוד מקור | עריכה]
  1. ^ Martello, Silvano; Toth, Paolo (1990). "4 Subset-sum problem". Knapsack problems: Algorithms and computer interpretations. Wiley-Interscience. pp. 105–136. ISBN 0-471-92420-2. MR 1086874.