c语言指针(深入了解指针)

news2024/10/5 12:58:05

前沿:

      有人曾说过不会指针等于没有学习c语言,而我也是非常认同这个观点的,要想学习好c语言,指针是比不可缺少的,如果指针学不会c语言也就没办法学好,而向如此越重要的东西越比较难学,但难学并不代表学不会,这片文章将由简单到复杂让你深刻的了解指针,从此不再害怕指针的学习。

思维导图:

目录

一、指针的初步认识:

1.1指针是什么:

1.2指针的类型:

1.3指针的大小:

1.4指针的定义和使用:

 1.5野指针:

二、指针的进阶认识:

2.1字符指针:

2.2指针数组:

2.3数组指针:

2.4函数指针:

2.5函数指针数组:

 总结:


一、指针的初步认识:

1.1指针是什么:

  1.1.1  学习一个知识我们需要了解他的本质,正如我们学习指针,我们要先了解他是什么,从生活中举例,指针就好比我们家的门牌号一样,每个房间都有自己的门牌号,而我们则可以通过门牌号找到我们想要找到的人,别人也可以通过门牌号找到你。为此我们可以初步的了解指针就表示地址,通过指针我们可以找到相应类型的变量。所以我们平常口语中的指针其实就是指针变量,是用来存放内存地址的变量。

1.1.2 这样我们就好理解一个变量所占的内存叫做字节,而地址指向的就是所占字节的位置也就是地址。

这里的0xFFFFFFFFF则就表示地址。

1.2指针的类型:

       1.2.1 像int double float char 类型一样指针也有自己相应的类型,而指针的类型就是指针所指向元素是什么类型,指向int的指针就是整型指针,指向char类型的就是字符型指针等等。

#include<stdio.h>
int main()
{
    int a;
    int *ra=&a;
    char b;
    char* rb=&b; //等等

    return 0;
}

1.3指针的大小:

1.3.1指针的大小都是确定的在32位的编译器下指针变量的大小都是4个字节,在64位编译器下指针的大小都是8个字节 为啥呢?要知道不管什么类型的指针都是指针变量,而指针变量都是用来存放地址的,所以指针变量的大小取决一个地址存放所需要的空间大小为此不同的编译器有不同的大小但一般都是4/8个字节。

1.4指针的定义和使用:

#include<stdio.h>
int main()
{
    int a=10; //在内存中申请空间 我们也知道局部变量是存放在栈中的
    int* p=&a; //定义一个指针,对变量a取地址用的符号是&,这里a占的字节是四个,这里是将a的4个字节的第一个字节的地址存放在p变量中,p就是一个之指针变量。
    printf("%d\n",*p); //*p是对指针的解引用,得到地址所指向的数
    printf("%p\n",p);  //这里可以得到p的地址
    return 0;
}

 1.5野指针:

  1.5.1 什么是野指针:野指针就是指针指向的位置是不可知的(随机的、不正确的、没有明确限制的)

1.5.2野指针形成的原因:

<1>没有初始化

#include<stdio.h>
int main()
{
    int *p; //局部变量的指针没有初始化,默认随机值 这就是个野指针
    return 0;
}

<2>数组的越界

#include<stdio.h>
int main()
{
    int arr[5]={1,2,3,4,5}
    int* p=arr;
    for(int i=0;i<6;i++)
        {
            * (p++)=i;
        }
    return 0;
}

<3>指针指向的空间释放

#include<stdio.h>
#include<stdlib.h>
int main()
{
    int *p=(int*)malloc(40);
        free(p);
    return 0;

}

1.5.3如何避免野指针:

<1>. 指针初始化

<2>. 小心指针越界
<3>.  指针指向空间释放即使置 NULL
<4>. 避免返回局部变量的地址
<5>. 指针使用之前检查有效性

二、指针的进阶认识:

2.1字符指针:

  #include<stdio.h>
    int main()
{
    char  ch='m';
     char *pr=&ch; //存放字符的指针    
    const char *pr1="hello xiaoma" //存放字符串的指针,他是存放字符传的第一个地址并不是全部地址的存放. 
 //为什要加个const 要知道权限不能被扩大,只能被缩小,*pr1指向的是数组常量不能被改变所以用const来进行修饰。
      printf("%c %s\n",*pr,pr1); //打印字符,和字符串
    return 0;
}

2.2指针数组:

2.2.1:指针数组,重点在于数组,是用来存放指针的数组。

int* arr[10]; 存放整型指针的数组
double* arr1[10];存放double型指针的数组
char* arr[10] ; 存放字符型指针的数组

2.2.2指针数组的应用,可以吧几个一维数组合并成一个二维数组一样。

#include<stdio.h>
int main()
{
      int arr1[]={1,2,3,4};
      int arr2[]={3,4,5,6};
      int arr3[]={7,8,9,10};
      int* arr[3]={arr1,arr2,arr3};
       for(int i=0;i<3;i++)
        {
             for(int j=0;j<4;j++)
                 {
                    printf("%d ",arr[i][j]);
                  }
                printf("\n");
        }

    return 0;
}

2.3数组指针:

2.3.1 数组指针,重点在于指针,他是指向数组的指针。

int arr[10]={0};
int *p=arr //这里表示p指向arr首元素的地址。这里是整型指针
int (*p1)[10]=&arr;  //这里表示p1指向整个数组的地址 这是数组指针

2.3.2 数组指针的应用:数组指针不怎么用在一维指针上大多是用在二维或者三维指针上

#include<stdio.h>
void print(int (*p)[3],int x,int y)
{
    for(int i=0;i<3;i++)
    {
        for(int j=0;j<3;j++)
        {
            printf("%d ",*(*(p+i)+j));  //先对p解引用表示指向第一行的地址 加i就是指向第i行的地址 加j表示指向第j列的地址 最后在解引用就得到数值
        }
        printf("\n");
    }
}
int main()
{
    int arr[3][3]={1,2,3,2,3,4,3,4,5};
    print(arr,3,3);
    return 0;
}

2.4函数指针:

2.4.1通过数组指针我们类比函数指针,数组指针是指向数组的指针,那么函数指针呢,就是指向函数的指针。

int Add(int x,int y)
{
    return x+y;
}
int (*p)(int,int)=&Add;  //这就是定义一个函数指针p指向的就是Add函数的地址,没错函数也是有自己的地址的

2.4.2函数指针的简单应用:


#include<stdio.h>
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 Func(int (*p)(int ,int ),int x,int y)//传的是函数指针
{
    
    int ret = p(x,y);
    printf("%d\n",ret);
}                     //可能觉得没什么用 看一下复杂的应用就知道指针函数的应用

int main()
{
    Func(Add,2,3);
    Func(Sub,2,3);
    Func(Mul,3,3);  //这里就是一个函数来实现了加减乘除 而要一个函数实现四个函数的功能则需要运用函数指针
    Func(Div,6,2);
    return 0;
}

2.4.3函数指针的复杂应用:

   ~~回调函数回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个 函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数 的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进 行响应。

         我们上面实现的简单应用就是一个回调函数,而现在我们要实现一个比较复杂的回调函数:

qsort函数,我们先去了解一下qsort函数怎么使用,这里我比较推荐一个网站就是专门用来查看c/c++库函数:https://legacy.cplusplus.com/https://legacy.cplusplus.com/这里我们来简单的讲述qsort函数:qsort的函数是通过快排的方法来实现不同类型的排序问题,我们来通过代码展示他的使用并且自己来实现一个qsort函数

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
int swap(const void*p1,const void*p2)
{
    return *(int*)p1-*(int*)p2;
}
int main()
{
    int arr[10]={1,3,6,2,5,4,8,7,10,9};
    int sz=sizeof(arr)/sizeof(arr[0]); //求数组的长度
    qsort(arr,sz,sizeof(int),swap); //第一个要传排序对象的地址,然后排序的长度,宽度,最后排序的判断方法。
    for(int i=0;i<sz;i++)
    {
        printf("%d ",arr[i]);
    }
    return 0;
}

