Pemrograman D - Pointer

Petunjuk pemrograman D mudah dan menyenangkan untuk dipelajari. Beberapa tugas pemrograman D dilakukan lebih mudah dengan pointer, dan tugas pemrograman D lainnya, seperti alokasi memori dinamis, tidak dapat dilakukan tanpanya. Penunjuk sederhana ditunjukkan di bawah ini.

Alih-alih menunjuk langsung ke variabel, penunjuk menunjuk ke alamat variabel. Seperti yang Anda ketahui, setiap variabel adalah lokasi memori dan setiap lokasi memori memiliki alamat yang ditentukan yang dapat diakses menggunakan operator ampersand (&) yang menunjukkan alamat dalam memori. Pertimbangkan berikut ini yang mencetak alamat dari variabel yang ditentukan -

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

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

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

Apa Itu Pointer?

SEBUAH pointeradalah variabel yang nilainya adalah alamat variabel lain. Seperti variabel atau konstanta apa pun, Anda harus mendeklarasikan sebuah pointer sebelum Anda dapat mengerjakannya. Bentuk umum dari deklarasi variabel pointer adalah -

type *var-name;

Sini, typeadalah tipe dasar pointer; itu harus merupakan jenis pemrograman yang valid danvar-nameadalah nama variabel penunjuk. Tanda bintang yang Anda gunakan untuk menyatakan penunjuk adalah tanda bintang yang sama yang Anda gunakan untuk perkalian. Namun; dalam pernyataan ini tanda bintang digunakan untuk menunjuk variabel sebagai penunjuk. Berikut adalah deklarasi penunjuk yang valid -

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

Tipe data sebenarnya dari nilai semua pointer, apakah integer, float, karakter, atau sebaliknya, adalah sama, angka heksadesimal panjang yang mewakili alamat memori. Satu-satunya perbedaan antara pointer dari tipe data yang berbeda adalah tipe data dari variabel atau konstanta yang ditunjuk pointer.

Menggunakan Pointer dalam pemrograman D.

Ada beberapa operasi penting, ketika kita sering menggunakan pointer.

  • kami mendefinisikan variabel pointer

  • tetapkan alamat variabel ke sebuah pointer

  • terakhir akses nilai di alamat yang tersedia di variabel pointer.

Ini dilakukan dengan menggunakan operator unary *yang mengembalikan nilai variabel yang terletak di alamat yang ditentukan oleh operannya. Contoh berikut memanfaatkan operasi ini -

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); 
}

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

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

Pointer Null

Itu selalu merupakan praktik yang baik untuk menetapkan pointer NULL ke variabel pointer jika Anda tidak memiliki alamat yang tepat untuk ditetapkan. Ini dilakukan pada saat deklarasi variabel. Sebuah pointer yang diberikan null disebut anull penunjuk.

Pointer null adalah konstanta dengan nilai nol yang ditentukan di beberapa pustaka standar, termasuk iostream. Pertimbangkan program berikut -

import std.stdio;

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

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

The value of ptr is null

Di sebagian besar sistem operasi, program tidak diizinkan untuk mengakses memori di alamat 0 karena memori tersebut dicadangkan oleh sistem operasi. Namun; alamat memori 0 memiliki arti khusus; itu menandakan bahwa penunjuk tidak dimaksudkan untuk menunjuk ke lokasi memori yang dapat diakses.

Secara konvensi, jika sebuah pointer berisi nilai null (nol), itu dianggap tidak menunjuk ke apa-apa. Untuk memeriksa null pointer Anda dapat menggunakan pernyataan if sebagai berikut -

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

Jadi, jika semua pointer yang tidak digunakan diberi nilai null dan Anda menghindari penggunaan pointer null, Anda dapat menghindari penyalahgunaan yang tidak disengaja dari pointer yang tidak diinisialisasi. Sering kali, variabel yang tidak diinisialisasi menyimpan beberapa nilai sampah dan menjadi sulit untuk men-debug program.

Aritmatika Pointer

Ada empat operator aritmatika yang dapat digunakan pada pointer: ++, -, +, dan -

