C语言第八讲,指针*

            C语言第八讲,指针*

一丶简单理解指针

说到指针,很多人都说是C语言的重点. 也说是C语言的难点.

其实指针并不是难.而是很多人搞不清地址 和 值.以及指针类型.

为什么这样说.

假设有两个变量,如下

int nNum1 = 1;
int nNum2 = 0x00401000;

变量nNum1 保存的值是1

变量nNum2 保存的是一个16进制数值.

那么如果我们把保存16进制的这个变量.当作指针. 也就是说它保存了一个16进制数值而已. 我们可以通过16进制找到里面存储的值.

但是因为是保存地址的变量.所以我们需要加上特殊符号进行定义.

例如:

  

int *nNum2 = 0x004010000;

指针其实就是保存了一个特殊的数值而已. 而通常这个数值我们说是地址.

如果我们保存了1的地址,那么我们就可以访问1了.

怎么访问?

可以通过取内容符号进行访问.

int nNum = 1;
int *p = &nNum1; //保存1的地址,我们可以不用管.也可以理解为一个16进制数值而已.
*p = 3;  //修改地址空间的值.

二丶指针的数据类型

说到指针.其实我们说的并不是保存地址的难点.难点是该如何解释这个地址.

例如我们知道定义一个变量.我们可以知道这个变量的内存空间跟随这数据类型走.

例如:

  

int nNum1 = 3;  //nNum1 大小是4个字节.  则 sizeof(nNum1) = 4;
short int nNum2 = 2; // nNum2 大小是2个字节. 则sizeof(nNum2) = 2;
....

那么我们指针其实也一样.也和普通变量一样.

int nNum1 = 2;
int *p = &nNum1;  p的大小也是4个字节. 跟地址无关.

比如我们定义指针的是否有很多数据类型解释.

例如:

  

int *p;

short *p;

double *p;

那么地址解释的方式也就不一样.

根据上图可以得知,1所占的内存空间是4个字节,而不同的类型则有不同的解释方式.

如果解释为short *,则获得的是1的高位,如果解释为int *则是整个的1所代表的值.

三丶指针的运算

指针和我们普通变量运算不一样.

普通变量进行运算

int nNum1 = 1;
nNum1 ++;  //nNum1 = 2;

指针变量进行运算

int *p = 0x00401000;
p++;  则P的值是0x00401004;
因为我们的类型是 int *解释方式,所以每次+1则是加一个int类型大小.而因为指针就是地址.
所以+1的时候就是+4

指针相减也是一样的.

四丶指针和数组的的组合

学习了指针,就要说下指针应用的场合. 数组和指针一般是很常见的.

例如定义数组:

  

int nNum1[10] = {1,2,3,4,5,6,7,8,9,10};

指针可以进行访问数组.

int *p = nNum1; //注意,数组是一个数据结构.所以我们不用加取地址符号.数组名就是数组的首地址.
如果我们是变量,则需要加上&符号.
例如:
   int n = 10;
    int *p = &n;
指针进行访问:
    *p = 10; //数组元素的第一项更改为10
    p++;      //指针++,根据数据类型来相加,+1等于+4,则到数组的第二个元素了.

一般遍历数组的时候我们会封装成一个函数.

则会写成下面这样.

void IteratorArr(int *p, int Count)
{
    while (Count--)      //遍历的个数自减
    {
        printf("%d \t",*p);
        p++;              //数组元素自加
    }
    puts("\r\n");
}
int main()
{
    int Arr[] = { 1,2,3,4,5,6,7,8,9,10 };
    IteratorArr(Arr,sizeof(Arr)/sizeof(Arr[0]));//传入数组首地址即可.但是我们函数的参数要定义为指针. 还需要传入数组的个数.进行遍历.
    system("pause");
    return 0;
}

输出结果.

五丶指针和函数的应用.

指针一般也用于函数的传参.例如上面我们遍历数组.其中我们的函数的形参就是指针.

如果在函数中想更改实际参数的值. 也可以用到指针.例如我们上一篇讲解函数的时候留下的一个小话题.

void ChangeValue(int *p, int *p1)
{
    *p = 0;        //传过来的是nNum1的地址,所以我们直接*地址对其更改即可.
    *p1 = 11;
}
int main()
{
    int nNum1 = 10;
    int nNum2 = 11;
    ChangeValue(&nNum1,&nNum2);
    printf("nNum1 = %d  nNum2 = %d\r\n",nNum1,nNum2);
    system("pause");
    return 0;
}

输出结果.

函数指针.

指针的高级应用还有函数指针. 就是我们定义一个指针. 这个指针是函数指针.可以指向自己定义的一个函数.

函数名字就是函数的首地址.

例如:

  

void ChangeValue(int *p, int *p1)
{
    *p = 0;
    *p1 = 11;
}
int main()
{
    int nNum1 = 10;
    int nNum2 = 11;
    ChangeValue(&nNum1,&nNum2);

    int(*p)(char const* const _Format, ...);         //定义函数指针
    p = printf;                                     //函数指针p指向了printf
    p("nNum1 = %d  nNum2 = %d\r\n",nNum1,nNum2);    //调用p实际和调用printf一样
    system("pause");
    return 0;
}

输出结果.

函数指针的语法

函数返回值 (*  指针变量名)(函数参数)

函数指针其实很好定义,只需要把函数的声明拿过来,然后函数名字前边加上括号. 括号里面加上*就可以了.

原文地址:https://www.cnblogs.com/iBinary/p/8483054.html

时间: 2024-08-28 17:38:55

C语言第八讲,指针*的相关文章

