एल्गोरिथ्म्स और डेटा स्ट्रक्चर्स

Recursion (रिकर्शन)

जब हम प्रोग्राम लिखते हैं, तो आमतौर पर किसी विशेष काम को करने के लिए अलग-अलग फ़ंक्शन को कॉल करते हैं। उदाहरण के लिए, किसी संख्या का वर्गमूल निकालने के लिए हम sqrt को कॉल कर लेते हैं, या अधिकतम/न्यूनतम मान खोजने के लिए max/min फ़ंक्शन को। लेकिन रिकर्शन उस प्रक्रिया को कहते हैं जिसमें वही फ़ंक्शन खुद को ही दोबारा कॉल करता है। यानी, अगर हमने कोई फ़ंक्शन f नाम से परिभाषित किया है और उसी f को उसके शरीर (बॉडी) के अंदर कहीं दोबारा (आमतौर पर भिन्न आर्ग्युमेंट के साथ) कॉल करते हैं, तो यह एक रिकर्सिव कॉल कहलाती है:
def f(n):                                   # f नाम का फ़ंक्शन परिभाषित करें
    print(f'f() called with argument {n}')  # प्रत्येक कॉल पर प्रिंट करें (डेमोंस्ट्रेशन के लिए)
    if n <= 0:                              # अगर n नकारात्मक हो जाए, तो रुक जाएँ
        print('Let\'s stop here...')
    else:                                   # वरना f को एक छोटे מספר पर_CALLBACK_CALL_        
        f(n - 1)

f(10)
यह प्रोग्राम क्या प्रिंट करेगा
f() called with argument 10
f() called with argument 9
f() called with argument 8
f() called with argument 7
f() called with argument 6
f() called with argument 5
f() called with argument 4
f() called with argument 3
f() called with argument 2
f() called with argument 1
f() called with argument 0
Let's stop here...
यह एक छोटा सा उदाहरण है जो दिखाता है कि एक बुनियादी रिकर्सिव फ़ंक्शन कैसे बनाया जा सकता है। व्यावहारिक उपयोग में हम अक्सर फ़ंक्शन के अंदर कुछ गणनाएँ (computation) करके उसके परिणाम लौटाते हैं, बजाय सिर्फ प्रिंट करने के।
एल्गोरिथम से जुड़ी कई समस्याओं में, खासतौर पर जो ग्राफ़, उन्नत सॉर्टिंग एल्गोरिथम या बैक्ट्रैकिंग से जुड़ी हों (जिन्हें हम बाद में कोर्स में देखेंगे), रिकर्शन बहुत मददगार होता है। ऐसी समस्याएँ आम हैं, और कई बार रिकर्सिव समाधान लिखना इटरेटिव (लूप इस्तेमाल करने वाला) समाधान लिखने से कहीं आसान होता है।
Video preview
Reducible द्वारा प्रस्तुत एक वीडियो - 5 Simple Steps for Solving Any Recursive Problem
किसी बड़े कार्य को हल करने के लिए उसे छोटे-छोटे उपकार्य में बाँटते जाना, जब तक कि हम सबसे छोटे/आसान स्थिति पर न पहुँच जाएँ, “recursive leap of faith” नामक अवधारणा से सबसे अच्छी तरह समझा जा सकता है।

Recursive Leap of Faith

जब हम किसी समस्या के लिए रिकर्सिव समाधान बनाते हैं, तो अक्सर ऐसा फ़ंक्शन बनाते हैं जो अपने अंदर ही खुद को किसी बिंदु पर कॉल करता है। इसी कॉल को रिकर्सिव कॉल कहते हैं।
रिकर्सिव समाधान की खूबी यह है कि आप मानकर चल सकते हैं कि जब फ़ंक्शन किसी सरल आर्ग्युमेंट के साथ कॉल किया जाएगा तो वह सही काम करेगा, और इसके आधार पर आप कठिन मामलों के लिए सही कार्यान्वयन कर सकते हैं। ठीक उसी तरह जैसे sqrt, max, या abs को कॉल करते हुए हम मान लेते हैं कि वे सही काम करेंगे, उसी तरह आप अपने बनने वाले फ़ंक्शन के कुछ छोटे/सरल पैरामीटर पर सही तरह से काम करने की कल्पना करके बड़े इनपुट पर उसी लॉजिक को लागू कर सकते हैं।
💡
एकमात्र शर्त यह है कि फ़ंक्शन किसी बेस केस (सबसे सरल स्थिति) पर ज़रूर पहुँचे। अगर ऐसा नहीं हुआ, तो रिकर्शन कभी खत्म नहीं होगी और StackOverflow जैसी समस्या आ सकती है!
रिकर्सिव फ़ंक्शन के अंदर किसी गणना को दिखाने के लिए हम यह उदाहरण देख सकते हैं कि कैसे 0 से n तक (0, 1, 2, …, n) की राशि (sum) को रिकर्सिव तरीक़े से निकाला जाए।
आइए sum फ़ंक्शन को चरणबद्ध तरीके से बनाते हैं। सबसे पहले, यह सुनिश्चित कीजिए कि फ़ंक्शन सबसे सरल केस (जब n = 0 या 1 हो) पर सही ढंग से काम करे:
def sum(n):
    print(f'sum({n})', end=' -> ')
    if n == 0:         # अगर n = 0 है, तो परिणाम 0 होना चाहिए
        return 0       # यहाँ फ़ंक्शन Execution रुक जाता है
    if n == 1:         # अगर n = 1 है, तो परिणाम 1 होना चाहिए
        return 1       # यहाँ फ़ंक्शन Execution रुक जाता है

