C语言指针——指针的运算

说指针之前,先说两个比较不相关的东西,也当做是一种知识的补充。看下面的代码:

#include <stdio.h>

int main(void)

{

char *s = "Hello world.";

printf(s);

}

运行这段代码,会出现错误吗?如果出现错误,是什么错误?

分析:printf一种常规的用法就是printf("Please input a data:");这种用法相信很多人都使用过,但是将这种用法分析一下,这样的用法printf函数接收到的参数是一个常量字符串的首地址,然后就可以输出这个字符串,所以上面的代码也是将字符串的首地址(存放于字符型指针s中)传递给了printf,所以这段代码的运行不会出现任何问题,甚至编译的时候连警告都不会出现。再看下面的一段代码:

#include <stdio.h>

int main(void)

{

char *s = "Hello world.";

printf(s, 1, 2, 3);

}

请问这段代码是否正确,如果不正确,错误出在哪里?

分析:要分析这个问题也还是比较容易的,printf函数和scanf函数是C语言中典型的两个变长参数函数,printf也不知道自己会接收到多少个参数,它唯一知道的就是如果在输入的字符串说明符中有多少个输出控制符(%d等)就会到后面去解析多少个参数,如果解析不到的话这种情况下才会报错。但是如果前面并没有说明需要参数,而后面跟了参数的话,这个它是不会理会的,所以这段代码编译和运行都不会出现问题,还是跟上面一样,编译的时候甚至警告都不会出现。

现在开始言归正传说说指针,这次就简单的说说指针的加减运算,请看下面一段代码:

#include <stdio.h>

int main(void)

{

int a[10];

printf("%p %p ", &a[0]+1, &a[1]);

}

请问程序输出的两个值一样吗?如果不一样,两者相差多少?

分析:这个问题需要对指针的运算理解比较深入才能答对,正确的答案是两者输出是一样的,可能很多人碰到这个问题的时候会想,&a[0]之后得到的是第0个单元的地址,这个地址和&a[1]也就是第1个单元的地址应该是相差sizeof(int)个字节,所以两者输出应该相差3。但是这样的算法忽略了指针的加减运算规则,指针或者说地址的运算往往依赖于指针或者地址的类型,比如我们常常会用int *p=a; printf("%d ", *p++);这种用法,通过p++就可以指向下一个int类型的单元!这就说明p里面的值被p++这条语句增加了4,这一层的运算程序员是看不见的,后面实质进行的运算相当于p=p+1*sizeof(int);这里也是一样,int型的地址加1,并不是地址值简单的加1,而是加了sizeof(int)个1,所以两者输出是一样的,再看一个更极端的例子:

#include <stdio.h>

int main(void)

{

printf("%d ", (int *)0x04-(int *)0x00);

}

这个程序的输出是多少呢?

程序运行之后输出了1,这个当然是在32位机上运行的结果,这个例子可以有助于理解上面的分析过程。

时间: 2024-09-29 23:44:06

C语言指针——指针的运算的相关文章

C语言 --- 高级指针

1. 指针赋值: C语言允许使用赋值运算进行指针的赋值,前提是两个指针具有相同的类型.                 int i,*p,*q;                 p = &i;                 q = p; 此时p和q存储的都是i的地址.                  *p = 1; 此时改变i的值为1;                  *q = 2; 此时改变i的值为2; 2. 指针作为函数的参数,可以实现在函数内部改变函数外部变量的值          

C语言的指针、链表的原理及学生管理系统

指针原理: 指针就是指向一个特定内存地址的一个变量.简化了的内存空间模型是按照从0到某一个数(比如1048575=1M-1)的一维线性空间,其中的每一个数对应一个存储单元,即1个字节.指针有两个属性:指向性和偏移性.指向性指的是指针一定要有一个确定的指向,偏移性则是体现指针重要应用的方面,即指针可以按程序员的要求向前或向后偏移. 指针的应用往往与数组联系在一起,为了方便说明问题,不妨从数组开始解释指针的偏移.数组就是许多的变量,它的一个重要特征就是在内存空间中连续地存放,而且是按下标顺序存放.比

