「2」指针进阶——详解

本文介绍了C语言中函数指针的概念,包括指向函数指针数组的指针和回调函数的使用。同时,详细讲解了快速排序和冒泡排序的实现,以及标准库函数qsort的使用,包括自定义比较函数。示例代码展示了如何使用这些概念进行实际编程操作。
摘要由CSDN通过智能技术生成
🚀🚀🚀大家觉不错的话,就恳求大家点点关注,点点小爱心,指点指点🚀🚀🚀 

目录

🐰指向函数指针数组的指针(很少用,了解)

🐰回调函数(通过函数指针调用函数)

🐰快速排序

🌸冒泡排序

🌸qsort()

🐰用冒泡排序类似实现qsort


🐰指向函数指针数组的指针(很少用,了解)

#include<stdio.h>
void Add(int ,int)
{
    printf("%d\n",1+1);
}
void Sub(int ,int)
{
    printf("%d\n",1-1);
}
int main()
{
    int (*pf)(int,int)=Add;//函数指针
    int (*pfArr[4])(int,int)={Add,Sub};//函数指针数组
    int (*(*ppfArr)[4])(int,int)=&pfArr;//ppfArr就是指向函数的指针数组的指针
    return 0;
}

🐰回调函数(通过函数指针调用函数)

通过回调函数实现 两个操作数的加减乘除:
#include<stdio.h>
void Calc(int(*pf)(int,int))
{
    int x=0,y=0;
    printf("请输入两个操作数\n");
    scanf("%d %d",&x,&y);
    int ret=pf(x,y);
    printf("%d\n",ret);
}
int Add(int x,int y)
{
    return x+y;
}
int Sub(int x,int y)
{
    return x-y;
}
int Mul(int x,int y)
{
    return x*y;
}
int Div(int x,int y)
{
    return x/y;
}
void menu(void)
{
    printf("****    两位数的计算器     ****\n");
    printf("****    1.Add   2.Sub   ****\n");
    printf("****    3.Mul   4.Div   ****\n");
    printf("****    0.exit          ****\n");
}
int main()
{

    int input=0;
    do
    {
        menu();
        printf("请选择\n");
        scanf("%d",&input);
        switch(input)
        {
            case 1:
                Calc(Add);
                break;
            case 2:
                Calc(Sub);
                break;
            case 3:
                Calc(Mul);
                break;
            case 4:
                Calc(Div);
                break;
            case 0:
                printf("exit\n");
                break;
            default:
                printf("输入错误\n");
        }
    }while(input);
}

🐰快速排序

qsort是一个库函数,是用来排序(使用的快速排序的方法)
1.库函数里的,可以直接使用        2.可以排序任意类型的数据

🌸冒泡排序

在这里插入图片描述

 #include<stdio.h>

void Bubble(int arr[],int len)
{
    int i=0,j=0;
    for(i=0;i<len-1;i++)
    {
        for(j=0;j<len-1-i;j++)
        {
            if(arr[i]>arr[j+1])
            {
                int temp=arr[j];
                arr[j]=arr[j+1];
                arr[j+1]=temp;
            }
        }
    }
}
void Print(int arr[],int len)
{
    for(int i=0;i<len;i++)
    {
        printf("%d ",arr[i]);
    }
    printf("\n");
}
int main()
{
    int arr[]={3,2,1,5,7,8,9,0};
    int len=sizeof(arr)/sizeof(arr[0]);
    Bubble(arr,len);
    Print(arr,len);
}

🌸qsort()

qsort的原型:void qsort (void* base, size_t num, size_t size,int (*compar)(const void*,const void*));
void qsort (void* base//指向了待排序数组的第一个元素的地址
            , size_t num(无符号整形)//待排序的元素个数
            , size_t size(无符号整形)//每个元素的大小,单位是字节
            ,int (*compar)(const void*,const void*)//这里是一个函数指针,指向一个函数,这个函数可以比较2个元素的大小
            );
比较函数:就是函数指针campar指向的函数,因为使用qsort时,要自己定义比较函数,以下是常见的比较函数
比较整形变量时
int cmp_int(const void* e1, const void* e2)
{
 return *(int*)e1 - *(int*)e2;
}
比较浮点型变量时
int cmp_float(const void* e1, const void* e2)
{
 return (int)(*(float*)e1 - *(float*)e2);
}
比较字符串变量时
int cmp_str_size(const void* e1, const void* e2)
{
 return strcmp((char*)e1,(char*)e2);
}
比较字符串长度时
int cmp_str_len(const void* e1, const void* e2)
{
 return strlen((char*)e1)-strlen((char*)e2);
}
比较结构体变量时
int cmp_by_age(const void*e1, const void*e2)
{
 return (int)((stu*)e1)->weight - ((stu*)e2)->weight));
}