print(sum(1))          # sum(1) -> 1
print(sum(0))          # sum(0) -> 0
print(sum(2))          # sum(2) -> None (क्योंकि हमने अभी वो हिस्सा नहीं लिखा है)
इस हिस्से से हमें इतना मिल जाता है कि sum(0) हमेशा 0 लौटाएगा और sum(1) हमेशा 1। ये दोनों कॉल हमारे फ़ंक्शन के अंदर भी ठीक से चलेंगे, यदि हम sum को इन्हीं वैल्यूज़ के साथ फिर से कॉल करें।
अब हम एक कदम आगे बढ़कर n = 2 और n = 3 के लिए भी sum फ़ंक्शन को तैयार कर सकते हैं, क्योंकि हमें sum(1) और sum(0) के सही परिणाम मिलेंगे:
def sum(n):
    print(f'sum({n})', end=' -> ')
    if n == 0:
        return 0
    if n == 1:
        return 1
    if n == 2:             # अगर n = 2 है, तो sum(1) के परिणाम में 2 जोड़ दें
        return n + sum(1)  # या n + sum(n - 1)
    if n == 3:             # अगर n = 3 है, तो sum(2) के परिणाम में 3 जोड़ दें
        return n + sum(2)  # या n + sum(n - 1)

print(sum(1))          # sum(1) -> 1
print(sum(0))          # sum(0) -> 0
print(sum(2))          # sum(2) -> sum(1) -> 3
print(sum(3))          # sum(3) -> sum(2) -> sum(1) -> 6
print(sum(4))          # sum(4) -> None (अभी यह हिस्सा Implement नहीं किया)
इस तरह, यह साफ़ है कि n = 0, 1, 2, 3 के लिए sum फ़ंक्शन सही परिणाम देता है।
अब बाकी मानों के लिए (e.g., n = 4, 5, 6, …) हमें बस इतना करना है कि फ़ंक्शन को बार-बार खुद को कॉल करते हुए इन छोटे मानों तक पहुँचना चाहिए (0, 1, 2, या 3)। उदाहरण के लिए, sum(4) के लिए हम 4 को sum(3) के परिणाम में जोड़ देंगे, sum(5) के लिए 5 को sum(4) के परिणाम में जोड़ेंगे:
def sum(n):
    print(f'sum({n})', end=' -> ')
    if n == 0:
        return 0
    if n == 1:
        return 1
    if n == 2:
        return n + sum(1)
    if n == 3:
        return n + sum(2)
    # बाक़ी सभी मामलों (4, 5, 6, ...)
    return n + sum(n - 1)

print(sum(2))          # sum(2) -> sum(1) -> 3
print(sum(3))          # sum(3) -> sum(2) -> sum(1) -> 6
print(sum(4))          # sum(4) -> sum(3) -> sum(2) -> sum(1) -> 10
# ...
एक रिकर्सिव फ़ंक्शन के दो प्रमुख हिस्से होते हैं:
  1. Base case: वह शर्त जो रिकर्शन को रोकती है। इसके बिना फ़ंक्शन अनंत रूप से खुद को कॉल करता रहेगा और कभी ख़तम नहीं होगा। यहाँ पर हमारा बेस केस n == 0 (और शुरू में हमने n == 1 भी रखा था) है।
  1. Recursive step: जहाँ फ़ंक्शन खुद को दोबारा कॉल करता है, आमतौर पर थोड़े बदले हुए आर्ग्युमेंट के साथ। यहाँ sum(n - 1) हमारी रिकर्सिव कॉल है।
 
क्योंकि n == 1, n == 2, और n == 3 के लिए कॉल ठीक उसी तरह काम करती है जैसे n == 4, 5, … के लिए, इसलिए हम sum फ़ंक्शन को और भी सरल बना सकते हैं ताकि हमारे पास केवल 1 बेस केस (n == 0) रह जाए:
def sum(n):
    print(f'sum({n})', end=' -> ')
    if n == 0:
        return 0
    return n + sum(n - 1)

print(sum(2))          # sum(2) -> sum(1) -> sum(0) -> 3
print(sum(3))          # sum(3) -> sum(2) -> sum(1) -> sum(0) -> 6
print(sum(4))          # sum(4) -> sum(3) -> sum(2) -> sum(1) -> sum(0) -> 10
# ...
इस तरह, “recursive leap of faith” का इस्तेमाल करते हुए हम यह मान लेते हैं कि छोटा/सरल केस सही ढंग से चलता है, और उसी आधार पर हम बड़े केस का हल निकाल लेते हैं। इससे समझ आता है कि रिकर्सिव फ़ंक्शन कैसे काम करते हैं और उन्हें कैसे बनाया जा सकता है।
💡
रिकर्सिव कॉल करना वैसा ही है जैसे एक बिल्कुल अलग फ़ंक्शन को कॉल करना, जिसके बारे में आपको पूरा यक़ीन होता है कि वह पास किए गए आर्ग्युमेंट पर सही काम करेगा।

Challenge

एक रिकर्सिव फ़ंक्शन बनाइए जो n! को से गुणा-भाग करने के बाद सही परिणाम दे।

इनपुट

इनपुट में एक मात्र पूर्णांक n दिया गया है (1 ≤ n ≤ 1000)।

आउटपुट

प्रोग्राम को n! को से मॉड लेकर उसका परिणाम प्रिंट करना चाहिए।

उदाहरण

इनपुट
आउटपुट
5
120
10
3628800
 

Constraints

Time limit: 2 seconds

Memory limit: 512 MB

Output limit: 1 MB

To check your solution you need to sign in
Sign in to continue