लूप इनवेरिएंट: Difference between revisions

From Vigyanwiki
No edit summary
No edit summary
Line 2: Line 2:
{{for|कंप्यूटर प्रोग्रामिंग अनुकूलन तकनीक|लूप-अपरिवर्तनीय कोड गति}}
{{for|कंप्यूटर प्रोग्रामिंग अनुकूलन तकनीक|लूप-अपरिवर्तनीय कोड गति}}


[[कंप्यूटर विज्ञान]] में, एक लूप इनवेरिएंट एक [[कंप्यूटर प्रोग्राम]] लूप की एक संपत्ति है जो प्रत्येक पुनरावृत्ति से पहले (और बाद में) सत्य होती है। यह एक तार्किक अभिकथन है, जिसे कभी-कभी कोड [[अभिकथन (सॉफ़्टवेयर विकास)]] के साथ जांचा जाता है। लूप के प्रभाव को समझने के लिए इसके अपरिवर्तनीयों को जानना आवश्यक है।
[[कंप्यूटर विज्ञान]] में, एक लूप इनवेरिएंट एक [[कंप्यूटर प्रोग्राम]] लूप की एक गुण है जो प्रत्येक पुनरावृत्ति से पहले (और बाद में) सत्य होती है। यह एक तार्किक अभिकथन है, जिसे कभी-कभी कोड [[अभिकथन (सॉफ़्टवेयर विकास)]] के साथ जांचा जाता है। लूप के प्रभाव को समझने के लिए इसके अपरिवर्तनीयों को जानना आवश्यक है।


[[औपचारिक सत्यापन|औपचारिक कार्यक्रम सत्यापन]] में, विशेष रूप से [[होरे तर्क|फ़्लॉइड-होरे]] दृष्टिकोण में, लूप इनवेरिएंट को औपचारिक [[विधेय तर्क]] द्वारा व्यक्त किया जाता है और लूप के गुणों को प्रमाणित करने के लिए और एक्सटेंशन [[कलन विधि]] द्वारा उपयोग किया जाता है, जो लूप (सामान्यतः [[शुद्धता (कंप्यूटर विज्ञान)]] गुण) को नियोजित करते हैं। लूप इनवेरिएंट एक लूप में प्रवेश करने और प्रत्येक पुनरावृत्ति के बाद सत्य होंगे, जिससे लूप से बाहर निकलने पर लूप इनवेरिएंट और लूप समाप्ति स्थिति दोनों की गारंटी दी जा सके।
[[औपचारिक सत्यापन|औपचारिक कार्यक्रम सत्यापन]] में, विशेष रूप से [[होरे तर्क|फ़्लॉइड-होरे]] दृष्टिकोण में, लूप इनवेरिएंट को औपचारिक [[विधेय तर्क]] द्वारा व्यक्त किया जाता है और लूप के गुणों को प्रमाणित करने के लिए और एक्सटेंशन [[कलन विधि]] द्वारा उपयोग किया जाता है, जो लूप (सामान्यतः [[शुद्धता (कंप्यूटर विज्ञान)]] गुण) को नियोजित करते हैं। लूप इनवेरिएंट एक लूप में प्रवेश करने और प्रत्येक पुनरावृत्ति के बाद सत्य होंगे, जिससे लूप से बाहर निकलने पर लूप इनवेरिएंट और लूप समाप्ति स्थिति दोनों की गारंटी दी जा सके।
Line 11: Line 11:


==अनौपचारिक उदाहरण==
==अनौपचारिक उदाहरण==
निम्नलिखित [[सी (प्रोग्रामिंग भाषा)|C (प्रोग्रामिंग भाषा)]] [[सबरूटीन]] <code>max()</code> अपने तर्क सरणी <code>a[]</code> में अधिकतम मान लौटाता है, इसकी लंबाई प्रदान की गई <code>n</code> कम से कम 1 हो। टिप्पणियाँ पंक्तियों 3, 6, 9, 11 और 13 पर प्रदान की जाती हैं। प्रत्येक टिप्पणी फलन के उस चरण में एक या अधिक चर के मूल्यों के बारे में सत्यापन करती है।
निम्नलिखित [[सी (प्रोग्रामिंग भाषा)|C (प्रोग्रामिंग भाषा)]] [[सबरूटीन]] <code>max()</code> अपने तर्क सरणी <code>a[]</code> में अधिकतम मान लौटाता है, इसकी लंबाई प्रदान की गई <code>n</code> कम से कम 1 हो। टिप्पणियाँ पंक्तियों 3, 6, 9, 11 और 13 पर प्रदान की जाती हैं। प्रत्येक टिप्पणी फलन के उस चरण में एक या अधिक चर के मूल्यों के बारे में सत्यापन करती है। लूप बॉडी के अन्दर, लूप के प्रारंभ और अंत में हाइलाइट किए गए प्रमाण (पंक्तियाँ 6 और 11), बिल्कुल समान हैं। इस प्रकार वे लूप की एक अपरिवर्तनीय गुण का वर्णन करते हैं। जब पंक्ति 13 पर पहुँच जाता है, तब भी यह अपरिवर्तनीय रहता है, और यह ज्ञात होता है कि पंक्ति 5 से लूप की स्थिति <code>i!=n</code> असत्य हो गयी है। दोनों गुण मिलकर यही दर्शाते हैं कि <code>m</code> में अधिकतम मान <code>a[0...n-1]</code> के बराबर है, अर्थात्, पंक्ति 14 से सही मान लौटाया जाता है।
लूप बॉडी के भीतर, लूप की शुरुआत और अंत में हाइलाइट किए गए दावे (पंक्तियाँ 6 और 11), बिल्कुल समान हैं। इस प्रकार वे लूप की एक अपरिवर्तनीय संपत्ति का वर्णन करते हैं।
 
