נפה ריבועית
שיטת הנפה הריבועית היא שיטה מהירה לפירוק לגורמים של מספר שלם, המתאימה בעיקר למספרים בני 40-100 ספרות עשרוניות (שיטת רו של פולארד עדיפה לפירוק מספרים קטנים יותר, בעוד שבמספרים ארוכים יותר נפת שדה המספרים היא השיטה היעילה ביותר).
שיטת הנפה הריבועית, שהייתה השיטה הראשונה בעלת סיבוכיות תת-אקספוננציאלית לבעיית הפירוק לגורמים, פותחה על ידי קארל פומרנץ בשנת 1981. פומרנץ הרחיב, למעשה, רעיונות קודמים של קרייטצניק (Kraitchnik) וג'ון ד' דיקסון. זו הייתה השיטה המהירה ביותר (באופן אסימפטוטי), עד להמצאתה של נפת שדה המספרים, ב-1993.
עקרונות הנפה הריבועית
בדומה לשיטות פירוק אחרות של מספרים שלמים הנקראות "ריבוע אקראי" (Random square), גם שיטת הנפה הריבועית מנסה לאתר זוג שלמים אקראיים, ו-, כך ש- אבל (מודולו ). זוג כזה מאפשר לפרק את , משום ש- אולם אינו מחלק של או . על כן בסבירות גבוהה המחלק המשותף המקסימלי (שאותו אפשר למצוא ביעילות באמצעות אלגוריתם אוקלידס) יהיה גורם אמיתי של .
בקווים כלליים, הנפה הריבועית מבוססת על ייצור מספרים שיש להם הצגה ידועה כמספר ריבועי מחד, והצגת מספרים השקולים למכפלות של מספרים כאלה מודולו n כריבועים בעזרת פירוק חלקי לגורמים, מאידך.
נגדיר , ונחשב את (בסעיפים הבאים יוסבר כיצד למצוא xi מתאימים).
מתוך קבוצת ה-(Q(x אנחנו מחפשים תת-קבוצה של מספרים, שמכפלתם היא ריבוע ידוע: . ברגע שמטרה זו הושגה, מתקיים , ולכן גם ; כך מצאנו x ו-y מתאימים לדרישות. אם המספר n הוא מכפלה של שני מספרים ראשוניים גדולים, אז מתקיים בהסתברות ½ , ואז תוצאת ה-GCD תהיה n או 1; במקרה כזה יש להמשיך את תהליך הייצור, ולמצוא זוג אחר של מספרים כמתואר לעיל.
תיאור האלגוריתם
קלט: מספר פריק
פלט: אחד הגורמים הראשוניים של
- בחירת גורמי בסיס
בוחרים בסיס גורמים (Factor base) אותו מסמנים , כאשר ויתר האלמנטים הם המספרים הראשוניים החל מ- ועד לגבול מסוים, כאשר בוחרים רק את הראשוניים אשר הוא שארית ריבועית מודולו , כלומר שלהם סימן לז'נדר . הגבול העליון נקרא גבול חלקות (Smoothness bound) ולמעשה מכתיב את גודל בסיס הגורמים. הערך מייצג את מספר הגורמים הראשוניים בבסיס, מספר זה ישפיע ישירות על כמות הווקטורים הדרושים לפירוק ועל גודלם. - הכנת מאגר זוגות שלמים
מכינים זוגות שלמים , כדלהלן: מחשבים את השורש השלם של אותו מסמנים ב-. מכינים את הפולינום , מחשבים את ובודקים אם הוא מספר חלק ביחס ל- (כלומר שכל גורמיו הראשוניים נמצאים בבסיס הגורמים). אם אינו מספר חלק ביחס ל-, בוחרים אחר (את ערכי בוחרים לפי הסדר הזה: ). אם הוא אכן מספר חלק כלומר ניתן לייצגו כמכפלה של גורמי בסיס: , אזי מגדירים את , זוג הערכים נוסף למאגר.
הערה: ניתן לבדוק חלקות באמצעות חלוקה נסיונית (פשוט על ידי חלוקה בכל גורמי הבסיס), אולם שלב זה הוא הקריטי והארוך ביותר באלגוריתם ועל כן יעילותו מתבטאת בעיקר באופן בו מיושמת בדיקת החלקות. באופן מעשי מקובל להשתמש בתהליך הניפוי (sieving) המתואר להלן. - מציאת מספר ריבועי.
כיוון שהגורמים של כל ידועים, קל למצוא תת-קבוצה של אשר מכפלתה היא מספר ריבועי. כיוון שצריך רק מספרים שהחזקה של כל גורם שלהם זוגית. ניתן לפשט זאת על ידי הכנת וקטור בינארי של המעריכים מודולו כלומר כאשר . כיוון שבהכרח בוקטור זה תמצא תלות לינארית כלשהי, משתמשים באלגברה לינארית כדי למצוא תת-קבוצה של וקטורים שסכומה הוא וקטור האפס . משנמצאה תת-קבוצה כזו, מכפילים את המספרים החלקים המתאימים (שהמציין שלהם ב-), כלומר מחשבים את . בהכרח מתקיים ש- הוא מספר ריבועי. כמו כן ניתן לראות שגם מכפלת ריבועיהם של כל המתאימים יהיה מספר ריבועי. כלומר נותן מספר ריבועי. - חישוב x ו-y
מחשבים את השורשים הריבועיים של התוצאה האמורה, את האחד מוצאים על ידי חישוב שורש ריבועי של התוצאה האמורה ואת השני על ידי הכפלת מספרי המתאימים. מציבים ב- את וב- את השורש הריבועי של . מספרים אילו עונים על הדרישה האמורה, קרי ריבועיהם שקולים מודולו . בדרך כלל קיימות מספר תלויות לינאריות כך שבסבירות גבוהה אחת מהן תניב ו- כאלו שאינם שקולים עצמם מודולו . - מציאת גורם ראשוני
מחשבים את המחלק המשותף המרבי (gcd) של ההפרש של עם . התוצאה תהיה גורם ראשוני של , אשר עשוי להיות גם גורם טריויאלי כמו עצמו או 1. במקרה כזה, מנסים שוב עם תלות לינארית אחרת של המספרים החלקים.
הגדרת בסיס הגורמים וטווח הניפוי
שיטת הנפה דורשת דרך יעילה למצוא xi כך שהמכפלה תתן מספר ריבועי. כדי שזה יתקיים צריך שכל גורם ראשוני המחלק את המכפלה יחלק אותה מספר זוגי של פעמים. לשם כך צריך לפרק לגורמים ראשוניים כל אחד מן המספרים המרכיבים את המכפלה.
כדי לפשט את הבדיקה אנחנו מעוניינים ש- יהיו קטנים ככל האפשר, ויתחלקו בראשוניים מתוך קבוצת ראשוניים ידועה לנו – לקבוצה זו נקרא "בסיס הגורמים" ונסמן אותה ב-B. גודלה של הקבוצה B משפיע על ביצועי האלגוריתם, ולכל ווריאנט של הניפוי הריבועי יש לחשב את ה-B האופטימלי לפי גודלו של n.
לצורך הצגת האלגוריתם, נסמן , המספרים הראשוניים הקטנים, לפי סדרם. כדי ש-(Q(x יהיה קטן אנחנו צריכים לבחור x קטן, ואז . נבחר איפה את טווח הניפוי [M,1].
שיפורים אפשריים
א. אם נוסיף לבסיס הגורמים גם את 1-, אז מספרים הקרובים ל-n מלמטה גם הם מספרים קטנים יחסית, וניתן לבחור x שלילי ועדיין לקבל מספרים שקל יחסית לעבוד איתם. כך הכפלנו את טווח הניפוי ל- [M,M-].
ב. את בסיס הגורמים אפשר להקטין על ידי "בדיקת היתכנות": אם , אז , ולכן צריך להכניס לבסיס הגורמים רק מספרים ראשוניים המקיימים, עבור x מתוך טווח הניפוי, את התנאי . זוהי בדיקה שאפשר להשלים בסיבוכיות נמוכה, לפי משפט ההיפוך הריבועי של גאוס, שמאפשר לחשב את סימן לז'נדר בזמן לוגריתמי (בדומה לחישוב המחלק המשותף המקסימלי באלגוריתם אוקלידס).
הניפוי - Sieving
הגדרה: המספר X הוא חלק מעל הקבוצה B, אם כל הגורמים הראשוניים של X נמצאים ב- B
בשלב הניפוי אנחנו מחפשים שיכולים להיות חלק מקבוצה המרכיבה מכפלה שהיא מספר ריבועי. לשם כך אנחנו צריכים לעבור על x מתוך טווח הניפוי, לחשב את ולבדוק האם חלק מעל B. עבור כל אחד מהמספרים שמתקבלים, אנו צריכים לבדוק אם הוא חלק על ידי מעבר וחלוקה בבסיס הגורמים. פעולה סדרתית כזו אינה מעשית מבחינת זמן ביצוע. לכן נעבוד על בסיס הגורמים במקביל.
לכל p בבסיס הגורמים אם אז גם .
ובכיוון ההפוך, אם אז גם .
לכל p נפתור:
את המשוואה הזאת ניתן לפתור על ידי האלגוריתם של Shanks-Tonelli.
פתרון המשוואה הריבועית יתן לנו שני שורשים, נסמן אותם ב-
ו-.
מכאן אנו רואים ש- עבור מתוך טווח החיפוש מתחלק ב-p כאשר
עבור k שלם.
כעת בתהליך שמזכיר מאד את מכונת השרשראות של להמר, אנחנו מתקדמים ולכל x אנחנו בודקים באלו ראשוניים מבסיס הגורמים מתחלק. כדי לבצע את החיפוש הזה על כל טווח הניפוי כדאי לחלק את העבודה למספר מחשבים במקביל – וכל מחשב יקבל חלק מטווח הניפוי. אנחנו בודקים האם הוא חלק, אם לא – זורקים אותו ועוברים ל-x הבא, ואם כן שומרים אותו בתוך מטריצה שתוגדר בהמשך.
שיפורים אפשריים
פעולת חילוק היא פעולה מורכבת. לכן במקום לעשות חישוב מדויק, כדי לבדוק במהירות את ה”חלקות” של נבצע הערכה באמצעות מספר הביטים של הגורמים ש- מתחלק בהם, וכך במהירות גבוהה נקבל תוצאות וודאיות במרבית המקרים, ובאלו שיש לגביהם ספק – פשוט זורקים ועוברים הלאה, לא חסרים לנו מספרים לבדוק.
שלב המטריצה
בשלב זה אנחנו מחזיקים קבוצה של מספרים גדולים – ואנו צריכים למצוא קבוצה חלקית ל- כך שמכפלת האיברים של הקבוצה החלקית תתן מספר ריבועי. לשם כך נגדיר ווקטור של חזקות שמייצג מספרים חלקים מעל בסיס הראשוניים שלנו .
עבור m מספר חלק מעל B:
אנו צריכים למצוא קבוצת וקטורים המייצגים מספרים מתוך כך שמכפלת המספרים תהיה מספר ריבועי. לשם כך אנחנו צריכים למצוא קבוצת וקטורי חזקות שסכומם יתן וקטור שכל איבריו זוגיים.
נבנה מטריצה V של וקטורי חזקות, המייצגים חלקים מעל B. ועכשיו אנו צריכים למצוא וקטור בינארי e שהכפלתו במטריצה תתן וקטור שכל איבריו זוגיים.
על ידי החילוץ הגאוסיאני (gaussian elimination) ניתן לפתור את המטריצה, ולקבל וקטור e מתאים.
שיפורים אפשריים
כדי לעשות זאת קל יותר לחישוב נמיר את המטריצה V למטריצה בינארית V2. ואז וקטור e≠0 שמקיים e·V2=0 יהווה פתרון גם עבור המטריצה V.
חשוב לציין שוקטורים מודולו 2 אינם ייצוג חד חד ערכי של וקטורי חזקות – אולם החסכון בזמן חישוב ובגודל הזיכרון – אפילו שהוא לינארי בלבד – משתלם למרות הצורך בהחזקת מיפוי בין הווקטור הבינארי למספר המקורי שהוא מייצג.
דוגמאות לייצוג וקטורי מעל בסיס B
סיבוכיות
סיבוכיות הנפה הריבועית תלויה בגודל הבסיס B. בסיס ראשוניים הכולל את כל הראשוניים עד n יביא לכך שכל (Q(x הוא חלק מעל B, וכך תהיה הסיבוכיות למציאת (Q(x מתאים (O(1 בלבד. אולם, גודלה של המטריצה במקרה כזה (כ- (n/log(n ) אינו מאפשר לנו ליהנות מהיעילות של השלב הראשון. מצד שני, בסיס ראשוניים קטן מאד (למשל הראשוניים עד 1000) ייצר לנו מטריצה קטנה ונוחה לפתרון, ועל כך נשלם בסיכוי זעיר למצוא (Q(x חלק מעל B.
נסמן ב- (φ(X,B את מספר המספרים החלקים מעל B בטווח [X,1]. הסיכוי שמספר אקראי בתחום [X,1] יהיה חלק מעל B הוא φ(X,B)/X, ולכן כדי למצוא מספר אחד מתאים אנחנו צריכים לעבור על (X/φ(X,B מספרים אקראיים. חשוב להדגיש שהמספרים (Q(x אינם אקראיים במובן הפורמלי, אבל כל הניתוחים ההיוריסטיים של שיטות הפירוק נעשים בהנחה שהם אקראיים במידה מספקת. כיוון שאנחנו צריכים B|=k| מספרים כאלו לבניית המטריצה (כדי להבטיח פתרון לא טריוויאלי), אנחנו צריכים לעבור על (k·X/φ(X,B מספרים. עלות הבדיקה שמועמד (Q(x הוא חלק מעל B היא לינארית ב- B, ולכן העבודה הכוללת בייצור היא (k²·X/φ(X,B פעולות (של ניסיון חילוק מספר בגודל ½n בראשוני קטן).
פומרנץ הראה שכדי לקבל את המינימום עבור הנוסחה הזו צריך שהאיבר הגדול ב-B, אותו סימנו ב- pk, יהיה קרוב ל-
, והעבודה הנדרשת היא:
, כלומר כ- pk4.
אבל מהו X?
באלגוריתם הניפוי הריבועי אנחנו מייצרים (Q(x מסדר גודל של n½+ε כאשר ε קטן מאד (מכיוון שמספר המועמדים שנבדוק הוא חזקה קטנה של n); כלומר, .
לכן, סדר הגודל של שלב הניפוי הוא:
לאחר שהנתונים נאספו במטריצה (בגודל k על k), צריך למצוא פתרון לא טריוויאלי, בעלות של (O(k3 (האלימינציה של גאוס). למעשה החלק הזה של האלגוריתם הוא מהיר בהרבה, מכיוון שהמטריצה דלילה. בכל אופן מדובר בסיבוכיות זניחה ביחס לשלב איסוף המשוואות.
נפת שדה המספרים מול הנפה הריבועית
שיטה נוספת לפירוק מספר לגורמים היא נפת שדה המספרים (Number Field Sieving), זוהי שיטה מהירה יותר אסימפטוטית מהנפה הריבועית (Quadratic Sieve), אבל מסובכת הרבה יותר.
שתי השיטות מחולקות לשני שלבים: שלב הניפוי ושלב מטריצה.
סדר הגודל של שלב הניפוי בנפת שדה המספרים הוא: כאשר c משתנה לפי סוג הניפוי המדויק בו משתמשים (ברוב המימושים: ).
אסימפטוטית זהו סדר גודל קטן יותר משל הנפה הריבועית, אבל במספרים "קטנים" (עד 100 ספרות) עדיף להשתמש בנפה הריבועית בגלל המסובכות של נפת שדה המספרים.
מקורות
- Landquist, E., “MATH 488: Cryptographic Algorithms”, “The Quadratic Sieve Factoring Algorithm”, December 14, 2001
- Pomerance, C., “A Tale of Two Sieves”, December, 1996.
ראו גם
- TWINKLE - מכשיר אלקטרואופטי תאורטי, המממש את שיטת הנפה הריבועית לפירוק מהיר לגורמים.