[C语言]进阶|指针与字符串

------------------------------------------------------------------------------------ 回顾:[C语言]指针与字符串 指针的使用: /** * main.c by weiChen in 2015-5-4 */ #include <stdio.h> //定义函数体,使在main方法中能调用 void swap(int *pa, int *pb); void minmax(int c[], int len, int

C语言中指针和数组

C语言数组与指针的那些事儿 在C语言中,要说到哪一部分最难搞,首当其冲就是指针,指针永远是个让人又爱又恨的东西,用好了可以事半功倍,用不好,就会有改不完的bug和通不完的宵.但是程序员一般都有一种迷之自信,总认为自己是天选之人,明知山有虎,偏向虎山行,直到最后用C的人都要被指针虐一遍. 指针 首先,明确一个概念,指针是什么,一旦提到这个老生常谈且富有争议性的话题,那真是1000个人有1000种看法. 在国内的很多教材中,给出的定义一般就是"指针就是地址",从初步理解指针的角度来说,这种

C#委托与C语言函数指针及函数指针数组

C#委托与C语言函数指针及函数指针数组 在使用C#时总会为委托而感到疑惑,但现在总新温习了一遍C语言后,才真正理解的委托. 其实委托就类似于C/C++里的函数指针,在函数传参时传递的是函数指针,在调用的时候通过指针访问这个函数. 在C语言中函数指针的申明如下: //可以理解为申明一个指着变量 Func ,它的类型是 返回Type(可以为 void )类型的参数,接收 (Type one,Type two,...)类型的//参数(可以不接受参数). Type *Func(Type one,Type

C语言的指针变量

C语言的指针变量 在C语言中,变量是固定范围的存储空间,它存储的是赋给他的值, 比如: int a = 12; /* 这里是定义一个整型变量a,并把12这个值存储在a的地址空间上 这个地址空间是系统随机分配的,对用户是透明的不用关心 */ 指针变量存储的是变量的地址, 比如: int a = 12 , *b; b = &a; /*如上,定义了一个整型变量a并赋值12,一个整型指针变量b 然后 用 & 取值运算符 取到变量a的地址空间值,存储到指针变量b中 此时变量b中存储的是变量a中的地址

黑 马 程 序 员_视频学习总结&lt;c语言&gt;----03 指针

---------------------- ASP.Net+Unity开发..Net培训.期待与您交流! ---------------------- 一.什么是指针? 用来存放变量地址的变量,就称为"指针变量". 二.指针的定义 一般形式:类名标识符  *指针变量名; int *p; float *q; "*"是一个说明符,用来说明这个变量是个指针变量,是不能省略的,但它不属于变量名的一部分 前面的类型标识符表示指针变量所指向的变量的类型,而且只能指向这种类型的

C语言的指针移位问题

先贴代码 #include <stdio.h> int main(void) { double a[]={1.1,2.2,3.3}; unsigned int b,c,d; b=&a[0]+1;c=&a[1];d=&a[2]; printf("diff -- %d", c-b); return 0; } 执行结果是 diff -- 0 这是32位的代码,所以可以用unsigned int类型来储存地址位. 由于a[0]是double类型,所以&

c语言函数指针实例

如果使用typedef 是这样的 //可以把一个小写字母变成大写 //char (*pFun)(char); typedef char (*PTRFUN)(char); PTRFUN pFun; char glFun(char a){ return a & 223;} void print(int a,PTRFUN call){ cout<<call(a)<<endl;}void main() { pFun = glFun; print('a',pFun);} 如果不使用t

go语言之指针

数组 Arrays 数组是内置(build-in)类型,是一组同类型数据的集合,它是值类型,通过从0开始的下标索引访问元素值.在初始化后长度是固定的,无法修改其长度.当作为方法的入参传入时将复制一份数组而不是引用同一指针.数组的长度也是其类型的一部分,通过内置函数len(array)获取其长度. 在开始前先了解指针与地址: Go 语言中指针是很容易学习的,Go 语言中使用指针可以更简单的执行一些任务,变量是一种使用方便的占位符,用于引用计算机内存地址,Go 语言的取地址符是 &,放到一个变量前使