जब पंक्ति 13 पर पहुँच जाता है, तब भी यह अपरिवर्तनीय रहता है, और यह ज्ञात होता है कि लूप की स्थिति क्या है <code>i!=n</code> पंक्ति 5 से असत्य हो गया है। दोनों गुण मिलकर यही दर्शाते हैं <code>m</code> में अधिकतम मान के बराबर है <code>a[0...n-1]</code>, अर्थात्, पंक्ति 14 से सही मान लौटाया जाता है।
<!---Usage of (source) tag see http://www.mediawiki.org/wiki/Extension:SyntaxHighlight_GeSHi--->
<सिंटैक्सहाइलाइट लैंग= सी लाइन हाइलाइट= 6,11 >
<सिंटैक्सहाइलाइट लैंग= सी लाइन हाइलाइट= 6,11 >
int max(int ​​n, const int a[]) {<syntaxhighlight>
int max(int ​​n, const int a[]) {<syntaxhighlight>
Line 82: Line 80:
:<math>\frac{\{C\land I\}\;\mathrm{body}\;\{I\}} {\{I\}\;\mathtt{while}\ (C)\ \mathrm{body}\;\{\lnot C\land I\}}</math>
:<math>\frac{\{C\land I\}\;\mathrm{body}\;\{I\}} {\{I\}\;\mathtt{while}\ (C)\ \mathrm{body}\;\{\lnot C\land I\}}</math>
इसका मतलब यह है:
इसका मतलब यह है:
*यदि कुछ संपत्ति {{mvar|I}} कोड द्वारा संरक्षित है <math>\mathrm{body}</math> -अधिक सटीक रूप से, यदि {{mvar|I}} के निष्पादन के बाद धारण करता है <math>\mathrm{body}</math> जब भी दोनों {{mvar|C}} और {{mvar|I}} पहले से आयोजित - (ऊपरी पंक्ति) फिर
*यदि कुछ गुण {{mvar|I}} कोड द्वारा संरक्षित है <math>\mathrm{body}</math> -अधिक सटीक रूप से, यदि {{mvar|I}} के निष्पादन के बाद धारण करता है <math>\mathrm{body}</math> जब भी दोनों {{mvar|C}} और {{mvar|I}} पहले से आयोजित - (ऊपरी पंक्ति) फिर
* {{mvar|C}} और {{mvar|I}} पूरे लूप के निष्पादन के बाद क्रमशः गलत और सत्य होने की गारंटी है <math>\mathtt{while}\ (C)\ \mathrm{body}</math>, बशर्ते {{mvar|I}} लूप (निचली रेखा) से पहले सत्य था।
* {{mvar|C}} और {{mvar|I}} पूरे लूप के निष्पादन के बाद क्रमशः गलत और सत्य होने की गारंटी है <math>\mathtt{while}\ (C)\ \mathrm{body}</math>, बशर्ते {{mvar|I}} लूप (निचली रेखा) से पहले सत्य था।


Line 115: Line 113:
चूँकि, बाद की स्थिति <math>\lnot(x<10)\land x\leq10</math> ({{mvar|x}} 10 से कम या उसके बराबर है, लेकिन यह 10 से कम नहीं है) [[तार्किक तुल्यता]] <math>x=10</math> है, जो हम दिखाना चाहते थे।
चूँकि, बाद की स्थिति <math>\lnot(x<10)\land x\leq10</math> ({{mvar|x}} 10 से कम या उसके बराबर है, लेकिन यह 10 से कम नहीं है) [[तार्किक तुल्यता]] <math>x=10</math> है, जो हम दिखाना चाहते थे।


संपत्ति <math>0 \leq x</math> उदाहरण लूप का एक और अपरिवर्तनीय और तुच्छ गुण है <math>\mathrm{true}</math> एक और है.
गुण <math>0 \leq x</math> उदाहरण लूप का एक और अपरिवर्तनीय और तुच्छ गुण है <math>\mathrm{true}</math> एक और है.
उपरोक्त अनुमान नियम को पूर्व अपरिवर्तनीय पैदावार पर लागू करना <math>\{0 \leq x\}\; \mathtt{while}\ (x<10)\ x := x+1\;\{10 \leq x\}</math>.
उपरोक्त अनुमान नियम को पूर्व अपरिवर्तनीय पैदावार पर लागू करना <math>\{0 \leq x\}\; \mathtt{while}\ (x<10)\ x := x+1\;\{10 \leq x\}</math>.
इसे अपरिवर्तनीय पर लागू करना <math>\mathrm{true}</math> पैदावार <math>\{\mathrm{true}\}\; \mathtt{while}\ (x<10)\ x := x+1\;\{10 \leq x\}</math>, जो थोड़ा अधिक अभिव्यंजक है।
इसे अपरिवर्तनीय पर लागू करना <math>\mathrm{true}</math> पैदावार <math>\{\mathrm{true}\}\; \mathtt{while}\ (x<10)\ x := x+1\;\{10 \leq x\}</math>, जो थोड़ा अधिक अभिव्यंजक है।
Line 168: Line 166:
एक लूप इनवेरिएंट निम्नलिखित उद्देश्यों में से एक को पूरा कर सकता है:
एक लूप इनवेरिएंट निम्नलिखित उद्देश्यों में से एक को पूरा कर सकता है:
# विशुद्ध रूप से वृत्तचित्र
# विशुद्ध रूप से वृत्तचित्र
# को कोड के भीतर जांचा जाना चाहिए, उदा. एक अभिकथन कॉल द्वारा
# को कोड के अन्दर जांचा जाना चाहिए, उदा. एक अभिकथन कॉल द्वारा
# होरे तर्क|फ्लोयड-होरे दृष्टिकोण के आधार पर सत्यापित किया जाना है
# होरे तर्क|फ्लोयड-होरे दृष्टिकोण के आधार पर सत्यापित किया जाना है


Line 198: Line 196:
}
}
</syntaxhighlight>
</syntaxhighlight>
इसके विपरीत, उदा. संपत्ति <code>0<=i && i<=n</code> मूल और अनुकूलित प्रोग्राम दोनों के लिए एक लूप अपरिवर्तनीय है, लेकिन यह कोड का हिस्सा नहीं है, इसलिए इसे लूप से बाहर ले जाने की बात करने का कोई मतलब नहीं है।
इसके विपरीत, उदा. गुण <code>0<=i && i<=n</code> मूल और अनुकूलित प्रोग्राम दोनों के लिए एक लूप अपरिवर्तनीय है, लेकिन यह कोड का हिस्सा नहीं है, इसलिए इसे लूप से बाहर ले जाने की बात करने का कोई मतलब नहीं है।


