गतिशील प्रेषण

कंप्यूटर विज्ञान में, डायनेमिक डिस्पैच एक बहुरूपता (कंप्यूटर विज्ञान) ऑपरेशन (विधि (कंप्यूटर प्रोग्रामिंग) या फ़ंक्शन) के कार्यान्वयन को रन टाइम (प्रोग्राम जीवनचक्र चरण) पर कॉल करने की प्रक्रिया है। यह आमतौर पर ऑब्जेक्ट ओरिएंटेड प्रोग्रामिंग (OOP) भाषाओं और प्रणालियों में कार्यरत है, और इसकी एक प्रमुख विशेषता मानी जाती है।

ऑब्जेक्ट-ओरिएंटेड सिस्टम एक समस्या को इंटरेक्टिंग ऑब्जेक्ट्स के एक सेट के रूप में मॉडल करते हैं जो नाम से संदर्भित संचालन को लागू करते हैं। बहुरूपता वह परिघटना है जिसमें कुछ विनिमेय वस्तुएं प्रत्येक एक ही नाम के एक ऑपरेशन को उजागर करती हैं लेकिन संभवतः व्यवहार में भिन्न होती हैं। एक उदाहरण के रूप में, ए File वस्तु और ए Database वस्तु दोनों में एक है StoreRecord विधि जिसका उपयोग भंडारण के लिए कार्मिक रिकॉर्ड लिखने के लिए किया जा सकता है। उनके कार्यान्वयन भिन्न हैं। एक प्रोग्राम एक वस्तु के लिए एक संदर्भ रखता है जो या तो हो सकता है File वस्तु या ए Database वस्तु। यह कौन सा हो सकता है एक रन-टाइम सेटिंग द्वारा निर्धारित किया गया हो, और इस स्तर पर, प्रोग्राम को पता नहीं हो सकता है या परवाह नहीं है। जब प्रोग्राम कॉल करता है StoreRecord वस्तु पर, कुछ को चुनने की जरूरत है कि कौन सा व्यवहार अधिनियमित हो। यदि कोई OOP को ऑब्जेक्ट को भेजे जाने वाले संदेश के रूप में सोचता है, तो इस उदाहरण में प्रोग्राम एक भेजता है StoreRecord किसी अज्ञात प्रकार के ऑब्जेक्ट को संदेश, संदेश को सही ऑब्जेक्ट पर भेजने के लिए इसे रन-टाइम सपोर्ट सिस्टम पर छोड़ देता है। वस्तु जो भी व्यवहार लागू करती है उसे लागू करती है।

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

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

सिंगल और मल्टीपल डिस्पैच
कॉल करने की विधि के किस संस्करण का विकल्प या तो एक वस्तु पर या वस्तुओं के संयोजन पर आधारित हो सकता है। पहले वाले को सिंगल डिस्पैच कहा जाता है और यह सामान्य ऑब्जेक्ट-ओरिएंटेड भाषाओं जैसे स्मॉलटाक, सी ++, जावा (प्रोग्रामिंग भाषा), सी शार्प (प्रोग्रामिंग लैंग्वेज)|सी#, उद्देश्य सी, स्विफ्ट (प्रोग्रामिंग भाषा), जावास्क्रिप्ट और सीधे समर्थित है। पायथन (प्रोग्रामिंग भाषा)। इन और इसी तरह की भाषाओं में, एक समान सिंटैक्स के साथ डिवीजन (गणित) के लिए एक विधि कह सकते हैं <वाक्यविन्यास लैंग = अजगर> डिविडेंड.डिवाइड (भाजक) # लाभांश / भाजक  जहां पैरामीटर वैकल्पिक हैं। इसे नाम का संदेश भेजने के रूप में माना जाता है divide पैरामीटर के साथ divisor को dividend. कार्यान्वयन के आधार पर ही चयन किया जाएगा dividendका प्रकार (शायद परिमेय संख्याएँ, तैरनेवाला स्थल, मैट्रिक्स (गणित)), के प्रकार या मूल्य की अवहेलना divisor.

