大话C语言:第34篇 指针与函数的关系

1 指针作为函数的参数

指针作为函数的参数,语法格式:

返回数据类型 函数名(数据类型 *指针变量1, ..., 数据类型 *指针变量1)

注意,指针变量的数据类型可以为任何数据类型,包括基本数类型和自定义数据类型

代码示例:

void swap(int *ptr1,int *ptr2)
{
    // 提前保存ptr1指向对象内容
    int temp = *ptr1;
    // ptr2指向的变量的值,给ptr1指向的变量赋值
    *ptr1=*ptr2;
    *ptr2=temp;
}
int main()
{
    int num1 = 15;
    int num2 = 20;
    swap(&num1,&num2);
    
    printf("num1=%d num2=%d\n",num1,num2);
    
    return 0;
}

注意,指针作为函数参数,称为地址传递方式;与之相对应的称为值传递,它们之间有着明显区别:

  • 值传递:值传递是直接将参数的值复制到函数内部的参数中。在函数执行完毕后,这个复制的值可能会被改变,但原始变量的值并不会受到影响。这是因为函数内部操作的是这个值的副本,而不是原始变量本身。

  • 地址传递是传递参数的地址(或引用)到函数中。这意味着函数内部操作的是原始变量的内存地址,因此任何在函数内部对这个地址的修改都会直接影响到原始变量。

其中,值传递的优缺点分别是:

  • 优点:简单、安全,因为函数内部的操作不会影响到函数外部的变量。

  • 缺点:如果传递的参数是大型对象或数据结构,那么复制这些值可能会消耗大量的时间和内存。

地址传递的优缺点分别是:

  • 优点:对于大型对象或数据结构,地址传递可以避免复制值带来的性能开销,因为它只需要传递一个地址(或引用)。

  • 缺点:由于函数内部可以直接修改原始变量,这可能导致函数外部的变量被意外地改变,从而增加了出错的可能性。因此,使用地址传递时需要更加小心。

值传递代码示例:

void swap(int num1,int num2)
{
    int temp = num1;
    num1 = num2;
    num2 = temp;
}
int main()
{
    int num1 = 15;
    int num2 = 20;
    swap(num1,num2);
    
    printf("num1=%d num2=%d\n",num1,num2);
    
    return 0;
}

2 指针作为函数返回值

指针可以作为函数的返回值。这通常用于动态分配内存并返回指向该内存的指针,或者返回指向某个已存在对象的指针。语法格式:

// 函数返回类型的指针  
数据类型* 函数名(参数列表);

代码示例:

#include <stdio.h>  
#include <stdlib.h> 

// 函数返回一个整数的指针  
int* createArray(int size) 
{  
    // 动态分配一个整数数组  
    int* array = (int*)malloc(size * sizeof(int));  
    if (array == NULL) 
    {  
        // 内存分配失败,返回NULL  
        return NULL;  
    }  
      
    // 初始化数组(可选)  
    for (int index = 0; index < size; ++index) 
    {  
        array[index] = 0;  
    }  
      
    // 返回指向数组的指针  
    return array;  
}  
  
int main() 
{  
    // 调用函数创建数组  
    int* myArray = createArray(10);  
    if (myArray != NULL) 
    {  
        // 使用数组...  
        for (int index = 0; index < 10; ++index) 
        {  
            printf("%d ", myArray[index]);  
        }  
        printf("\n");  
          
        // 释放内存  
        free(myArray);  
    } 
    else 
    {  
        printf("内存分配失败!\n");  
    }  
      
    return 0;  
}

注意,返回地址的时候,地址指向的内存的内容不能释放。如果返回的指针指向的内容已经被释放了,返回这个地址,也没有意义了。例如,

#include <stdio.h>  
  
// 错误的示例:返回指向局部变量的指针  
int* GetLocalVar() 
{  
    int localVar = 42; // 局部变量  
    
    return &localVar; // 返回指向局部变量的指针  
}  
  
int main() 
{  
	// 获取指向局部变量的指针  
    int* ptr = GetLocalVar();
    
    // 尝试访问该值,但此时内容可能已失效  
    printf("The value is: %d\n", *ptr); 
    
    return 0;  
}

上述函数返回了指向局部变量的指针,那么这是非常危险的,因为局部变量在函数返回后其存储空间可能会被覆盖或释放,导致指针指向的内容失效。

3 函数指针

函数指针是一个特殊的指针,它指向某个函数的入口地址,使得程序可以通过该指针变量调用该函数。函数指针本身是一种数据类型,其类型与所指向函数的返回值类型和参数类型有关。函数指针的定义方式类似于变量的定义,只不过需要加上括号和星号。

函数指针定义,语法格式

// 先定义函数
返回类型 函数名(形参列表);
// 定义函数指针
返回类型 (*函数指针名)(形参列表);
// 函数指针指向函数
函数指针名 = 函数名;

代码示例:

#include <stdio.h>  
  
void print_number_one(int num) 
{  
    printf("Function 1: The number is %d\n", num);  
}  
  
void print_number_two(int num) 
{  
    printf("Function 2: The number is %d\n", num);  
}

int main()
{  
	void (*function_pointer)(int);
	
    // 指向第一个函数  
    function_pointer = print_number_one;  
    function_pointer(10); // 输出:Function 1: The number is 10  
  
    // 指向第二个函数  
    function_pointer = print_number_two;  
    function_pointer(20); // 输出:Function 2: The number is 20  
  
    return 0;  
}

函数指针在C/C++编程语言中非常有用,它们可以用于多种情况:

  • 声明函数指针:函数指针的声明格式为“返回类型(*指针变量名)(参数类型列表)”。例如,int (*p)(int, int)表示p是一个指向返回类型为int,参数类型为int, int的函数指针。

  • 定义函数指针:定义函数指针时,需要将其指向具体的函数。例如,如果有一个函数int add(int a, int b) { return a + b; },那么可以通过int (*p)(int, int) = add;将函数指针p指向函数add。

  • 函数指针作为参数:函数指针还可以作为函数的参数,这种用法通常用于回调函数。这样,可以通过一个函数调用不同的回调函数,实现不同的操作。

  • 函数指针数组:函数指针还可以放在数组中,形成函数指针数组。

4 函数指针数组

函数指针数组是一个存储了多个函数指针的数组。每个数组元素都是一个指向函数的指针,这些函数具有相同的返回类型和参数列表。语法格式:

返回数据类型 (*数组名[])(参数列表);

代码示例:

#include <stdio.h>  
  
// 定义两个函数,它们将被存储在函数指针数组中  
void PrintFun1() 
{  
    printf("调用的是PrintFun1\n");  
}  
  
void PrintFun2() 
{  
    printf("调用的是PrintFun2\n");  
}  
  
int main() 
{  
    // 定义函数指针数组,可以存储两个返回类型为void、没有参数的函数指针  
    void (*funPtrArr[])() = {PrintFun1, PrintFun2};  
      
    // 通过函数指针数组调用函数  
    funPtrArr[0](); // 输出: Function 1 called  
    funPtrArr[1](); // 输出: Function 2 called  
      
    return 0;  
}

注意,函数指针数组中的所有函数指针必须指向具有相同返回类型和参数列表的函数。如果尝试将具有不同签名(即返回类型或参数列表不同)的函数指针存储在同一个数组中,编译器将会报错。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值