指针详细知识
指针有许多但很简单的概念,它们对C编程非常重要。任何C程序员都应该清楚以下重要的指针概念-
指针算术 - 指针中可以使用四种算术运算符:++,-,+,-
指针是一个地址,它是一个数值。因此,您可以像对数值一样对指针执行算术运算。可以在指针上使用四种算术运算符:++,-,+和-
为了理解指针算术,让我们考虑ptr是一个指向地址1000的整数指针。假设32位(4字节)整数,让我们对指针执行以下算术运算-
在上述操作之后,由于ptr每次递增,因此ptr将指向位置1004,它将指向下一个整数位置,即当前位置旁边的4个字节。此操作将指针移到下一个存储位置,而不会影响该存储位置的实际值。如果ptr指向地址为1000的字符(char),则上述操作将指向位置1001,因为下一个字符将在1001处可用(因为char类型只占用一个字节)。
递增指针
我们更喜欢在程序中使用指针而不是数组,因为变量指针可以递增,这与数组名不同,因为数组名是常量指针,因此不能递增。以下程序递增变量指针以访问数组的每个后续元素-
#include <stdio.h>
const int MAX = 3;
int main () {
int var[] = {10, 100, 200};
int i, *ptr;
/* let us have array address in pointer */
ptr = var;
for ( i = 0; i < MAX; i++) {
printf("Address of var[%d] = %x\n", i, ptr );
printf("Value of var[%d] = %d\n", i, *ptr );
/* move to the next location */
ptr++;
}
return 0;
}
尝试一下
递减指针
相同的注意事项适用于递减指针,指针的值按其数据类型的字节数减少,如下所示-
#include <stdio.h>
const int MAX = 3;
int main () {
int var[] = {10, 100, 200};
int i, *ptr;
/* let us have array address in pointer */
ptr = &var[MAX-1];
for ( i = MAX; i > 0; i--) {
printf("Address of var[%d] = %x\n", i-1, ptr );
printf("Value of var[%d] = %d\n", i-1, *ptr );
/* move to the previous location */
ptr--;
}
return 0;
}
尝试一下
指针比较
可以通过使用关系运算符(例如==,<和>)来比较指针。如果p1和p2指向彼此相关的变量(例如同一数组的元素),则可以有意义地比较p1和p2。下面的程序修改了前面的示例- 通过递增变量指针,只要它指向的地址小于或等于数组的最后一个元素的地址,即&var [MAX-1]
#include <stdio.h>
const int MAX = 3;
int main () {
int var[] = {10, 100, 200};
int i, *ptr;
/* let us have address of the first element in pointer */
ptr = var;
i = 0;
while ( ptr <= &var[MAX - 1] ) {
printf("Address of var[%d] = %x\n", i, ptr );
printf("Value of var[%d] = %d\n", i, *ptr );
/* point to the next location */
ptr++;
i++;
}
return 0;
}
尝试一下
指针数组 - 您可以定义数组以容纳多个指针。
int a[3][4] = {0,1,2,3,4,5,6,7,8,9,10,11};
在理解指针数组的概念之前,让我们考虑以下示例,该示例使用3个整数的数组-
#include <stdio.h>
const int MAX = 3;
int main () {
int var[] = {10, 100, 200};
int i;
for (i = 0; i < MAX; i++) {
printf("Value of var[%d] = %d\n", i, var[i] );
}
return 0;
}
尝试一下
在某些情况下,我们想要维护一个数组,该数组可以存储指向int或char或任何其他可用数据类型的指针。以下是指向整数的指针数组的声明-
它将ptr声明为MAX整数大小指针的数组。因此,ptr中的每个元素都持有一个指向int值的指针。以下示例使用三个整数,它们存储在指针数组中,如下所示:
#include <stdio.h>
const int MAX = 3;
int main () {
int var[] = {10, 100, 200};
int i, *ptr[MAX];
for ( i = 0; i < MAX; i++) {
ptr[i] = &var[i]; /* assign the address of integer. */
}
for ( i = 0; i < MAX; i++) {
printf("Value of var[%d] = %d\n", i, *ptr[i] );
}
return 0;
}
尝试一下
您还可以使用指向字符的指针数组来存储字符串列表,如下所示:
#include <stdio.h>
const int MAX = 4;
int main () {
char *names[] = {
"Zara Ali",
"Hina Ali",
"Nuha Ali",
"Sara Ali"
};
int i = 0;
for ( i = 0; i < MAX; i++) {
printf("Value of names[%d] = %s\n", i, names[i] );
}
return 0;
}
尝试一下
多重指针- C允许您将指针放在指针上。
指向指针的指针是多种间接形式或一系列指针。通常,指针包含变量的地址。当我们定义一个指向指针的指针时,第一个指针包含第二个指针的地址,该地址指向包含实际值的位置,如下所示。
作为指针的指针的变量必须这样声明。这是通过在其名称前面放置一个额外的星号来完成的。例如,以下声明声明了一个指向int类型的指针的指针-
当指向指针的指针间接指向目标值时,访问该值需要两次应用星号运算符,如下面的示例所示-
#include <stdio.h>
int main () {
int var;
int *ptr;
int **pptr;
var = 3000;
/* take the address of var */
ptr = &var;
/* take the address of ptr using address of operator & */
pptr = &ptr;
/* take the value using pptr */
printf("Value of var = %d\n", var );
printf("Value available at *ptr = %d\n", *ptr );
printf("Value available at **pptr = %d\n", **pptr);
return 0;
}
尝试一下
将指针传递给C中的函数 - 通过引用或地址传递参数可以使被调用函数在调用函数中更改传递的参数。
C编程允许将指针传递给函数。为此,只需将函数参数声明为指针类型。 以下是一个简单的示例,其中我们将无符号长指针传递给函数,并更改函数内部的值,该值会在调用函数中反映出来-
#include <stdio.h>
#include <time.h>
void getSeconds(unsigned long *par);
int main () {
unsigned long sec;
getSeconds( &sec );
/* print the actual value */
printf("Number of seconds: %ld\n", sec );
return 0;
}
void getSeconds(unsigned long *par) {
/* get the current number of seconds */
*par = time( NULL );
return;
}
尝试一下
该函数可以接受一个指针,也可以接受一个数组,如以下示例所示:
#include <stdio.h>
/* function declaration */
double getAverage(int *arr, int size);
int main () {
/* an int array with 5 elements */
int balance[5] = {1000, 2, 3, 17, 50};
double avg;
/* pass pointer to the array as an argument */
avg = getAverage( balance, 5 ) ;
/* output the returned value */
printf("Average value is: %f\n", avg );
return 0;
}
double getAverage(int *arr, int size) {
int i, sum = 0;
double avg;
for (i = 0; i < size; ++i) {
sum += arr[i];
}
avg = (double)sum / size;
return avg;
}
尝试一下
从C函数返回指针 - C允许函数返回指向局部变量,静态变量和动态分配的内存的指针。
,我们已经了解了C编程如何允许从函数返回数组。同样,C还允许从函数返回指针。为此,您必须声明一个返回指针的函数,如以下示例所示:
int * myFunction() {
.
.
.
}
要记住的第二点是,在函数外部返回局部变量的地址不是一个好主意,因此您必须将局部变量定义为静态变量。现在,考虑以下函数,该函数将生成10个随机数,并使用表示指针的数组名称(即第一个数组元素的地址)返回它们。
#include <stdio.h>
#include <time.h>
/* function to generate and return random numbers. */
int * getRandom( ) {
static int r[10];
int i;
/* set the seed */
srand( (unsigned)time( NULL ) );
for ( i = 0; i < 10; ++i) {
r[i] = rand();
printf("%d\n", r[i] );
}
return r;
}
/* main function to call above defined function */
int main () {
/* a pointer to an int */
int *p;
int i;
p = getRandom();
for ( i = 0; i < 10; i++ ) {
printf("*(p + [%d]) : %d\n", i, *(p + i) );
}
return 0;
}
尝试一下