इसके विपरीत, कुछ भाषाएँ ऑपरेंड के संयोजन के आधार पर विधियों या कार्यों को प्रेषित करती हैं; विभाजन मामले में, के प्रकार dividend और divisor एक साथ निर्धारित करें कि कौन सा divide ऑपरेशन किया जाएगा। इसे एकाधिक प्रेषण के रूप में जाना जाता है। एकाधिक प्रेषण का समर्थन करने वाली भाषाओं के उदाहरण सामान्य लिस्प, डायलन (प्रोग्रामिंग भाषा) और जूलिया (प्रोग्रामिंग भाषा) हैं।

गतिशील प्रेषण तंत्र
विभिन्न गतिशील प्रेषण तंत्रों के साथ एक भाषा लागू की जा सकती है। किसी भाषा द्वारा प्रदान किए गए गतिशील प्रेषण तंत्र के विकल्प काफी हद तक उन प्रोग्रामिंग प्रतिमानों को बदल देते हैं जो किसी भाषा में उपलब्ध हैं या उपयोग करने के लिए सबसे स्वाभाविक हैं।

आम तौर पर, एक टाइप की गई भाषा में, प्रेषण तंत्र तर्कों के प्रकार के आधार पर किया जाएगा (आमतौर पर संदेश के रिसीवर के प्रकार के आधार पर)। कमजोर या कोई टाइपिंग सिस्टम वाली भाषाएं अक्सर प्रत्येक वस्तु के लिए वस्तु डेटा के भाग के रूप में एक प्रेषण तालिका ले जाती हैं। यह इंस्टेंस व्यवहार की अनुमति देता है क्योंकि प्रत्येक इंस्टेंस किसी दिए गए संदेश को एक अलग विधि में मैप कर सकता है।

कुछ भाषाएँ हाइब्रिड दृष्टिकोण प्रदान करती हैं।

डायनेमिक डिस्पैच में हमेशा एक ओवरहेड होता है इसलिए कुछ भाषाएँ विशेष तरीकों के लिए स्टैटिक डिस्पैच की पेशकश करती हैं।

सी ++ कार्यान्वयन
C++ अर्ली बाइंडिंग का उपयोग करता है और डायनेमिक और स्टैटिक डिस्पैच दोनों प्रदान करता है। प्रेषण का डिफ़ॉल्ट रूप स्थिर है। गतिशील प्रेषण प्राप्त करने के लिए प्रोग्रामर को एक विधि घोषित करनी चाहिए virtual.

सी ++ कंपाइलर्स आमतौर पर वर्चुअल विधि तालिका (vtable) नामक डेटा संरचना के साथ गतिशील प्रेषण को कार्यान्वित करते हैं जो किसी दिए गए वर्ग के लिए सदस्य फ़ंक्शन पॉइंटर्स के सेट के रूप में नाम-से-कार्यान्वयन मैपिंग को परिभाषित करता है। (यह विशुद्ध रूप से एक कार्यान्वयन विवरण है; C ++ विनिर्देश vtables का उल्लेख नहीं करता है।) उस प्रकार के उदाहरण तब इस तालिका में उनके उदाहरण डेटा के भाग के रूप में एक सूचक को संग्रहीत करेंगे। यह जटिल है जब एकाधिक वंशानुक्रम का उपयोग किया जाता है। चूंकि सी ++ देर से बाध्यकारी का समर्थन नहीं करता है, सी ++ ऑब्जेक्ट में वर्चुअल टेबल को रन-टाइम पर संशोधित नहीं किया जा सकता है, जो प्रेषण लक्ष्य के संभावित सेट को संकलन समय पर चुने गए सीमित सेट तक सीमित करता है।