16位汇编第八讲指令第四讲

16位汇编第八讲指令第四讲 一丶串操作类指令 1.什么是串操作? 1.串操作指令是8086指令系统中比较独特的一类指令,采用比较特殊的数据串寻址方式,在操作主存连续区域 的数据是,特别好用.因而比较常用 简而言之,就是内存中的一段数据,拷贝/读取/修改... 到另一块另内存 重点掌握  MOVS  STOS  LODS CMPS SCAS REP 2.串操作的简介 1.串操作指令的操作数,是驻村中连续存放的数据串(String 注意string表示串的意思)--也就是一段数据在内存中 是连续的,

C语言第八课

主要内容:指针(C语言的代表,会C语言不会指针,恭喜你C语言白学了) 一.数据的存储 字节是最小的存储单位(byte),一个字节成为一个存储单位(内存单元),不同的数据类型占用的存储单元是不同的(1个字节是8位) char1个字节(byte)8位(bit) short2个字节(byte)16位(bit) int4个字节(byte)32位(bit) long8个字节(byte)64位(bit) float4个字节(byte)32位(bit) double8个字节(byte)64位(bit) 为了正

函数指针玩得不熟,就不要自称为C语言高手(函数指针是解耦对象关系的最佳利器,还有signal)

记得刚开始工作时,一位高手告诉我说,longjmp和setjmp玩得不熟,就不要自称为C语言高手.当时我半信半疑,为了让自己向高手方向迈进,还是花了一点时间去学习longjmp和setjmp的用法.后来明白那不单是跳来跳去那样简单,而是一种高级的异常处理机制,在某些情况下确实很有用. 事实上,longjmp和 setjmp玩得熟不熟与是不是C语言高手,不是因果关系.但是,如果可以套用那位高手的话,我倒想说如果函数指针玩得不熟,就不要自称为C语言高手.为什么这么说呢,函数指针有那么复杂吗?当然不是

C语言第四讲,typedef 关键字,以及作用域

C语言第四讲,typedef 关键字,以及作用域 一丶typedef关键字 在C语言中,有typedef 关键字,这个关键字的作用就是允许你为类型定义一个新的名字,也就是 起个别的名字 例如: typedef unsigned int DWORD 那么以后,我们定义unsigned int 类型的变量的时候,这不用敲写 unsigned int 了.直接DWORD即可. 例如: DWORD name = 0; 一般常常用在 结构体的定义,和函数指针上. 例如: typedef struct My

嵌入式Linux C语言(三)——指针与函数

嵌入式Linux C语言(三)--指针与函数 指针对函数的功能有巨大的贡献,指针能够将数据传递给函数,并且允许函数对数据进行修改.指针对于函数的作用主要有两方面:将指针传递给函数和声明函数指针. 一.程序的栈和堆 程序的栈和堆是C语言程序运行的运行时元素. 1.程序栈 程序栈是支持函数执行的内存区域,通常和堆共享一块内存区域,通常程序栈占据内存区域的下部,堆用内存区域的上部.程序栈存放栈帧,栈帧存放函数参数和局部变量.调用函数时,函数的栈帧被推倒栈上,栈向上长出一个栈帧,当函数终止时,函数的栈帧

Stanford机器学习---第八讲. 支持向量机SVM

本文原始文章见http://blog.csdn.net/abcjennifer/article/details/7849812,本文添加了一些自己的理解 本栏目(Machine learning)包括单参数的线性回归.多参数的线性回归.Octave Tutorial.Logistic Regression.Regularization.神经网络.机器学习系统设计.SVM(Support Vector Machines 支持向量机).聚类.降维.异常检测.大规模机器学习等章节.所有内容均来自Sta

Swift语言指南(八)--语言基础之元组

元组 元组(Tuples)将多个值组合为一个复合值.元组内的值可以是任何类型,各个元素不需要为相同类型(各个元素之间类型独立,互不干扰--Joe.Huang). 下例中,(404, "Not Found") 是一个描述HTTP状态码的元组.HTTP状态码是当你向WEB服务器请求页面时服务器返回的一个特殊值,如果你(向WEB服务器)请求了一个不存在的网页,返回的状态码就是 404 Not Found : 1 let http404Error = (404, "Not Found

嵌入式 Linux C语言(八)——存储类型、作用域、生命周期、链接属性

嵌入式 Linux C语言(八)--存储类型.作用域.生命周期.链接属性 一.存储类型 C语言中,每个变量和函数都有两个属性:数据类型和数据的存储类型. 变量的存储类型是指存储变量值的内存类型.变量的存储类型决定变量何时创建.何时销毁以及它的值将保持多久.计算机中有三个地方可以用于存储变量:普通内存,运行时堆和栈,硬件寄存器.变量的存储类型取决于声明变量的位置. C语言存储类别说明符: 说明符 用    法 auto 只在代码块内变量声明中被允许, 表示变量具有本地生存期 extern 出现在顶

嵌入式 Linux C语言(九)——C语言的安全问题和指针陷阱

嵌入式 Linux C语言(九)--C语言的安全问题和指针陷阱 C语言是灵活度和自由度较大的编程语言,作为C语言核心的指针更是让C语言程序员可以越过安全的栅栏,对某些内存区域进行破坏性访问,引发安全风险.很多安全问题都能追根溯源到指针的误用.本文将从指针的角度解读C语言常见的安全问题和指针陷阱. 一.指针的声明和初始化 1.不恰当的指针声明 int* ptr1, ptr2;//声明ptr1为int指针,ptr2为整型 int *ptr1, *ptr2;//ptr1,ptr2都声明为指针 #def