在用qsort按名字排一个结构体:

#include<stdio.h>
#include<string.h>
#include<stdlib.h>
struct Student
{
    char name[20];
    int age ;
};
int swap(const void*p1,const void*p2)
{
    return strcmp(((struct Student*)p1)->name,((struct Student*)p2)->name);
}

int main()
{
    
    struct Student arr[]={{"xiaomma",18},{"lisi",17},{"wanger",20}};
    int  sz=sizeof(arr)/sizeof(arr[0]);
    qsort(arr,sz,sizeof(struct Student),swap);
    for(int i=0;i<sz;i++)
    {
        printf("%s\n",arr[i].name);
    }
    
    return 0;
}

创建自己的qsort函数m_qsort :

这里m_qsort我用的是冒泡法,虽然效率低了,但是我们要掌握的是函数指针的用法!

void Swap(char*buf1,char*buf2,int width){
    int i=0;
    for(i=0;i<width;i++)  //进行位置的交换
    {
        char tmp=*buf1;
        *buf1=*buf2;
        *buf2=tmp;
        buf1++;
        buf2++;
    }
}
  //这里是实现和qsort一样的功能不过我使用的冒泡的形式来实现的
   void bubble_sort(void* base ,int sz,int width,int (*cmp)(const void*e1,const void*e2))
{
    int i=0;
    int j=0;
    for(i=0;i<sz-1;i++)
    {
        for(j=0;j<sz-i-1;j++)
        {
            if(cmp((char*)base+j*width,(char*)base+(j+1)*width)>0)
            {
                Swap ((char*)base+j*width,(char*)base+(j+1)*width,width);
                //因为比较类型不同所以用char传地址比较好,char才占一个地址,我们知道总长度也知道宽度就好进行交哈。
            }
        }
    }
}

int cmp_int(const void* e1,const void*e2)
{
    return (*(int *)e1-*(int *)e2);
}
int main()
{
    int arr[]={9,8,7,6,5,4,3,2,1,0};
    int sz=sizeof(arr)/sizeof(arr[0]);
    bubble_sort(arr,sz,sizeof(arr[0]),cmp_int);
    for(int i=0;i<sz;i++)
    {
        printf("%d ",arr[i]);
    }
    return 0;
}

  代码很长但很有必要自己来实现一下就能够深入的理解函数指针存在的意义啦,如果要有不理解的地方可以直接问我的。^ _ ^

2.5函数指针数组:

       2.5.1函数指针数组也就是存放函数指针的数组

    

 int Add(int x,int y)
{
    return x+y;
}
 int (*p)(int ,int )=&Add //函数指针 
 int (*arr[4])(int,int ); //函数指针数组  //通过对比我们就很容易发现就是把p换成一个数组 他的类型就是函数指针数组 
 int (*(*prr)[5])(int ,int )//函数指针数组的指针。哈哈哈哈类似于套娃是的 就是便于大家取了解这个东西
 

  2.5.2函数指针数组的应用:也是很容易的可以把我们上面写的四个功能用一遍

#include<stdio.h>
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;
}


int main()
{
    int i=0;
int (*arr[])(int ,int )={Add,Sub,Mul,Div};
          for(i=0;i<4;i++)
        {
            printf("%d\n",arr[i](6,2));
        }
    return 0;
}

 总结:

1、从指针的初级指针是什么,和如何使用到指针的进阶一大堆不同的指针,这里小马都已经全部写完,还有一个结构体指针,我觉得看过我的链表就应该很容易的掌握,这里我都不进行写啦,我觉得比较难的点就在于函数指针这里的qsort函数自己的实现,我认为很有必要多看几遍自己在写一遍,才能深入理解qsort函数,和函数指针。

2、最后码文不易,如果觉得文章有帮助的话,请多多关注,希望我们能够一同进步,共同发展!!!!^ _ ^

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/5205.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Qt基础之四:Qt信号与槽机制原理及优缺点

