《彻底搞定C指针》文档整理
一.变量存储
#include <stdio.h>
int main(void)
{
int i = ;
printf(“%d\n”, i);
printf(“%d\n”, &i); //十进制打印i的地址
printf(“%x\n”, &i); //十六进制打印i的地址
return();
}
/*得到结果:
ray@ray-Lenovo:~/Documents/Exec_important/Daily$ ./0128-2
39
23300652
1638a2c
ray@ray-Lenovo:~/Documents/Exec_important/Daily$ ./0128-2
39
1367552540
5183321c
ray@ray-Lenovo:~/Documents/Exec_important/Daily$ ./0128-2
39
1684339276
6464fa4c
每次得到的地址都不同*/
二. 什么是指针
#include <stdio.h>
int main(void)
{
char a, *pa;
a = ;
pa = &a; /* pa指向变量a的地址 */
*pa = ; /* 给变量a的地址的内容赋值为20, 即a=20*/
printf("%d\n", a);
}
/* 得到结果: 20 */
三. 指针与数组名
#include <stdio.h>
int main(void)
{
int i, *pa, a[] = {, , , , , , , , , };
//int * const pa = a; //不行, pa被定义为常量指针了
pa = a; printf("array mode:\n");
for(i = ; i < ; i++)
printf("%d ", a[i]);
printf("\npointer mode:\n");
for(i = ; i < ; i++)
printf("%d ", *(a+i));
printf("\n\n");
for(i = ; i < ; i++)
printf("%d ", pa[i]);
printf("\n");
for(i = ; i < ; i++)
printf("%d ", *(pa+i));
printf("\n");
for(i = ; i < ; i++)
{
printf("%d ", *pa);
pa++;
}
printf("\n");
return ;
}
/*执行后显示:
array mode:
0 1 2 3 4 5 6 7 8 9
pointer mode:
0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9
0 1 2 3 4 5 6 7 8 9
0 1 2 3 4 5 6 7 8 9
*/
四. const int *pi与int *const pi的区别
#include <stdio.h>
int main(void)
{
int i1 = , i2 = ;
const int *pi = &i1;
pi = &i2; /* 注意这里, pi 可以在任意时候重新赋值一个新内存地址*/
i2 = ; /* 这里能用*pi = 80 来代替吗?当然不能! */
printf("%d\n", *pi); /* 输出是 80 */
return ;
}
/*pi 的值是可以被修改的。即它可以重新指向另一个地址的,但是,不能通过*pi 来修改 i2 的值*/
#include <stdio.h>
int main(void)
{
int i1 = , i2 = ;
int *const pi = &i1;
/* pi = &i2; 注意这里, pi 不能再这样重新赋值了,即不能再指向另一个新地址 */
i1 = ; /* 这里能用 *pi = 80; 来代替吗?可以,这里可以通过*pi 修改 i1 的 */
printf("%d\n", *pi); /* 输出是 80 */
return ;
}
/*pi 的值是可以被修改的。即它可以重新指向另一个地址的,但是,不能通过*pi 来修改 i2 的值*/
总结:
1) 如果 const 修饰在*pi 前,则不能改的是*pi(即不能类似这样: *pi=50;赋值)而不是指 pi。
2) 如果 const 是直接写在 pi 前,则 pi 不能改(即不能类似这样:pi=&i; 赋值)。
五. 函数参数的传递
Q1. 值传递
#include <stdio.h>
void Exchg1(int x, int y)
{
int tmp;
tmp = x;
x = y;
y = tmp;
printf("x = %d, y = %d\n", x, y);
} int main(void)
{
int a = , b = ;
Exchg1(a, b);
printf("a = %d, b = %d\n", a, b);
return ;
}
/*编译执行,输出:
x = 6, y = 4
a = 4, b = 6
*/
Q2. 地址传递
#include <stdio.h>
void Exchg2(int *px, int *py)
{
int tmp = *px;
*px = *py;
*py = tmp;
printf("*px = %d, *py = %d\n", *px, *py);
} int main(void)
{
int a = , b = ;
Exchg2(&a, &b);
printf("a = %d, b = %d.\n", a, b);
return ;
}
/*编译执行,输出:
*px = 6, *py = 4
a = 6, b = 4
*/
Q3. 引用传递
#include <stdio.h>
void Exchg3(int &x, int &y) /* 注意定义处的形式参数的格式与
值传递不同 */
{
int tmp = x;
x = y;
y = tmp;
printf("x = %d, y = %d.\n", x, y);
} main()
{
int a = ;
int b = ;
Exchg3(a, b); /*注意:这里调用方式与值传递一样*/
printf("a = %d, b = %d.\n", a, b);
}
/*编译报错,因为C没有引用的用法,C++才*/
六. 指向另一指针的指针
#include <stdio.h>
void find1(char array[], char search, char **ppa)
{
int i;
for(i = ; *(array + i) != ; i++)
{
if(*(array + i) == search)
{
*ppa = array + i;
break;
}
else if(*(array + i) == )
{
*ppa = ;
break;
}
}
} int main(void)
{
char str[] = {"afsdfsdfdf\0"};
char a = 'd';
char *p = ;
find1(str, a, &p);
if( == p)
printf("没找到!\n");
else
printf("找到了,p = %x\n", p);
return ;
}
/*编译执行,输出:
找到了,p = 6393eb53
*/
七. 函数名与函数指针
1.
#include <stdio.h>
void MyFun(int x); //也可写成void MyFun(int)
void (*FunP)(int); //也可写成void (*FunP)(int x)
int main(int argc, char* argv[])
{
MyFun();
FunP = &MyFun;
(*FunP)();
} void MyFun(int x)
{
printf("%d\n", x);
}
/*编译执行,输出:
10
20
*/
2.
#include <stdio.h>
void MyFun(int x); //也可写成void MyFun(int)
void (*FunP)(int); //也可写成void (*FunP)(int x)
int main(int argc, char* argv[])
{
MyFun();
FunP = MyFun;
FunP();
return ;
} void MyFun(int x)
{
printf("%d\n", x);
}
/*编译执行,输出:
10
20
*/
3.
#include <stdio.h>
void MyFun(int x); //也可写成void MyFun(int)
void (*FunP)(int); //也可写成void (*FunP)(int x)
int main(int argc, char* argv[])
{
MyFun();
FunP = &MyFun;
FunP();
} void MyFun(int x)
{
printf("%d\n", x);
}
/*编译执行,输出:
10
20
*/
4.
#include <stdio.h>
void MyFun(int x); //也可写成void MyFun(int)
void (*FunP)(int); //也可写成void (*FunP)(int x)
int main(int argc, char* argv[])
{
MyFun();
FunP = MyFun;
(*FunP)();
} void MyFun(int x)
{
printf("%d\n", x);
}
/*编译执行,输出
10
20
*/
5.
#include <stdio.h>
void MyFun(int x); //也可写成void MyFun(int)
void (*FunP)(int); //也可写成void (*FunP)(int x)
int main(int argc, char* argv[])
{
(*MyFun)();
FunP = &MyFun;
(*FunP)();
} void MyFun(int x)
{
printf("%d\n", x);
}
/*编译执行,输出:
10
20
*/
总结上面: 1) 其实, MyFun 的函数名与 FunP 函数指针都是一样的, 即都是函数指针。 MyFun 函数名是一个函数指针常量,而 FunP 是一个函数数指针变量,这是它们的关系。
2) 但函数名调用如果都得如(*MyFun)(10)这样,那书写与读起来都是不方便和不习惯的。所以 C 语言的设计者们才会设计成又可允许 MyFun(10)这种形式地调用(这样方便多了并与数学中的函数形式一样,不是吗?)。
3) 为统一起见, FunP 函数指针变量也可以 FunP(10)的形式来调用。
4) 赋值时,即可 FunP = &MyFun 形式,也可 FunP = MyFun。
6.
#include <stdio.h>
void MyFun(int x); //也可写成void MyFun(int)
typedef void (*FunType)(int);
FunType FunP;
int main(int argc, char* argv[])
{
//FuncType FunP;/*函数指针变量也可以是局部的*/
MyFun();
FunP = &MyFun;
FunP();
return ;
} void MyFun(int x)
{
printf("%d\n", x);
}
/*编译执行,输出:
10
20
*/
7.
#include <stdio.h>
void MyFun1(int x);
void MyFun2(int x);
void MyFun3(int x);
typedef void (*FunType)(int);
void CallMyFun(FunType fp, int x);
int main(int argc, char* argv[])
{
CallMyFun(MyFun1, );
CallMyFun(MyFun2, );
CallMyFun(MyFun3, );
} void CallMyFun(FunType fp, int x)
{
fp(x);
} void MyFun1(int x)
{
printf("函数MyFun1中输出:%d\n", x);
} void MyFun2(int x)
{
printf("函数MyFun2中输出:%d\n", x);
} void MyFun3(int x)
{
printf("函数MyFun3中输出:%d\n", x);
}
/*
函数MyFun1中输出:10
函数MyFun2中输出:20
函数MyFun3中输出:30
*/
《彻底搞定C指针》文档整理的更多相关文章
- 五分钟轻松搞定产品需求文档!这可能史上最全PRD文档模板
本文由 @JustWu 原创发布于社区 为什么写这篇文章? 第一:写PMCAFF的PRD文档,大家都是用户,比较好参考与理解,方便大家来找我写的不好的地方. 第二:我在自学PRD文档的编写过程中,总 ...
- 将Html文档整理为规范XML文档
有多种方式可以在.NET 平台进行HTML文件解析.数据提取,其中最简单.稳妥的办法是先使用工具将Html文档整理成XML文档,再通过XML Dom模型或XPath灵活地进行数据处理.SGML便是一个 ...
- NodeJS-001-Nodejs学习文档整理(转-出自http://www.cnblogs.com/xucheng)
Nodejs学习文档整理 http://www.cnblogs.com/xucheng/p/3988835.html 1.nodejs是什么: nodejs是一个是javascript能在后台运行的平 ...
- Ionic2文档整理
来自:Rainey's Blog 原文地址:http://rainey.space/2016/04/06/Ionic2_Chinese_Document/ Github:https://github. ...
- dotNET跨平台相关文档整理
一直在从事C#开发的相关技术工作,从C# 1.0一路用到现在的C# 6.0, 通常情况下被局限于Windows平台,Mono项目把我们C#程序带到了Windows之外的平台,在工作之余花了很多时间在M ...
- 彻底搞定c指针
第一篇 变量的内存实质 一.先来理解C语言中变量的实质 要理解C指针,我认为一定要理解C中“变量”的存储实质,所以我就从“变量”这个东西开始讲起吧! 先来理解理解内存空间吧!请看下图: 内存地址→ ...
- appium python andiroid自动化文档整理笔记。
利用一天时间去整理appium for android文档.传送门 利用业余时间自己翻阅资料,google.百度等去查找,费劲一番功夫,最后终于成行了这篇文档. 也是作者对最近自己的学习的一个总结吧, ...
- redis module 学习—官网文档整理
前言 redis在4.0版本中,推出了一个非常吸引的特性,可以通过编写插件的模式,来动态扩展redis的能力.在4.0之前,如果用户想拥有一个带TTL的INCRBY 命令,那么用户只能自己去改代码,重 ...
- Linux文档整理之【Jenkins+Docker自动化部署.Net Core】
这次整理的文档是Jenkins+Docker实现自动化部署,很早之前就写的,今天有时间就搬到博客园做个记录. Jenkins是基于Java开发的一种持续集成工具,主要用于持续.自动的构建/测试软件等相 ...
- Linux文档整理之【Nginx安装Docker】
这次整理的文档是Docker安装 先说明下我用的系统是Linux Centos,不同的Linux版本差别不大. 1.安装依赖包 sudo yum install -y yum-utils device ...
随机推荐
- javascript中with语句应用
语法格式: with(obj){} obj指明了语句组中对象缺省时的参考对象,就是代表该语句块中的默认作用域为obj. <html> <head> <title>& ...
- 纯CSS3技术 加载中
你能相信吗?这些都是由一个DIV元素实现的动画,纯CSS3技术 html <div class="loader">加载中...</div> css: 图( ...
- 运算符 - PHP手册笔记
运算符优先级 每种编程语言都有运算符,运算符要学会灵活使用. 运算符拥有不同的优先级和结合方向. <?php var_dump(1 <= 1 == 1); // true var_dump ...
- 常量 - PHP手册笔记
常量语法 常量在脚本执行期间其值不能改变.常量大小写敏感,传统上常量标识符总是大写.常量一旦定义就不能被重新定义或取消定义,常量的值只能是标量. 可以用define()函数来定义常量,也可以使用con ...
- Oracle EBS-SQL (PO-18):检查工作台下达的PR在系统找不到.sql
select * From apps.po_requisitions_interface_all---------------------------------------------------- ...
- Oracle EBS-SQL (PO-13):检查采购物料无一揽子协议价格.sql
Select msi.segment1 物料编码, msi.DESCRIPTION ...
- Centos7网络配置+图形界面设置
一. 查看网络地址: centos7取消了ifconfig命令,使用ip addr命令查看IP地址 二.配置网络 用VirtualBox安装的CentOS7,安装完成后,发现无法上网,于是到网上查了一 ...
- 折腾Python中的Tkinter
折腾Python中的Tkinter 从oschina看到了关于Python的Tkinter简介: Tk图形用户界面 Tkinter 又从Python官网文档: Tkinter — Python int ...
- Baby Ming and Matrix games(dfs计算表达式)
Baby Ming and Matrix games Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Ja ...
- 关于 HTML5 的 11 个让人难以接受的事实
HTML5为Web开发者提供了很多强大的新特性,但是它的一些特定的限制会让它无法和本地应用匹敌. HTML5整合进了很多新的特性,并且有可能提升Web编程模式.和每一个阅读技术资讯的人所知道的一样,没 ...