《彻底搞定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 ...
随机推荐
- C# 调用c++报错可能性分析
1.在调用之前,可以用工具(Dependency)检测下c++库所依赖的文件,看是否有错误.如果有错误,请先下补充所需运行环境. 2.如果c++ 函数 形参需要C#传入结构体,可如下: [Struct ...
- golang Date format
package main import ( "fmt" "time" ) // @link https://golang.org/pkg/time/ func ...
- php讲中文json数据编码
<?php function show_jsonmsg($data){ if(is_array($data)){ $return = $data; }else{ $return = array( ...
- iwebshop 改版页面
- 第一章——Activity的生命周期
问题总结: 1.Activity完整的生命周期 2.当打开第二个Activity并关闭时候的生命周期. ①.解释为什么onPause()方法不要有耗时操作 3.Activity发生异常重启的时候问题: ...
- 检测delphi的程序的内存泄漏
在主窗体的FormCreate 加入ReportMemoryLeaksOnShutdown := True;
- px,dp,dip,sp,in,mm,pt详细分析
px,dp,dip,sp,in,mm,pt详细分析 px :(pixels),屏幕的像素点,不同的设备显示效果相同,一般我们HVGA代表320x480像素,这个用的比较多. dip :(devi ...
- SQL Server 内存管理在64位时代的改变
64位机上 地址空间比以前大了去了.它引起的改变多了去了 1.MemToLeave这个词不存在了.因为SQL Server以不再做这种预留空间的事了,也就是说multiple page 想用多少就用 ...
- MySQL 指定数据库字符集的 3 种方法。
方法 1.创建数据库时指定字符集. create database Studio character set utf8; 方法 2.创建表的时候针对列指定字符集. create table T( x ...
- 【写一个自己的js库】 4.完善跨浏览器事件操作
1.阻止冒泡. function stopPropagation(event){ event = event || getEvent(event); if(event.stopPropagation) ...