信号和插槽用于对象之间的通信。信号和槽机制是Qt的核心特性&#xff0c;可能也是与其他框架提供的特性最大不同的部分。Qt的元对象系统使信号和槽成为可能。 一.简介 在GUI编程中&#xff0c;当我们改变一个控件&#xff0c;通常希望其他控件被通知到。更一般的&#xff0c;…

【优化覆盖】基于matlab果蝇算法求解无线传感器覆盖优化问题【含Matlab源码 2215期】

一、⛄果蝇算法 果蝇优化算法(Fruit Fly Optimization Algorithm, FOA) 是台湾学者潘文超经过研究发现, 并于2011年提出的一种新型智能优化算法, 它具有很好的全局优化性能, 能够解决很多的优化求解问题.果蝇有着优于其它生物的感官知觉, 特别是视觉与嗅觉, 依靠灵敏的嗅觉, 果…

Linux服务器被入侵后的排查思路(应急响应思路)

目录Linux-暴力破解、替换ps命令并留存后门事件复现一、事件背景二、应急响应过程排查crontab后门排查是否有命令被替换响应过程回顾三、事件还原查看后门排查安全日志Linux-暴力破解、替换ps命令并留存后门事件复现 一、事件背景 服务器疑似被入侵&#xff0c;与恶意IP进行通信…

手眼标定笔记

文章目录基本介绍&#xff1a;坐标系变换运算规则&#xff1a;关系运算说明&#xff1a;坐标系运算规则一&#xff1a;坐标系运算规则二&#xff1a;齐次坐标系&#xff1a;齐次坐标系下的坐标变换&#xff1a;眼在手外&#xff1a;眼在手内&#xff1a;解方程&#xff1a;- Ta…

Spring学习第4篇:Spring 的依赖注入

大家家好&#xff0c;我是一名网络怪咖&#xff0c;北漂五年。相信大家和我一样&#xff0c;都有一个大厂梦&#xff0c;作为一名资深Java选手&#xff0c;深知Spring重要性&#xff0c;现在普遍都使用SpringBoot来开发&#xff0c;面试的时候SpringBoot原理也是经常会问到&…

数据结构 | 单链表

… &#x1f333;&#x1f332;&#x1f331;本文已收录至&#xff1a;数据结构 | C语言 更多知识尽在此专栏中!文章目录&#x1f333;前言&#x1f333;正文&#x1f332;链表打印与销毁&#x1fab4;打印&#x1fab4;销毁&#x1f332;尾部插入与删除&#x1fab4;节点申请&…

javaweb JAVA JSP销售系统购物系统jsp购物系统购物商城系统源码(jsp电子商务系统)网上在线销售

JSP销售系统购物系统jsp购物系统购物商城系统源码&#xff08;jsp电子商务系统&#xff09;网上在线销售

Linux基础 - 虚拟化介绍(KVM)

‍‍&#x1f3e1;博客主页&#xff1a; Passerby_Wang的博客_CSDN博客-系统运维,云计算,Linux基础领域博主 &#x1f310;所属专栏&#xff1a;『Linux基础』 &#x1f30c;上期文章&#xff1a; Linux基础 - 服务管理&#xff08;systemd&#xff09; &#x1f4f0;如觉得博…

爬取医药卫生知识服务系统的药品数据——超详细流程

爬取医药卫生知识服务系统的药品数据——超详细流程 文章目录爬取医药卫生知识服务系统的药品数据——超详细流程前言一、寻找药品数据二、爬取药品ID1.资源获取2.数据提取3.资源保存4.主函数5.总体代码三、爬取药品信息1.加载资源ID2.获取数据3.数据提取4.保存信息5.主函数6.总…

SpringBoot-属性绑定和bean属性校验

目录 属性绑定 自定义类属性绑定 第三方bean属性匹配 规则:松散绑定&#xff08;宽松绑定&#xff09; Bean属性校验 属性绑定 属性绑定&#xff1a;我们可以使用配置文件对类的属性进行赋值绑定。 自定义类属性绑定 我们自定义一个类&#xff0c;在此使用yml文件进行类…