लूप-इनवेरिएंट कोड संबंधित लूप-इनवेरिएंट संपत्ति को प्रेरित कर सकता है।{{clarify|reason=I guess, once the notion of 'loop-invariant code' is defined in a sufficiently formal way, it can be proven that loop-invariant code *always* induces a corresponding loop invariant property.|date=March 2016}} उपरोक्त उदाहरण के लिए, इसे देखने का सबसे आसान तरीका एक प्रोग्राम पर विचार करना है जहां लूप इनवेरिएंट कोड की गणना लूप के पहले और भीतर दोनों जगह की जाती है:
लूप-इनवेरिएंट कोड संबंधित लूप-इनवेरिएंट गुण को प्रेरित कर सकता है।{{clarify|reason=I guess, once the notion of 'loop-invariant code' is defined in a sufficiently formal way, it can be proven that loop-invariant code *always* induces a corresponding loop invariant property.|date=March 2016}} उपरोक्त उदाहरण के लिए, इसे देखने का सबसे आसान तरीका एक प्रोग्राम पर विचार करना है जहां लूप इनवेरिएंट कोड की गणना लूप के पहले और अन्दर दोनों जगह की जाती है:
<syntaxhighlight lang="C">
<syntaxhighlight lang="C">
x1 = y+z;
x1 = y+z;
Line 209: Line 207:
}
}
</syntaxhighlight>
</syntaxhighlight>
इस कोड की एक लूप-अपरिवर्तनीय संपत्ति है <code>(x1==x2 && t1==x2*x2) || i==0</code>, यह दर्शाता है कि लूप से पहले गणना किए गए मान भीतर गणना किए गए मानों से सहमत हैं (पहले पुनरावृत्ति से पहले को छोड़कर)।
इस कोड की एक लूप-अपरिवर्तनीय गुण है <code>(x1==x2 && t1==x2*x2) || i==0</code>, यह दर्शाता है कि लूप से पहले गणना किए गए मान अन्दर गणना किए गए मानों से सहमत हैं (पहले पुनरावृत्ति से पहले को छोड़कर)।


== यह भी देखें ==
== यह भी देखें ==

Revision as of 12:33, 16 July 2023

कंप्यूटर विज्ञान में, एक लूप इनवेरिएंट एक कंप्यूटर प्रोग्राम लूप की एक गुण है जो प्रत्येक पुनरावृत्ति से पहले (और बाद में) सत्य होती है। यह एक तार्किक अभिकथन है, जिसे कभी-कभी कोड अभिकथन (सॉफ़्टवेयर विकास) के साथ जांचा जाता है। लूप के प्रभाव को समझने के लिए इसके अपरिवर्तनीयों को जानना आवश्यक है।

औपचारिक कार्यक्रम सत्यापन में, विशेष रूप से फ़्लॉइड-होरे दृष्टिकोण में, लूप इनवेरिएंट को औपचारिक विधेय तर्क द्वारा व्यक्त किया जाता है और लूप के गुणों को प्रमाणित करने के लिए और एक्सटेंशन कलन विधि द्वारा उपयोग किया जाता है, जो लूप (सामान्यतः शुद्धता (कंप्यूटर विज्ञान) गुण) को नियोजित करते हैं। लूप इनवेरिएंट एक लूप में प्रवेश करने और प्रत्येक पुनरावृत्ति के बाद सत्य होंगे, जिससे लूप से बाहर निकलने पर लूप इनवेरिएंट और लूप समाप्ति स्थिति दोनों की गारंटी दी जा सके।

प्रोग्रामिंग पद्धति के दृष्टिकोण से, लूप इनवेरिएंट को लूप के अधिक अमूर्त विनिर्देश के रूप में देखा जा सकता है, जो इस कार्यान्वयन के विवरण से हटकर लूप के गहरे उद्देश्य को दर्शाता है। एक सर्वेक्षण आलेख [1] कंप्यूटर विज्ञान (खोज, छँटाई, अनुकूलन, अंकगणित आदि) के कई क्षेत्रों से मौलिक एल्गोरिदम को सम्मिलित करता है, उनमें से प्रत्येक को उसके अपरिवर्तनीय के दृष्टिकोण से चित्रित करता है।

लूप और प्रत्यावर्तन प्रोग्राम की समानता के कारण, इनवेरिएंट के साथ लूप की आंशिक शुद्धता प्रमाणित करना संरचनात्मक प्रेरण के माध्यम से रिकर्सिव प्रोग्राम की शुद्धता प्रमाणित करने के समान है। वास्तव में, लूप इनवेरिएंट अधिकांशतः किसी दिए गए लूप के समतुल्य पुनरावर्ती कार्यक्रम के लिए सिद्ध की जाने वाली आगमनात्मक परिकल्पना के समान होता है।