टाइप ओवरलोडिंग सी ++ में गतिशील प्रेषण का उत्पादन नहीं करता है क्योंकि भाषा संदेश पैरामीटर के प्रकारों को औपचारिक संदेश नाम का हिस्सा मानती है। इसका मतलब यह है कि प्रोग्रामर जो संदेश नाम देखता है वह बाइंडिंग के लिए इस्तेमाल किया जाने वाला औपचारिक नाम नहीं है।

जाओ और जंग कार्यान्वयन
जाओ (प्रोग्रामिंग भाषा), जंग (प्रोग्रामिंग भाषा) और निम (प्रोग्रामिंग भाषा) में, शुरुआती बाइंडिंग के अधिक बहुमुखी रूपांतर का उपयोग किया जाता है। Vtable पॉइंटर्स को ऑब्जेक्ट रेफरेंस के साथ 'फैट पॉइंटर्स' (गो में 'इंटरफेस', या रस्ट में 'ट्रेट ऑब्जेक्ट्स') के रूप में ले जाया जाता है। यह अंतर्निहित डेटा संरचनाओं से समर्थित इंटरफेस को अलग करता है। प्रत्येक संकलित लाइब्रेरी को किसी प्रकार का सही ढंग से उपयोग करने के लिए समर्थित इंटरफेस की पूरी श्रृंखला को जानने की आवश्यकता नहीं है, केवल विशिष्ट व्यवहार्य लेआउट जिसकी उन्हें आवश्यकता होती है। कोड अलग-अलग इंटरफेस को अलग-अलग कार्यों के लिए डेटा के एक ही टुकड़े में पास कर सकता है। यह बहुमुखी प्रतिभा प्रत्येक वस्तु संदर्भ के साथ अतिरिक्त डेटा की कीमत पर आती है, जो कि समस्याग्रस्त है यदि ऐसे कई संदर्भों को लगातार संग्रहीत किया जाता है।

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

स्मालटॉक कार्यान्वयन
स्मॉलटाक एक प्रकार-आधारित संदेश डिस्पैचर का उपयोग करता है। प्रत्येक उदाहरण में एक ही प्रकार होता है जिसकी परिभाषा में विधियाँ होती हैं। जब एक उदाहरण एक संदेश प्राप्त करता है, तो डिस्पैचर प्रकार के लिए संदेश-टू-मेथड मैप में संबंधित विधि को देखता है और फिर विधि को आमंत्रित करता है।

क्योंकि एक प्रकार में आधार प्रकार की श्रृंखला हो सकती है, यह लुक-अप महंगा हो सकता है। स्मॉलटाक के तंत्र का एक सरल कार्यान्वयन सी ++ की तुलना में काफी अधिक ओवरहेड प्रतीत होता है और यह ओवरहेड प्रत्येक ऑब्जेक्ट को प्राप्त होने वाले प्रत्येक संदेश के लिए खर्च किया जाएगा।

रियल स्मॉलटाक कार्यान्वयन अक्सर इनलाइन कैशिंग नामक तकनीक का उपयोग करते हैं यह विधि प्रेषण को बहुत तेज़ बनाता है। इनलाइन कैशिंग मूल रूप से कॉल साइट के पिछले डेस्टिनेशन मेथड एड्रेस और ऑब्जेक्ट क्लास (या मल्टी-वे कैशिंग के लिए कई जोड़े) को स्टोर करता है। विधि चयनकर्ता के आधार पर कैश्ड विधि को सबसे आम लक्ष्य विधि (या केवल कैश मिस हैंडलर) के साथ प्रारंभ किया गया है। जब विधि कॉल साइट निष्पादन के दौरान पहुंच जाती है, तो यह केवल कैश में पते को कॉल करती है। (डायनेमिक कोड जनरेटर में, यह कॉल एक डायरेक्ट कॉल है क्योंकि डायरेक्ट एड्रेस कैश मिस लॉजिक द्वारा वापस पैच किया जाता है।) कॉल किए गए तरीके में प्रोलॉग कोड फिर कैश्ड क्लास की तुलना वास्तविक ऑब्जेक्ट क्लास से करता है, और यदि वे मेल नहीं खाते हैं, कक्षा में सही विधि खोजने के लिए एक कैश मिस हैंडलर को निष्पादन शाखाएँ। एक तेज़ कार्यान्वयन में कई कैश प्रविष्टियाँ हो सकती हैं और प्रारंभिक कैश मिस पर सही तरीके से निष्पादन प्राप्त करने के लिए अक्सर केवल कुछ निर्देशों की आवश्यकता होती है। सामान्य मामला एक कैश्ड क्लास मैच होगा, और विधि में निष्पादन जारी रहेगा।

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

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

