אלגוריתם KMP
אלגוריתם KMP הוא אלגוריתם בתחום מדעי המחשב להתאמת תבניות הפועל בזמן ליניארי במקרה הגרוע. האלגוריתם התגלה בשנת 1970 על ידי דונלד קנות' ווואן פראת', ובאופן בלתי תלוי על ידי ג'יימס מוריס. השלושה פרסמו יחדיו את האלגוריתם בכתב העת SIAM בשנת 1977.
מטרתו של האלגוריתם היא למצוא מופעים של מחרוזת P (תבנית, pattern) בתוך מחרוזת T (טקסט), כאשר . בעוד האלגוריתם הנאיבי לביצוע פעולה זו פועל בזמן בזמן הגרוע, אלגוריתם KMP עובד בזמן בזמן גרוע.
תיאור האלגוריתם
עריכהנניח וגם
האלגוריתם הנאיבי להתאמת מחרוזות, יחפש עבור כל מיקום בטקסט האם הוא התחלה של המחרוזת. אלגוריתם זה עובד בזמן . אלגוריתם KMP משתמש בטרנזיטיביות של התאמת מחרוזות על מנת לחסוך השוואות מיותרות. בכל פעם שהאלגוריתם יתקל בנקודת חוסר התאמה בטקסט, הוא יחזור להתאים את הרישא הארוכה ביותר של התבנית המתאימה לסיפא הארוכה ביותר של הטקסט בנקודה בו נתקלנו בחוסר התאמה.
על מנת להתאים בין הרישא הארוכה ביותר לסיפא הארוכה ביותר, נבנה טבלה ייעודית ונוכל להציץ בה בזמן קבוע.
הסבר ויזואלי
עריכהכאשר ניתקל בתבנית ABCABCD ובטקסט ABCABCABCABCABCABCD, נקודת אי ההתאמה הראשונה תהיה באינדקס 6.
ניתן לשים לב, שהאלגוריתם הנאיבי יתחיל לחפש מהאינדקס הבא (1), למרות שניתן לדלג מידית לאינדקס 3. KMP יודע לדלג לאינדקס 3, מכיוון שלאחר שהתאמנו את A באינדקס 0, B באינדקס 1 ו-C באינדקס 2, ברור שלא נוכל להתאים את A באינדקסים 1 או 2.
בנוסף, לאחר שנתחיל את הבדיקה מחדש באינדקס 3, אין טעם לבדוק ששלושת התווים הראשונים זהים, מכיוון שווידאנו זאת כבר קודם, כלומר המחרוזת שהותאמה עד כה היא ABCABC, ומחרוזת זו נגמרת ב-ABC. לכן, יש סיפא באורך 3 (ABC) שמתאימה לרישא באורך 3 (גם כן, ABC), ולכן ניתן לחסוך את ההתאמה של שלושת התווים הראשונים.
על ידי התאמת הרישא לסיפא, אנחנו הלכה למעשה מוודאים שלא ייווצר מצב בו אנו מדלגים על התאמה. ניתן לחלק לשני מקרים:
- אם קיימת סיפא שמתאימה לרישא, אז אי אפשר לדלג על כל אזור אי ההתאמה, אבל בתוך הדילוג אנחנו כבר יודעים שהרישא שווה לסיפא, ולכן אין צורך לבדוק אותה.
- אם לא קיימת רישא שמתאימה לסיפא, אזי בכל מקום אליו נקפוץ בטווח שכרגע בדקנו לא תמצא התאמה[1], ולכן ניתן לדלג ישירות על כל החלק המאומת.
מהגדרה זאת ניתן לקבל את ההרגשה שהאלגוריתם פועל בזמן ליניארי, בהנחה שפעולות הבדיקה על הרישא והסיפא פועלות בזמן קבוע.
פסאודו קוד
עריכהחיפוש
עריכהקלט - T, P אורך תבנית - m, אורך טקסט - n
- בנה טבלת סיפות A עבור התבנית P
- כל עוד בצע:
- כל עוד וגם בצע:
- אם j=m, דווח על התאמה באינדקס i-j
- אם i=n, סיים
- כל עוד וגם בצע:
בניית טבלה
עריכהקלט - T באורך n
- צור מערך A בגודל n עם אפסים
- שים באיבר הראשון במערך -1
- לכל בצע:
- כל עוד בצע
- אם צא מהלולאה
- אחרת
הוכחת נכונות
עריכהאלגוריתם
עריכהנוכיח שכל התאמה אפשרית במחרוזת נמצאת על ידי האלגוריתם, בהנחה שהטבלה הנוצרת נכונה.
תהא התאמה באינדקס k, כלומר .
על מנת שתימצא התאמה באינדקס k, צריך להתחיל את הלולאה הפנימית עם ערכי i ,j המקיימים . נשים לב שההפרש בין i ל-j משתנה בשתי אפשרויות -
קל לראות שבדיוק אחד מהמקרים האלה מתרחש בכל פעם[2]. במקרה הראשון, ההפרש גדל בדיוק ב-1, ולכן לא נדלג על שום התאמה.
במקרה השני, ההפרש גדל בדיוק ב- וגם j לא שווה 0. נניח בשלילה שקיים אינדקס התאמה k המקיים , זאת אומרת ש . בנוסף, מכיוון שהאלגוריתם הגיע לאינדקס j, מתקיים . לכן, מתקיים , כלומר אורך הרישא הארוכה ביותר שמתאימה לסיפא הארוכה ביותר באינדקס j היא לכל הפחות , ולפי הגדרת הטבלה בסתירה להנחת השלילה.
ההוכחה שהאלגוריתם לא מחזיר התאמות שאינן נכונות פשוט למדי - מכיוון שהאלגוריתם משתמש בטרנזיטיביות של שוויון תווים, אזי אם התאמנו כבר סיפא אין צורך לבדוק מחדש את הרישא, ועל מנת שהאלגוריתם יודיע על התאמה יש לבדוק את כל האיברים שאחרי הרישא איבר-איבר, וזהו האלגוריתם הנאיבי.
טבלה
עריכהנרצה להוכיח שבהינתן מחרוזת האלגוריתם מוציא את טבלת הסיפות הנכונה, ו-(-1) במקום הראשון.
הערה - קל לראות . נניח שהאלגוריתם עובד עבור ערכים קטנים מ-i, נוכיח שהוא עובד עבור i -
מקרה ראשון - אם מתקיים וגם הוא אינדקס שהסיפא הארוכה ביותר הנגמרת אצלו המתאימה לרישא הארוכה ביותר זהה לסיפא באינדקס . נוכיח שאכן . הערה - לפי ההגדרה, מקיים את זה. קל לראות שמתקיים [3]. נניח שמתקיים , זאת אומרת . אבל אם נוריד את התו האחרון מהסיפא והתו האחרון מהרישא, אזי יש לנו התאמה באורך לכל הפחות באינדקס , בסתירה להנחה שהאלגוריתם צודק עבור ערכים קטנים מ-i ובפרט .
מקרה שני - . במקרה זה, מכיוון שלפי הגדרה וגם הוא חסם עליון לאורך הסיפא המקסימלית[4], אזי מתקיים בוודאות . ואכן, .
מקרה שלישי - אם מתקיים . במקרה זה, התו החדש שנוסף אינו המשך של הרישא הקודמת, ולכן נחפש רישא קצרה יותר. מכיוון שהרישא הקודמת שווה לסיפא הקודמת, אזי הסיפא הארוכה ביותר המתאימה לרישא הארוכה ביותר ברישא הקודמת, היא גם סיפא של הסיפא הקודמת. מההנחה שהאלגוריתם פועל עבור קלטים קטנים, מתקיים שאורך הסיפא החדשה המתקבלת הוא המקסימלי, ולכן כל לא יכול להיות אורך הסיפא שאינה כוללת את התו האחרון. כלומר - גם אם האלגוריתם לא ימצא בשלב זה את אורך הסיפא הוא יתחיל באיטרציה הבאה עם , ולכן האלגוריתם תמיד יגיע בסופו של דבר למקרה הראשון או השני.
נוכיח שלא קיים כך ש-k הוא האורך של הסיפא הארוכה ביותר הנגמרת ב- ומתאימה לרישא. אם זה אכן היה מתקיים, אז לפי הגדרה , אבל מכיוון ש- הוא אורך הסיפא הארוכה ביותר המתאימה ל- , וגם לפי הטרנזיטיביות הסיפא הנגמרת ב- זהה לסיפא הנגמרת ב- , אז למעשה , וזאת אומרת שאורך הסיפא המקסימלית הנגמרת ב- היא לפחות k, בסתירה לכך שהנחנו שהאלגוריתם צודק עבור קלטים קטנים[5]. זאת גם למעשה ההוכחה ש- הוא חסם עליון לאורך הסיפא, שהרי עבור כלשהו.
הוכחת זמן ריצה
עריכהזמן הריצה הכולל של האלגוריתם הוא , כאשר הזמן הדרוש לבניית טבלה של התבנית הוא וזמן החיפוש הדרוש הוא . נוכיח:
אלגוריתם
עריכהכאשר אנחנו מבצעים את החיפוש עצמו, בכל איטרציה של כל אחת מהלולאות יש לפחות קידום אחד של i, כלומר לא קיימת איטרציה בה i לא קודם. בנוסף, כל איטרציה של הלולאות מפעילות פעולות, ולכן בסך הכל יש לכל היותר פעולות.
טבלה
עריכהנסמן מונה C1 הסופר כמה פעמים עודפות האלגוריתם ביקר בתא בודד, ומונה C2 הסופר כמה פעמים האלגוריתם ביקר פעם בודדת. מתקיים מכיוון שיש n+1 תאים בטבלה. כפי שצוין קודם, מתקיים , וגם אם ורק אם קיימת איטרציה אחת בדיוק (כלומר C2 מקודם). בנוסף, מתקיים לפי הגדרת הרישא. לכן, ניתן לכתוב , כלומר במקרה הגרוע יש לולאה יורדת. נחבר את התנאים - על מנת שיתקיים צריך שיתקיים (לפי התנאי הראשון), ועל מנת שנוכל לקדם את C1, צריך שיתקיים . בסך הכל, על מנת לקדם את C1 יש לקדם את C2, ולכן מתקיים , ומספר האיטרציות הכולל קטן מ- , או , כנדרש.
דוגמה
עריכההרחבות
עריכהקישורים חיצוניים
עריכההערות שוליים
עריכה- ^ אם הייתה התאמה בנקודה כלשהי, אותה התאמה בהכרח מכילה כל רישא של התבנית, אבל מכיוון שהיא בטווח שסיימנו לבדוק, אז היא מכילה גם סיפא
- ^ מכיוון שאם המינימום הוא לא 0, אזי j=0 בוודאות, בעוד התנאי השני מתקיים רק כאשר j אינו שווה ל-0, דבר הנובע ישירות מהגדרת הטבלה
- ^ מכיוון שהתו האחרון נבדק ידנית, אז הסיפא הקודמת בצירוף התו החדש היא סיפא חוקית, והרישא בצירוף תו היא גם כמובן רישא חוקית
- ^ הוכחה בהמשך
- ^ דרשנו במפורש ש-k יהיה גדול מהסיפא, אבל k עצמו הוא סיפא