अनौपचारिक उदाहरण

निम्नलिखित C (प्रोग्रामिंग भाषा) सबरूटीन max() अपने तर्क सरणी a[] में अधिकतम मान लौटाता है, इसकी लंबाई प्रदान की गई n कम से कम 1 हो। टिप्पणियाँ पंक्तियों 3, 6, 9, 11 और 13 पर प्रदान की जाती हैं। प्रत्येक टिप्पणी फलन के उस चरण में एक या अधिक चर के मूल्यों के बारे में सत्यापन करती है। लूप बॉडी के अन्दर, लूप के प्रारंभ और अंत में हाइलाइट किए गए प्रमाण (पंक्तियाँ 6 और 11), बिल्कुल समान हैं। इस प्रकार वे लूप की एक अपरिवर्तनीय गुण का वर्णन करते हैं। जब पंक्ति 13 पर पहुँच जाता है, तब भी यह अपरिवर्तनीय रहता है, और यह ज्ञात होता है कि पंक्ति 5 से लूप की स्थिति i!=n असत्य हो गयी है। दोनों गुण मिलकर यही दर्शाते हैं कि m में अधिकतम मान a[0...n-1] के बराबर है, अर्थात्, पंक्ति 14 से सही मान लौटाया जाता है।

<सिंटैक्सहाइलाइट लैंग= सी लाइन हाइलाइट= 6,11 >

int max(int ​​n, const int a[]) {

int max(int n, const int a[]) {
    int m = a[0];
    // m equals the maximum value in a[0...0]
    int i = 1;
    while (i != n) {
        // m equals the maximum value in a[0...i-1]
        if (m < a[i])
            m = a[i];
        // m equals the maximum value in a[0...i]
        ++i;
        // m equals the maximum value in a[0...i-1]
    }
    // m equals the maximum value in a[0...i-1], and i==n
    return m;
}
   int m = a[0];
   // m a[0...0] में अधिकतम मान के बराबर है
   पूर्णांक मैं = 1;
   जबकि (i != n) {
       // m a[0...i-1] में अधिकतम मान के बराबर है
       अगर (एम < ए[i])
           एम = ए[आई];
       // m a[0...i] में अधिकतम मान के बराबर है
       ++मैं;
       // m a[0...i-1] में अधिकतम मान के बराबर है
   }
   // m a[0...i-1] में अधिकतम मान के बराबर है, और i==n
   वापसी एम;

} </सिंटैक्सहाइलाइट> रक्षात्मक प्रोग्रामिंग प्रतिमान के बाद, लूप स्थिति i!=n पंक्ति 5 में बेहतर ढंग से संशोधित किया जाना चाहिए i<n, नाजायज नकारात्मक मूल्यों के लिए अंतहीन लूपिंग से बचने के लिए n. चूँकि सहज रूप से कोड में इस बदलाव से कोई फर्क नहीं पड़ना चाहिए, लेकिन इसके सही होने की ओर ले जाने वाला तर्क कुछ अधिक जटिल हो जाता है, तभी से i>=n पंक्ति 13 में ज्ञात है। उसे भी प्राप्त करने के लिए i<=n धारण करता है, उस शर्त को लूप इनवेरिएंट में सम्मिलित करना होगा। यह देखना आसान है i<=n, भी, लूप का एक अपरिवर्तनीय है i<n पंक्ति 6 ​​में पंक्ति 5 में (संशोधित) लूप स्थिति से प्राप्त किया जा सकता है, और इसलिए i<=n के बाद पंक्ति 11 में रहता है i पंक्ति 10 में वृद्धि की गई है। चूँकि, जब औपचारिक कार्यक्रम सत्यापन के लिए लूप इनवेरिएंट को मैन्युअल रूप से प्रदान करना पड़ता है, तो ऐसे सहज रूप से बहुत स्पष्ट गुण होते हैं i<=n अधिकांशतः नजरअंदाज कर दिया जाता है.

फ्लोयड-होरे तर्क

होरे तर्क में|फ्लोयड-होरे तर्क,[2][3] थोड़ी देर के लूप की आंशिक शुद्धता अनुमान के निम्नलिखित नियम द्वारा नियंत्रित होती है:

इसका मतलब यह है:

  • यदि कुछ गुण I कोड द्वारा संरक्षित है -अधिक सटीक रूप से, यदि I के निष्पादन के बाद धारण करता है जब भी दोनों C और I पहले से आयोजित - (ऊपरी पंक्ति) फिर
  • C और I पूरे लूप के निष्पादन के बाद क्रमशः गलत और सत्य होने की गारंटी है , बशर्ते I लूप (निचली रेखा) से पहले सत्य था।

दूसरे शब्दों में: उपरोक्त नियम एक निगमनात्मक कदम है जिसका आधार होरे ट्रिपल है . यह त्रिगुण वास्तव में मशीन अवस्थाओं पर एक संबंध (गणित) है। यह तब भी लागू होता है जब बूलियन अभिव्यक्ति एक ऐसी स्थिति से शुरू होती है सत्य है और कुछ कोड को सफलतापूर्वक निष्पादित कर रहा है , मशीन ऐसी स्थिति में समाप्त हो जाती है I क्या सच है। यदि यह संबंध सिद्ध किया जा सकता है, तो नियम हमें कार्यक्रम के सफल निष्पादन का निष्कर्ष निकालने की अनुमति देता है जिस राज्य से नेतृत्व करेंगे I उस स्थिति के लिए सत्य है जिसमें धारण करता है. बूलियन सूत्र I इस नियम को लूप इनवेरिएंट कहा जाता है।

प्रयुक्त नोटेशन में कुछ भिन्नताओं के साथ, और इस आधार पर कि लूप रुक जाता है, इस नियम को अपरिवर्तनीय संबंध प्रमेय के रूप में भी जाना जाता है।[4][5] जैसा कि 1970 के दशक की एक पाठ्यपुस्तक इसे छात्र प्रोग्रामरों के लिए सुलभ बनाने के तरीके से प्रस्तुत करती है:[4]

चलो संकेतन P { seq } Q मतलब कि अगर P कथनों के अनुक्रम से पहले सत्य है seq फिर भागो Q इसके बाद सच है। तब अपरिवर्तनीय संबंध प्रमेय यही मानता है

P & c { seq } P
तात्पर्य
P { DO WHILE (c); seq END; } P & ¬c


उदाहरण

निम्नलिखित उदाहरण दिखाता है कि यह नियम कैसे काम करता है। कार्यक्रम पर विचार करें

जबकि (x <10)
    एक्स := एक्स+1;

फिर कोई निम्नलिखित होरे ट्रिपल को प्रमाणित कर सकता है:

की स्थिति सी while लूप है . एक उपयोगी लूप अपरिवर्तनीय I अनुमान लगाना होगा; ऐसा हो जायेगा उपयुक्त है। इन धारणाओं के अनुसार निम्नलिखित होरे ट्रिपल को प्रमाणित करना संभव है:

चूँकि इस ट्रिपल को औपचारिक रूप से फ्लोयड-होरे लॉजिक गवर्निंग असाइनमेंट के नियमों से प्राप्त किया जा सकता है, यह सहज रूप से उचित भी है: गणना उस स्थिति में शुरू होती है जहां सत्य है, जिसका सीधा सा अर्थ है क्या सच है। गणना में 1 जोड़ा जाता है x, जिसका अर्थ है कि अभी भी सत्य है (पूर्णांक x के लिए)।

इस आधार के अंतर्गत, नियम के लिए while लूप्स निम्नलिखित निष्कर्ष की अनुमति देता है:

चूँकि, बाद की स्थिति (x 10 से कम या उसके बराबर है, लेकिन यह 10 से कम नहीं है) तार्किक तुल्यता है, जो हम दिखाना चाहते थे।

गुण उदाहरण लूप का एक और अपरिवर्तनीय और तुच्छ गुण है एक और है. उपरोक्त अनुमान नियम को पूर्व अपरिवर्तनीय पैदावार पर लागू करना . इसे अपरिवर्तनीय पर लागू करना पैदावार , जो थोड़ा अधिक अभिव्यंजक है।

प्रोग्रामिंग भाषा समर्थन

एफिल