Untuk memahami aritmatika pointer, mari kita pertimbangkan pointer integer bernama ptr, yang menunjuk ke alamat 1000. Dengan asumsi bilangan bulat 32-bit, mari kita lakukan operasi aritmatik berikut pada penunjuk -

ptr++

lalu ptrakan menunjuk ke lokasi 1004 karena setiap kali ptr bertambah, ia menunjuk ke bilangan bulat berikutnya. Operasi ini akan memindahkan penunjuk ke lokasi memori berikutnya tanpa memengaruhi nilai sebenarnya di lokasi memori.

Jika ptr menunjuk ke karakter yang beralamat 1000, kemudian operasi di atas menunjuk ke lokasi 1001 karena karakter berikutnya akan tersedia di 1001.

Menambahkan Pointer

Kami lebih suka menggunakan pointer dalam program kami daripada array karena pointer variabel bisa bertambah, tidak seperti nama array yang tidak bisa bertambah karena itu adalah pointer konstan. Program berikut menambah penunjuk variabel untuk mengakses setiap elemen berikutnya dari array -

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); 
   } 
}

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

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

Pointer vs Array

Pointer dan array sangat terkait. Namun, pointer dan array tidak sepenuhnya dapat dipertukarkan. Misalnya, pertimbangkan program berikut -

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); 
   } 
}

Pada program di atas, Anda dapat melihat var.ptr [2] untuk menyetel elemen kedua dan ptr [0] yang digunakan untuk menyetel elemen ke nol. Operator increment dapat digunakan dengan ptr tetapi tidak dengan var.

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

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

Pointer ke Pointer

Sebuah penunjuk ke penunjuk adalah bentuk dari beberapa tipuan atau rangkaian penunjuk. Biasanya, pointer berisi alamat variabel. Saat kita mendefinisikan sebuah pointer ke sebuah pointer, pointer pertama berisi alamat dari pointer kedua, yang menunjuk ke lokasi yang berisi nilai sebenarnya seperti yang ditunjukkan di bawah ini.

Variabel yang merupakan penunjuk ke penunjuk harus dideklarasikan seperti itu. Ini dilakukan dengan menempatkan tanda bintang tambahan di depan namanya. Sebagai contoh, berikut adalah sintaks untuk mendeklarasikan pointer ke pointer tipe int -

int **var;

Ketika nilai target secara tidak langsung diarahkan oleh penunjuk ke penunjuk, maka untuk mengakses nilai tersebut mengharuskan operator tanda bintang diterapkan dua kali, seperti yang ditunjukkan di bawah ini dalam contoh -

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); 
}

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

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

Meneruskan Pointer ke Functions

D memungkinkan Anda untuk memberikan pointer ke suatu fungsi. Untuk melakukannya, ini cukup mendeklarasikan parameter fungsi sebagai tipe pointer.

Contoh sederhana berikut meneruskan pointer ke suatu fungsi.

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; 
}

Ketika kode di atas dikompilasi dan dieksekusi, itu menghasilkan hasil sebagai berikut -

Average is :214.4

Kembalikan Pointer dari Functions

Perhatikan fungsi berikut, yang mengembalikan 10 angka menggunakan pointer, artinya alamat elemen array pertama.

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]; 
}

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

*(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

Pointer ke Array

Nama array adalah penunjuk konstan ke elemen pertama dari array. Oleh karena itu, dalam deklarasi -

double balance[50];

balanceadalah penunjuk ke & keseimbangan [0], yang merupakan alamat elemen pertama keseimbangan larik. Jadi, fragmen program berikut menetapkanp alamat elemen pertama balance -

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

Adalah legal untuk menggunakan nama array sebagai pointer konstan, dan sebaliknya. Oleh karena itu, * (balance + 4) adalah cara yang sah untuk mengakses data secara seimbang [4].

Setelah Anda menyimpan alamat elemen pertama dalam p, Anda dapat mengakses elemen array menggunakan * p, * (p + 1), * (p + 2) dan seterusnya. Contoh berikut menunjukkan semua konsep yang dibahas di atas -

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)); 
   } 
}

Ketika kode di atas dikompilasi dan dijalankan, itu menghasilkan hasil sebagai berikut -

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