cmp函数的返回值:返回值<0(不进行置换),>0(进行置换),0(不进行置换)。记得返回的结果一定是整形的,如果不是需要强制转为整形的

‼️注:void*的指针不能解引用,也不能算术运算
下面是使用qsort排序整形变量和结构体变量的原码:
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int cmp_int(const void* e1,const void* e2)//对整形比较
{
    return *(int*)e1-*(int*)e2;
}

void test_1()
{
    int arr[]={2,3,4,5,6,7,1};
    int sz=sizeof(arr)/sizeof(arr[0]);//计算出数组下标,就不用手动去数有多少个元素了
    //这里需要提供一个比较函数,这个比较函数能够比较2个整数的大小
    //qsort默认为升序
    qsort(arr,sz,sizeof(arr[0]),cmp_int);
    for(int i=0;i<sz;i++)
    {
        printf("%d ",arr[i]);
    }
}
struct stu//定义了一个包含字符类型,整形,浮点型的结构体
{
    char name[20];
    int age;
    float weight;
};
int sort_by_name(const void* e1,const void* e2)//对字符串的比较
{
    return strcmp(((struct stu*)e1)->name,((struct stu*)e2)->name);
}

int sort_by_age(const void* e1,const void* e2)//对整形的比较
{
    return ((struct stu*)e1)->age-((struct stu*)e2)->age;
}

int sort_by_weight(const void* e1,const void* e2)//对浮点型比较
{
    return ((struct stu*)e1)->weight-((struct stu*)e2)->weight;
}
void test_2()//对结构体进行排序
{
    struct stu s[3]={{"zhangsan",23,65.5},{"lisi",27,56.5},{"wangwu",24,64}};
    int sz=sizeof(s)/sizeof(s[0]);
    qsort(s, sz, sizeof(s[0]), sort_by_name);//对名字排序
    for(int i=0;i<sz;i++)//输出排序后的结果
    {
        printf("%s ",s[i].name);
    }
    printf("\n");
    qsort(s, sz, sizeof(s[0]), sort_by_age);//对年龄排序
    for(int i=0;i<sz;i++)
    {
        printf("%d ",s[i].age);
    }
    printf("\n");
    qsort(s, sz, sizeof(s[0]), sort_by_weight);//对体重排序
    for(int i=0;i<sz;i++)
    {
        printf("%.2f ",s[i].weight);
    }
    printf("\n");
}
int main()
{
    test_1();
    test_2();
    return 0;
}

🐰用冒泡排序类似实现qsort

 qsort()的底层是快速排序,但是没有学过快速排序,可以使用冒泡排序来代替
#include<stdio.h>
void swap(char* buf1,char*buf2,int width)
//为什么不直接进行交换,而是交换每个字节的内容?这是因为这交换的不只是整形变量,这里还可以交换其它类型的变量

{
    for(int i=0;i<width;i++)
    {
        char temp=*buf1;
        *buf1=*buf2;
        *buf2=temp;
        buf1++;
        buf2++;
    }
}
void buble_sort(void* base,int sz,int width,int (*cmp)(const void*e1,const void*e2))//这里的函数指针可以方便调用各种类型比较,不同类型的变量比较,可以调用不同类型的比较函数
{
    int i=0,j=0;
    //sz个元素就有sz-1趟
    for(i=0;i<sz-1;i++)
    {
        for(j=0;j<sz-1-i;i++)
        {
            //两个元素的比较
            //arr[j] arr[j+1]
            if(cmp((char*)base+j*width,(char*)base+(j+1)*width)>0)//为什么将base强制转化为(char*)呢?假如比较的是整形变量,我们将base转化为(char*),加上width(就是这里的整形变量的大小,4字节)就可以找到下个元素的地址,
            {
                //交换
                swap((char*)base+j*width,(char*)base+(j+1)*width,width);//然后把这个变量的地址传给交换函数
            }
        }
    }
}
int cmp_int(const void* e1,const void* e2)//对整形比较
{
    return *(int*)e1-*(int*)e2;
}
int main()
{
    int arr[10]={2,3,4,5,6,7,1,9,13,10};
        int sz=sizeof(arr)/sizeof(arr[0]);
        buble_sort(arr,sz,sizeof(arr[0]),cmp_int);
        for(int i=0;i<sz;i++)
        {
            printf("%d ",arr[i]);
        }
    return 0;
}

 🌸🌸🌸如果大家还有不懂或者建议都可以发在评论区,我们共同探讨,共同学习,共同进步。谢谢大家! 🌸🌸🌸

  • 38
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 31
    评论
评论 31
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值