एफिल (प्रोग्रामिंग भाषा) प्रोग्रामिंग भाषा लूप इनवेरिएंट के लिए मूल समर्थन प्रदान करती है।[6] एक लूप इनवेरिएंट को वर्ग अपरिवर्तनीय के लिए उपयोग किए गए समान सिंटैक्स के साथ व्यक्त किया जाता है। नीचे दिए गए नमूने में, लूप अपरिवर्तनीय अभिव्यक्ति x <= 10 लूप आरंभीकरण के बाद और लूप बॉडी के प्रत्येक निष्पादन के बाद सत्य होना चाहिए; इसे रनटाइम पर जांचा जाता है।

    from
        x := 0
    invariant
        x <= 10
    until
        x > 10
    loop
        x := x + 1
    end


जबकि

वाइली (प्रोग्रामिंग भाषा) प्रोग्रामिंग भाषा लूप इनवेरिएंट के लिए प्रथम श्रेणी का समर्थन भी प्रदान करती है।[7] लूप इनवेरिएंट को एक या अधिक का उपयोग करके व्यक्त किया जाता है where उपवाक्य, जैसा कि निम्नलिखित में दर्शाया गया है:

function max(int[] items) -> (int r)
// Requires at least one element to compute max
requires |items| > 0
// (1) Result is not smaller than any element
ensures all { i in 0..|items| | items[i] <= r }
// (2) Result matches at least one element
ensures some { i in 0..|items| | items[i] == r }:
    //
    nat i = 1
    int m = items[0]
    //
    while i < |items|
    // (1) No item seen so far is larger than m
    where all { k in 0..i | items[k] <= m }
    // (2) One or more items seen so far matches m
    where some { k in 0..i | items[k] == m }:
        if items[i] > m:
            m = items[i]
        i = i + 1
    //
    return m
max() e> फलन पूर्णांक सरणी में सबसे बड़ा तत्व निर्धारित करता है। इसे परिभाषित करने के लिए, सरणी में कम से कम एक तत्व होना चाहिए। के बाद के नियम max() आवश्यक है कि लौटाया गया मान है: (1) किसी भी तत्व से छोटा नहीं; और, (2) कि यह कम से कम एक तत्व से मेल खाता हो। लूप इनवेरिएंट को दो के माध्यम से आगमनात्मक रूप से परिभाषित किया गया है where खंड, जिनमें से प्रत्येक पोस्टकंडीशन में एक खंड से मेल खाता है। मूलभूत अंतर यह है कि लूप इनवेरिएंट का प्रत्येक खंड परिणाम को वर्तमान तत्व तक सही होने की पहचान करता है i, जबकि पोस्टकंडिशन परिणाम को सभी तत्वों के लिए सही होने की पहचान करता है।

लूप इनवेरिएंट का उपयोग

एक लूप इनवेरिएंट निम्नलिखित उद्देश्यों में से एक को पूरा कर सकता है:

  1. विशुद्ध रूप से वृत्तचित्र
  2. को कोड के अन्दर जांचा जाना चाहिए, उदा. एक अभिकथन कॉल द्वारा
  3. होरे तर्क|फ्लोयड-होरे दृष्टिकोण के आधार पर सत्यापित किया जाना है