पाइथन (प्रोग्रामिंग लैंग्वेज), रूबी (प्रोग्रामिंग भाषा), ऑब्जेक्टिव-सी और ग्रूवी (प्रोग्रामिंग भाषा) सहित कई अन्य गतिशील रूप से टाइप की गई भाषाएँ समान दृष्टिकोण का उपयोग करती हैं।

पायथन
में उदाहरण <वाक्यविन्यास लैंग = अजगर> वर्ग बिल्ली: डेफ बोलो (स्वयं): प्रिंट (म्याऊ)

वर्ग कुत्ता: डेफ बोलो (स्वयं): प्रिंट (वूफ़)

डेफ बोलो (पीईटी): # डायनेमिकली स्पीक मेथड डिस्पैच करता है # पालतू या तो बिल्ली या कुत्ते का उदाहरण हो सकता है पालतू बोलो

बिल्ली = बिल्ली बोलो (बिल्ली) कुत्ता = कुत्ता बोलो (कुत्ते) 

सी ++
में उदाहरण <वाक्यविन्यास लैंग = सीपीपी>
 * 1) शामिल

// पेट को एब्स्ट्रैक्ट वर्चुअल बेस क्लास बनाएं कक्षा पालतू { जनता: आभासी शून्य बोलो = 0; };

क्लास डॉग : पब्लिक पेट { जनता: शून्य बोलें ओवरराइड {       एसटीडी :: अदालत << वाह!\n; } };

क्लास कैट : पब्लिक पेट { जनता: शून्य बोलें ओवरराइड {       एसटीडी :: अदालत << म्याऊ!\n; } };

// बोलें पेट से प्राप्त कुछ भी स्वीकार करने में सक्षम होगा शून्य बोल (पालतू और पालतू) {   पालतू बोलो ; }

मुख्य प्रवेश बिंदु {   डॉग फिडो; बिल्ली सिम्बा; बोलो (फ़िदो); बोलो (सिम्बा); वापसी 0; } 

यह भी देखें

 * समारोह बहु-संस्करण
 * फंक्शन ओवरलोडिंग
 * संदेश देना
 * ओवरराइडिंग विधि
 * दोहरा प्रेषण
 * नाम बंधन

अग्रिम पठन

 * (NB. Something similar to "fat pointers" specifically for Intel's real-mode segment:offset addressing on x86 processors, containing both a deliberately denormalized pointer to a shared code entry point and some info to still distinguish the different callers in the shared code. While, in an open system, pointer-normalizing 3rd-party instances (in other drivers or applications) cannot be ruled out completely on public interfaces, the scheme can be used safely on internal interfaces to avoid redundant entry code sequences.)
 * 
 * (NB. Something similar to "fat pointers" specifically for Intel's real-mode segment:offset addressing on x86 processors, containing both a deliberately denormalized pointer to a shared code entry point and some info to still distinguish the different callers in the shared code. While, in an open system, pointer-normalizing 3rd-party instances (in other drivers or applications) cannot be ruled out completely on public interfaces, the scheme can be used safely on internal interfaces to avoid redundant entry code sequences.)
 *