【滤波器】基于matlab实现微波带低通高通带通滤波器设计

✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;修心和技术同步精进&#xff0c;matlab项目合作可私信。 &#x1f34e;个人主页&#xff1a;Matlab科研工作室 &#x1f34a;个人信条&#xff1a;格物致知。 更多Matlab仿真内容点击&#x1f447; 智能优化算法 …

python科研做图系列之雷达图

文章目录参考资料matplotlib库画的复现一个 pyecharts的雷达图尝试在上面的基础上&#xff0c;把pyecharts 导出存为一般的png图尝试在上面的基础上&#xff0c;把pyecharts 导出存为一般的矢量图用pygal画雷达图参考资料 参考知乎 CSDN给出了一些参数 matplotbib库雷达图官网 …

Python实现九九乘法表的几种方式,入门必备案例~超级简单~

前言 嗨喽~大家好呀&#xff0c;这里是魔王呐 ❤ ~! 们在学习Python的过程中需要不断的积累和练习&#xff0c;这样才能够走的更远&#xff0c; 今天一起来学习怎么用Python写九九乘法表~ 第一种方法&#xff1a;for-for 代码&#xff1a; for i in range(1, 10):for j in…

数据挖掘面试经总结【私人版,仅供参考】

1特征归一化 1.1为什么需要对数值类型的特征做归一化&#xff1f; 线性函数归一化零均值归一化 1.2在对数据进行预处理时&#xff0c;应该怎样处理类别型特征&#xff1f; 序号编码独热编码二进制编码 1.3什么是组合特征&#xff1f;如何处理高维组合特征&#xff1f; 例…

【python】云打印实现

这两天为了实现云打印功能找了很多相关的文章 记录一下这一篇&#xff0c;python云打印实现-朝花夕拾&#xff0c;代码通过监听文件夹有无产生新文件来判断是否执行&#xff0c;我尝试运行了下没问题&#xff0c;于是打算转载一下 程序运行结果 由于对方的代码和我实现的有点出…

【Maven】你好,Maven >>> 与Maven的初次见面~

个人主页&#xff1a;.29.的博客 学习社区&#xff1a;进去逛一逛~ 个人简介&#xff1a;Java领域新星创作者&#xff1b;阿里云技术博主、星级博主、专家博主&#xff1b;正在Java学习的道路上摸爬滚打&#xff0c;记录学习的过程~ 与Maven的初次见面~一、了解Maven二、Maven的…

Flink双流join导致数据重复

大家都知道flink sql 中 left join 数据不会互相等待&#xff0c;存在 retract 问题&#xff0c;会导致写入 kafka 的数据量变大&#xff0c;就会导致出现数据重复的问题。 举例&#xff1a;即常见的曝光日志流&#xff08;show_log&#xff09;通过 log_id 关联点击日志流&am…

SQL:数据去重的三种方法

1、使用distinct去重 distinct用来查询不重复记录的条数&#xff0c;用count(distinct id)来返回不重复字段的条数。用法注意&#xff1a; distinct【查询字段】&#xff0c;必须放在要查询字段的开头&#xff0c;即放在第一个参数&#xff1b;只能在SELECT 语句中使用&#…

spring整合fastdfs客户端

解决Dependency ‘com.github.tobato:fastdfs-client:1.27.2’ not found 错误问题。 一、 将fastdfs客户端git下来 git https://github.com/happyfish100/fastdfs-client-java.gitcd fastdfs-client-java然后将fastdfs-client-java构建到本地maven仓库 mvn clean install&…

Pandas的数据结构

Pandas的数据结构 处理CSV 文件 CSV&#xff08;Comma-Separated Values&#xff0c;逗号分隔值&#xff0c;有时也称为字符分隔值&#xff0c;因为分隔字符也可以不是逗号&#xff09;&#xff0c;其文件以纯文本形式存储表格数据&#xff08;数字和文本&#xff09;。 Pan…