डी प्रोग्रामिंग - पॉइंटर्स

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

चर की ओर सीधे संकेत करने के बजाय, सूचक चर के पते की ओर इशारा करता है। जैसा कि आप जानते हैं कि प्रत्येक चर एक मेमोरी लोकेशन है और हर मेमोरी लोकेशन में इसका पता परिभाषित होता है जिसे एम्परसेंड (&) ऑपरेटर का उपयोग करके एक्सेस किया जा सकता है जो मेमोरी में एड्रेस को दर्शाता है। निम्नलिखित पर विचार करें जो परिभाषित चर का पता प्रिंट करता है -

import std.stdio;
 
void main () { 
   int var1; 
   writeln("Address of var1 variable: ",&var1);  
   
   char var2[10]; 
   writeln("Address of var2 variable: ",&var2); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Address of var1 variable: 7FFF52691928 
Address of var2 variable: 7FFF52691930

पॉइंटर्स क्या हैं?

pointerएक वैरिएबल है जिसका मान दूसरे वैरिएबल का पता है। किसी भी चर या स्थिर की तरह, इससे पहले कि आप इसके साथ काम कर सकें, आपको एक सूचक घोषित करना चाहिए। सूचक चर घोषणा का सामान्य रूप है -

type *var-name;

यहाँ, typeसूचक का आधार प्रकार है; यह एक मान्य प्रोग्रामिंग प्रकार और होना चाहिएvar-nameसूचक चर का नाम है। जिस तारांकन चिह्न को आप सूचक घोषित करने के लिए उपयोग करते हैं, वही तारांकन चिह्न है जिसका उपयोग आप गुणन के लिए करते हैं। तथापि; इस कथन में एक तारांकन सूचक के रूप में नामित करने के लिए तारांकन का उपयोग किया जा रहा है। मान्य सूचक घोषणा निम्नलिखित हैं -

int    *ip;    // pointer to an integer 
double *dp;    // pointer to a double 
float  *fp;    // pointer to a float 
char   *ch     // pointer to character

सभी बिंदुओं के मूल्य का वास्तविक डेटा प्रकार, चाहे पूर्णांक, फ्लोट, वर्ण, या अन्यथा, एक ही, एक लंबा हेक्साडेसिमल संख्या है जो एक स्मृति पते का प्रतिनिधित्व करता है। अलग-अलग डेटा प्रकारों के पॉइंटर्स के बीच एकमात्र अंतर वैरिएबल या कॉन्टिनेंट का डेटा प्रकार है जो पॉइंटर को इंगित करता है।

डी प्रोग्रामिंग में पॉइंटर्स का उपयोग करना

कुछ महत्वपूर्ण ऑपरेशन होते हैं, जब हम पॉइंटर्स का बहुत बार उपयोग करते हैं।

  • हम एक सूचक चर को परिभाषित करते हैं

  • चर के पते को एक पॉइंटर पर असाइन करें

  • अंत में पॉइंटर चर में उपलब्ध पते पर मूल्य का उपयोग करें।

यह अपरेंट्री ऑपरेटर का उपयोग करके किया जाता है *यह उसके ऑपरेंड द्वारा निर्दिष्ट पते पर स्थित चर का मान लौटाता है। निम्न उदाहरण इन कार्यों का उपयोग करता है -

import std.stdio; 

void main () { 
   int var = 20;   // actual variable declaration. 
   int *ip;        // pointer variable
   ip = &var;   // store address of var in pointer variable  
   
   writeln("Value of var variable: ",var); 
   
   writeln("Address stored in ip variable: ",ip); 
   
   writeln("Value of *ip variable: ",*ip); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Value of var variable: 20 
Address stored in ip variable: 7FFF5FB7E930 
Value of *ip variable: 20

अशक्त संकेत

आपके द्वारा असाइन किए जाने के लिए सटीक पता नहीं होने की स्थिति में पॉइंटर NULL को पॉइंटर चर असाइन करना हमेशा एक अच्छा अभ्यास होता है। यह चर घोषणा के समय किया जाता है। एक सूचक जिसे null सौंपा गया है उसे a कहा जाता हैnull सूचक।

अशक्त सूचक कई मानक पुस्तकालयों में परिभाषित शून्य के मान के साथ एक स्थिर है, जिसमें iostream शामिल है। निम्नलिखित कार्यक्रम पर विचार करें -

import std.stdio;

void main () { 
   int  *ptr = null; 
   writeln("The value of ptr is " , ptr) ;  
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

The value of ptr is null

अधिकांश ऑपरेटिंग सिस्टम पर, प्रोग्रामों को पता 0 पर स्मृति तक पहुंचने की अनुमति नहीं है क्योंकि यह मेमोरी ऑपरेटिंग सिस्टम द्वारा आरक्षित है। तथापि; स्मृति पता 0 का विशेष महत्व है; यह इंगित करता है कि सूचक एक सुलभ मेमोरी स्थान को इंगित करने का इरादा नहीं है।

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

if(ptr)     // succeeds if p is not null 
if(!ptr)    // succeeds if p is null

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

सूचक अंकगणित

चार अंकगणितीय ऑपरेटर हैं जो पॉइंटर्स पर उपयोग किए जा सकते हैं: ++, -, +, और -

पॉइंटर अंकगणित को समझने के लिए, आइए एक पूर्णांक पॉइंटर नाम पर विचार करें ptr, जो पता 1000 की ओर इशारा करता है। 32-बिट पूर्णांक मानते हुए, आइए हम संकेत पर निम्न अंकगणितीय ऑपरेशन करें

ptr++

फिर ptrयह स्थान 1004 को इंगित करेगा क्योंकि हर बार ptr बढ़ जाता है, यह अगले पूर्णांक को इंगित करता है। यह ऑपरेशन पॉइंटर को मेमोरी स्थान पर वास्तविक मूल्य को प्रभावित किए बिना अगले मेमोरी लोकेशन पर ले जाएगा।

अगर ptr एक ऐसे चरित्र की ओर इशारा करता है जिसका पता 1000 है, तो उपरोक्त ऑपरेशन 1001 के स्थान पर इंगित करता है क्योंकि अगला चरित्र 1001 पर उपलब्ध होगा।

एक संकेतक बढ़ाना

हम सरणी के बजाय अपने प्रोग्राम में एक पॉइंटर का उपयोग करना पसंद करते हैं क्योंकि वेरिएबल पॉइंटर को ऐरे नाम के विपरीत इंक्रीमेंट किया जा सकता है, जिसे इंक्रीमेंट नहीं किया जा सकता क्योंकि यह एक निरंतर पॉइंटर है। निम्नलिखित कार्यक्रम सरणी के प्रत्येक सफल तत्व तक पहुंचने के लिए चर सूचक को बढ़ाता है -

import std.stdio; 
 
const int MAX = 3; 
 
void main () { 
   int var[MAX] = [10, 100, 200]; 
   int *ptr = &var[0];  

   for (int i = 0; i < MAX; i++, ptr++) { 
      writeln("Address of var[" , i , "] = ",ptr); 
      writeln("Value of var[" , i , "] = ",*ptr); 
   } 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Address of var[0] = 18FDBC 
Value of var[0] = 10 
Address of var[1] = 18FDC0 
Value of var[1] = 100 
Address of var[2] = 18FDC4 
Value of var[2] = 200

संकेत बनाम ऐरे

संकेत और सरणियाँ दृढ़ता से संबंधित हैं। हालाँकि, संकेत और सरणियाँ पूरी तरह से विनिमेय नहीं हैं। उदाहरण के लिए, निम्नलिखित कार्यक्रम पर विचार करें -

import std.stdio; 
 
const int MAX = 3;
  
void main () { 
   int var[MAX] = [10, 100, 200]; 
   int *ptr = &var[0]; 
   var.ptr[2]  = 290; 
   ptr[0] = 220;  
   
   for (int i = 0; i < MAX; i++, ptr++) { 
      writeln("Address of var[" , i , "] = ",ptr); 
      writeln("Value of var[" , i , "] = ",*ptr); 
   } 
}

उपरोक्त कार्यक्रम में, आप दूसरे तत्व को सेट करने के लिए var.ptr [2] देख सकते हैं और ptr [0] जिसका उपयोग शून्य तत्व को सेट करने के लिए किया जाता है। इंक्रीमेंट ऑपरेटर का उपयोग ptr के साथ किया जा सकता है लेकिन var के साथ नहीं।

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Address of var[0] = 18FDBC 
Value of var[0] = 220 
Address of var[1] = 18FDC0 
Value of var[1] = 100 
Address of var[2] = 18FDC4 
Value of var[2] = 290

सूचक को इंगित करने के लिए

एक पॉइंटर को पॉइंटर कई अप्रत्यक्ष या बिंदुओं की श्रृंखला का एक रूप है। आम तौर पर, एक सूचक में एक चर का पता होता है। जब हम एक पॉइंटर को पॉइंटर में परिभाषित करते हैं, तो पहले पॉइंटर में दूसरे पॉइंटर का पता होता है, जो उस स्थान को इंगित करता है जिसमें वास्तविक मूल्य होता है जैसा कि नीचे दिखाया गया है।

एक वैरिएबल जो एक पॉइंटर को पॉइंटर होता है उसे इस तरह घोषित किया जाना चाहिए। यह इसके नाम के सामने एक अतिरिक्त तारांकन चिह्न लगाकर किया जाता है। उदाहरण के लिए, निम्न प्रकार के सूचक को पॉइंटर घोषित करने के लिए निम्नलिखित सिंटैक्स है -

int **var;

जब किसी टार्गेट वैल्यू को अप्रत्यक्ष रूप से एक पॉइंटर को पॉइंटर द्वारा इंगित किया जाता है, तो उस मान को एक्सेस करने के लिए आवश्यक है कि तारांकन ऑपरेटर को दो बार लागू किया जाए, जैसा कि नीचे उदाहरण में दिखाया गया है -

import std.stdio;  

const int MAX = 3;
  
void main () { 
   int var = 3000; 
   writeln("Value of var :" , var); 
   
   int *ptr = &var; 
   writeln("Value available at *ptr :" ,*ptr); 
   
   int **pptr = &ptr; 
   writeln("Value available at **pptr :",**pptr); 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Value of var :3000 
Value available at *ptr :3000 
Value available at **pptr :3000

कार्य करने के लिए पासिंग पॉइंटर

D आपको किसी फ़ंक्शन को पॉइंटर पास करने की अनुमति देता है। ऐसा करने के लिए, यह केवल फ़ंक्शन पैरामीटर को पॉइंटर प्रकार के रूप में घोषित करता है।

निम्नलिखित सरल उदाहरण एक फ़ंक्शन को एक पॉइंटर पास करता है।

import std.stdio; 
 
void main () { 
   // an int array with 5 elements. 
   int balance[5] = [1000, 2, 3, 17, 50]; 
   double avg; 
   
   avg = getAverage( &balance[0], 5 ) ; 
   writeln("Average is :" , avg); 
} 
 
double getAverage(int *arr, int size) { 
   int    i; 
   double avg, sum = 0; 
   
   for (i = 0; i < size; ++i) {
      sum += arr[i]; 
   } 
   
   avg = sum/size; 
   return avg; 
}

जब उपरोक्त कोड को एक साथ संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Average is :214.4

कार्य से रिटर्न पॉइंटर

निम्नलिखित फ़ंक्शन पर विचार करें, जो एक पॉइंटर का उपयोग करके 10 नंबर लौटाता है, जिसका अर्थ है पहले सरणी तत्व का पता।

import std.stdio;
  
void main () { 
   int *p = getNumber(); 
   
   for ( int i = 0; i < 10; i++ ) { 
      writeln("*(p + " , i , ") : ",*(p + i)); 
   } 
} 
 
int * getNumber( ) { 
   static int r [10]; 
   
   for (int i = 0; i < 10; ++i) {
      r[i] = i; 
   }
   
   return &r[0]; 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

*(p + 0) : 0 
*(p + 1) : 1 
*(p + 2) : 2 
*(p + 3) : 3 
*(p + 4) : 4 
*(p + 5) : 5 
*(p + 6) : 6 
*(p + 7) : 7 
*(p + 8) : 8 
*(p + 9) : 9

एक सरणी को इंगित करता है

एक सरणी नाम सरणी के पहले तत्व के लिए एक स्थिर सूचक है। इसलिए, घोषणा में -

double balance[50];

balanceएक सूचक और संतुलन [0] है, जो सरणी संतुलन के पहले तत्व का पता है। इस प्रकार, निम्न प्रोग्राम टुकड़ा असाइन होता हैp के पहले तत्व का पता balance -

double *p; 
double balance[10]; 
 
p = balance;

सरणी नामों को निरंतर पॉइंटर्स के रूप में उपयोग करना कानूनी है, और इसके विपरीत। इसलिए, * (शेष + 4) संतुलन [4] पर डेटा तक पहुंचने का एक वैध तरीका है।

एक बार जब आप पी में पहले तत्व का पता संग्रहीत करते हैं, तो आप सरणी तत्वों का उपयोग * पी, * (पी + 1), * (पी + 2) और इतने पर कर सकते हैं। निम्नलिखित उदाहरण उपरोक्त सभी अवधारणाओं को दिखाता है -

import std.stdio;
 
void main () { 
   // an array with 5 elements. 
   double balance[5] = [1000.0, 2.0, 3.4, 17.0, 50.0]; 
   double *p;  
   
   p = &balance[0]; 
  
   // output each array element's value  
   writeln("Array values using pointer " ); 
   
   for ( int i = 0; i < 5; i++ ) { 
      writeln( "*(p + ", i, ") : ", *(p + i)); 
   } 
}

जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो यह निम्नलिखित परिणाम उत्पन्न करता है -

Array values using pointer  
*(p + 0) : 1000 
*(p + 1) : 2 
*(p + 2) : 3.4 
*(p + 3) : 17
*(p + 4) : 50