1. के लिए, एक प्राकृतिक भाषा टिप्पणी (जैसे // m equals the maximum value in a[0...i-1] #अनौपचारिक उदाहरण उदाहरण में) पर्याप्त है।

2. के लिए, प्रोग्रामिंग भाषा समर्थन की आवश्यकता है, जैसे सी (प्रोग्रामिंग भाषा) लाइब्रेरीassert.h, या #एफिल-दिखाया गया invariant एफिल में खंड. अधिकांशतः, कंपाइलर या रनटाइम विकल्प द्वारा रन-टाइम चेकिंग को चालू (डिबगिंग रन के लिए) और बंद (प्रोडक्शन रन के लिए) किया जा सकता है।[citation needed]

3. के लिए, गणितीय प्रमाणों का समर्थन करने के लिए कुछ उपकरण उपस्थित हैं, जो सामान्यतः #फ्लोयड-होरे तर्क-दिखाए गए फ़्लॉइड-होरे नियम पर आधारित होते हैं, कि एक दिया गया लूप कोड वास्तव में दिए गए (सेट) लूप इनवेरिएंट को संतुष्ट करता है।

अमूर्त व्याख्या की तकनीक का उपयोग दिए गए कोड के लूप इनवेरिएंट का स्वचालित रूप से पता लगाने के लिए किया जा सकता है। चूँकि, यह दृष्टिकोण बहुत ही सरल अपरिवर्तनीयों तक ही सीमित है (जैसे 0<=i && i<=n && i%2==0).

लूप-अपरिवर्तनीय कोड से अंतर

लूप-इनवेरिएंट कोड में ऐसे कथन या अभिव्यक्ति सम्मिलित होते हैं जिन्हें प्रोग्राम शब्दार्थ को प्रभावित किए बिना लूप बॉडी के बाहर ले जाया जा सकता है। ऐसे परिवर्तन, जिन्हें लूप-अपरिवर्तनीय कोड गति कहा जाता है, कुछ कंपाइलरों द्वारा कंपाइलर प्रोग्राम को अनुकूलित करने के लिए किए जाते हैं। एक लूप-इनवेरिएंट कोड उदाहरण (सी (प्रोग्रामिंग भाषा) में) है

for (int i=0; i<n; ++i) {
    x = y+z;
    a[i] = 6*i + x*x;
}

जहां गणना x = y+z और x*x लूप से पहले ले जाया जा सकता है, जिसके परिणामस्वरूप एक समतुल्य, लेकिन तेज़ प्रोग्राम बनता है:

x = y+z;
t1 = x*x;
for (int i=0; i<n; ++i) {
    a[i] = 6*i + t1;
}

इसके विपरीत, उदा. गुण 0<=i && i<=n मूल और अनुकूलित प्रोग्राम दोनों के लिए एक लूप अपरिवर्तनीय है, लेकिन यह कोड का हिस्सा नहीं है, इसलिए इसे लूप से बाहर ले जाने की बात करने का कोई मतलब नहीं है।

लूप-इनवेरिएंट कोड संबंधित लूप-इनवेरिएंट गुण को प्रेरित कर सकता है।[clarification needed] उपरोक्त उदाहरण के लिए, इसे देखने का सबसे आसान तरीका एक प्रोग्राम पर विचार करना है जहां लूप इनवेरिएंट कोड की गणना लूप के पहले और अन्दर दोनों जगह की जाती है:

x1 = y+z;
t1 = x1*x1;
for (int i=0; i<n; ++i) {
    x2 = y+z;
    a[i] = 6*i + t1;
}

इस कोड की एक लूप-अपरिवर्तनीय गुण है (x1==x2 && t1==x2*x2) || i==0, यह दर्शाता है कि लूप से पहले गणना किए गए मान अन्दर गणना किए गए मानों से सहमत हैं (पहले पुनरावृत्ति से पहले को छोड़कर)।

यह भी देखें

संदर्भ

  1. Carlo A. Furia, Bertrand Meyer and Sergey Velder. "Loop invariants: analysis, classification, and examples."ACM Computing Surveys. vol. 46, no. 3, February 2014([1]
  2. Robert W. Floyd (1967). "Assigning Meanings to Programs" (PDF). In J.T. Schwartz (ed.). Proceedings of Symposia in Applied Mathematics. Mathematical Aspects of Computer Science. Vol. 19. Providence, RI: American Mathematical Society. pp. 19–32.
  3. Hoare, C. A. R. (October 1969). "An axiomatic basis for computer programming" (PDF). Communications of the ACM. 12 (10): 576–580. doi:10.1145/363235.363259. S2CID 207726175. Archived from the original (PDF) on 2016-03-04.
  4. 4.0 4.1 Conway, Richard; Gries, David (1973). An Introduction to Programming: A Structured Approach using PL/1 and PL/C. Cambridge, Massachusetts: Winthrop. pp. 198–200.
  5. Huang, J. C. (2009). परीक्षण और विश्लेषण के माध्यम से सॉफ़्टवेयर त्रुटि का पता लगाना. Hoboken, New Jersey: John Wiley & Sons. pp. 156–157.
  6. Meyer, Bertrand, Eiffel: The Language, Prentice Hall, 1991, pp. 129–131.
  7. Pearce, David J.; Groves, Lindsay (2015). "Designing a Verifying Compiler: Lessons Learned from Developing Whiley". Science of Computer Programming. 113: 191–220. doi:10.1016/j.scico.2015.09.006.


अग्रिम पठन