c基础-指针、函数与预处理器
指针、函数、预处理器
1、指针
指针是一个变量,其值为地址。
声明指针或者不再使用后都要将其置为0 (NULL)
野指针 未初始化的指针
悬空指针 指针最初指向的内存已经被释放了的一种指针
int *a; //正规
int* a;
int * a;
//因为 其他写法看起来有歧义
int* a,b;
使用:
//声明一个整型变量
int i = 10;
//将i的地址使用取地址符给p指针
int *p = &i;
//输出 0xffff 16进制地址
printf("%#x\n", &i);
printf("%#x\n", &p);
指针多少个字节?指向地址,存放的是地址
地址在 32位中指针占用4字节 64为8
//32位:
sizeof(p) == 4;
//64位:
sizeof(p) == 8;
解引用
解析并返回内存地址中保存的值
int i = 10;
int *p = &i;
//解引用
//p指向一个内存地址,使用*解出这个地址的值 即为 10
int pv = *p;
//修改地址的值,则i值也变成100
//为解引用的结果赋值也就是为指针所指的内存赋值
*p = 100;
指针运算
//对指针 进行算数运算
//数组是一块连续内存 分别保存 11-55
//*p1 指向第一个数据 11,移动指针就指向第二个了
int i1[] = {11,22,33,44,55};
int *p1 = i1;
for (size_t i = 0; i < 5; i++)
{
//自增++ 运算符比 解引用* 高,但++在后为先用后加
//如果++在前则输出 22-55+xx
printf("%d\n", *p1++);
//p1[0] == *(p1+1) == s[1]
}
指针指向地址,指针运算实际上就是移动指针指向的地址位置,移动的位数取决于指针类型(int就是32位)
数组和指针
在c语言中,指针和数组名都表示地址
1、数组是一块内存连续的数据。
2、指针是一个指向内存空间的变量
int *p[4]
由于[]比* 优先级高,因此p先于[4]结合,形成p[4]形式,这显然是数组形式,它有4个元素。然后再与p前面的“ * ”结合,“ * ”表示此数组是指针类型的,每个数组元素(相当于一个指针变量)都可以指向一个整形变量。
注意:不要写成“int(*p)[4];”,这是指向一维数组的指针变量。
int i1[] = {11,22,33,44,55};
//直接输出数组名会得到数组首元素的地址
printf("%#x\n",i1);
//解引用
printf("%d\n",*i1);
//将数组名赋值给一个指针,这时候指针指向数组首元素地址
int *p1 = i1;
//数组指针
//二维数组类型是 int (*p)[x]
int array[2][3] = { {11,22,33},{44,55,66} };
//也可以 int array[2][3] = { 11,22,33 ,44,55,66 };
//array1 就是一个 int[3] 类型的指针
int (*array1)[3] = array;
//怎么取 55 ?
//通过下标
array[1][1] == array1[1][1]
//通过解引用
int i = *(*(array1 + 1) + 1);
//拆分
//1、 指针偏移 因为array1的类型是3个int的数组 所以 +1 移动了12位
array1 + 1
//2、获得{44,55,66}数组 (*(array1 + 1))[1] = 55
(*(array1 + 1)
//3、对数组执行 +/- 相当于隐式的转为指针
//获得 55 的地址 再解地址
*(array1 + 1) + 1
//指针数组
int *array2[2];
array2[0] = &i;
array2[1] = &j;
const char *, char const *, char * const,char const * const
const :常量 = final
- 常量指针(指向常量的指针变量)
const 类型名 * 指针变量名
(const放在*号左侧 修饰p指针指向的内存空间不能修改,但可修改指针的指向)
int a = 12;
int b = 15;
//注意int const *p与int const *p效果相同。
const int * p = &a;//定义了p为指向整型变量a的const指针变量
p = &b;//正确
*p = b;//非法
- 指针常量(常指针)
类型名 * const 指针变量名
(const放在*号的右侧, 修饰指针的指向不能修改,但是可修改指针指向的内存空间)
int * const p = &a;
p = &b;//非法
*p = b;//正确
- 指向常量的指针常量
const 基本类型名 * const 指针变量名
const int * const p = &a;
p = &b;//非法
*p = b;//非法
总结:
const修饰规则,优先修饰左边,左边没有修饰右边,例如
const int*p=&a;
:左边没有,修饰右边的int,意味着int值不能变。int* const p=&a;
:左边是,修饰的是指针,意味着指针地址不能变。const int* const p=&a;
:第一个左边没有,修饰右边的int,第二个左边是*,则int不能变,指针也不能变const int const *p=&a;
:第一个左边没有,修饰int,第二个左边int,所以2个const都是修饰的int,值不能变。
总体就是const先看左边再看右边
- const在前:表示const修饰的为所申请的类型。常量指针
- const在后:表示const修饰的为指针。指针常量
- 前后均有:表示const同时修改类型和指针。指向常量的指针常量
(指针)和const(常量)谁在前先读谁:象征着地址,const象征着内容:谁在前面谁就不允许改版。
//从右往左读
//P是一个指针 指向 const char类型
char str[] = "hello";
const char *p = str;
str[0] = 'c'; //正确
p[0] = 'c'; //错误 不能通过指针修改 const char
//可以修改指针指向的数据
//意思就是 p 原来 指向david,
//不能修改david爱去天之道的属性,
//但是可以让p指向lance,lance不去天之道的。
p = "12345";
//性质和 const char * 一样
char const *p1;
//p2是一个const指针 指向char类型数据
char * const p2 = str;
p2[0] = 'd'; //正确
p2 = "12345"; //错误
//p3是一个const的指针变量 意味着不能修改它的指向
//同时指向一个 const char 类型 意味着不能修改它指向的字符
//集合了 const char * 与 char * const
char const* const p3 = str;
多级指针
指向指针的指针
一个指针包含一个变量的地址。当我们定义一个指向指针的指针时,第一个指针包含了第二个指针的地址,第二个指针指向包含实际值的位置。
int a = 10;
int *i = &a;
int **j = &i;
// *j 解出 i
printf("%d\n", **j);
多级指针的意义
见函数部分的引用传值
2、函数
C中的函数与java没有区别。都是一组一起执行一个任务的语句,也都由 函数头与函数体构成
函数的位置
声明在使用之前
函数参数
传值调用
把参数的值复制给函数的形式参数。修改形参不会影响实参
引用调用
形参为指向实参地址的指针,可以通过指针修改实参。
void change1(int i) {
i = 10;
}
void change2(int *i) {
*i = 10;
}
int i = 1;
change1(i);
printf("%d\n",i); //i == 1
change2(&i);
printf("%d\n",i); //i == 10
可变参数
与Java一样,C当中也有可变参数
#include <stdarg.h>
int add(int num, ...)
{
//表示...的参数列表
va_list valist;
int sum = 0;
// 初始化 valist指向第一个可变参数 (...)
va_start(valist, num);
for (size_t i = 0; i < num; i++)
{
//访问所有赋给 valist 的参数
int j = va_arg(valist, int);
printf("%d\n", j);
sum += j;
}
//清理为 valist 内存
va_end(valist);
return sum;
}
函数指针
函数指针是指向函数的指针变量
void println(char *buffer) {
printf("%s\n", buffer);
}
/**
* void(*p)(char*)函数
* void 返回值
* (*p) p变量用来表示这个函数
* (char*) 函数的参数列表
*/
//接受一个函数作为参数
void say(void(*p)(char*), char *buffer) {
p(buffer);
}
void(*p)(char*) = println;
p("hello");
//传递参数
say(println, "hello");
//typedef 创建别名 由编译器执行解释
//typedef unsigned char u_char;
typedef void(*Fun)(char *);
Fun fun = println;
fun("hello");
say(fun, "hello");
//类似java的回调函数
typedef void(*Callback)(int);
void test(Callback callback) {
callback("成功");
callback("失败");
}
void callback(char *msg) {
printf("%s\n", msg);
}
test(callback);
3、预处理器
预处理器不是编译器,但是它是编译过程中一个单独的步骤。
预处理器是一个文本替换工具
所有的预处理器命令都是以井号(#)开头(凡是以“#”开头的均为预处理命令)
常用预处理器
预处理器 | 说明 |
---|---|
#include | 导入头文件 |
#if | if |
#elif | else if |
#else | else |
#endif | 结束 if |
#define | 宏定义 |
#ifdef | 如果定义了宏 |
#ifndef | 如果未定义宏 |
#undef | 取消宏定义 |
宏
预处理器是一个文本替换工具
宏就是文本替换
//宏一般使用大写区分
//宏变量
//在代码中使用 A 就会被替换为1
#define A 1
//宏函数
#defind test(i) i > 10 ? 1: 0
//其他技巧
// # 连接符 连接两个符号组成新符号
#define DN_INT(arg) int dn_ ## arg
DN_INT(i) = 10;
dn_i = 100;
// \ 换行符
#define PRINT_I(arg) if(arg) { \
printf("%d\n",arg); \
}
PRINT_I(dn_i);
//可变宏
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,"NDK", __VA_ARGS__);
//陷阱
#define MULTI(x,y) x*y
//获得 4
printf("%d\n", MULTI(2, 2));
//获得 1+1*2 = 3
printf("%d\n", MULTI(1+1, 2));
宏函数
优点:
文本替换,每个使用到的地方都会替换为宏定义。
不会造成函数调用的开销(开辟栈空间,记录返回地址,将形参压栈,从函数返回还要释放堆栈。)
缺点:
生成的目标文件大,不会执行代码检查
内联函数
和宏函数工作模式相似,但是两个不同的概念,首先是函数,那么就会有类型检查同时也可以debug
在编译时候将内联函数插入。不能包含复杂的控制语句,while、switch,并且内联函数本身不能直接调用自身。
如果内联函数的函数体过大,编译器会自动的把这个内联函数变成普通函数。
#ifndef、#define、#endif
和pragma once
区别
我们经常使用#ifndef、#define、#endif
来防止头文件的内容被重复包含
#pragma once
可以防止整个文件的内容被重复包含
区别
#ifndef、#define、#endif
受C\C++标准的支持,不受编译器的任何限制- 有些编译器不支
持#pragma once
(较老编译器不支持,如GCC3.4版本之前),兼容性不够好。 #ifndef、#define、 #endif
可以针对一个文件中的部分代码,而#prag a once只能针对整个文件。
c基础-指针、函数与预处理器的更多相关文章
- ndk学习之C语言基础复习----指针、函数、预处理器
指针: 指针乃C.C++的灵魂之所在,所以有必要好好的复习复习.什么是指针?一句话来概括:“指针是一个变量,它的值是一个地址.”,其中指针变量的声明有如下三种形式: 其中第一种是被推荐的写法. 其中还 ...
- C基础知识(10):预处理器
C预处理器不是编译器的组成部分,但是它是编译过程中一个单独的步骤.简言之,C预处理器只不过是一个文本替换工具而已,它们会指示编译器在实际编译之前完成所需的预处理. 所有的预处理器命令都是以井号(#)开 ...
- CSS预处理器Sass、LESS 和 Stylus
CSS 预处理器技术已经非常的成熟,而且也涌现出了越来越多的 CSS 的预处理器框架.本文向你介绍使用最为普遍的三款 CSS 预处理器框架,分别是 Sass.Less CSS.Stylus. 首先我们 ...
- 为您详细比较三个 CSS 预处理器(框架):Sass、LESS 和 Stylus
CSS 预处理器技术已经非常的成熟,而且也涌现出了越来越多的 CSS 的预处理器框架.本文向你介绍使用最为普遍的三款 CSS 预处理器框架,分别是 Sass.Less CSS.Stylus. 首先我们 ...
- 详细比较三个 CSS 预处理器(框架):Sass、LESS 和 Stylus
[大伽说]如何运维千台云服务器 » CSS 预处理器技术已经非常的成熟,而且也涌现出了越来越多的 CSS 的预处理器框架.本文向你介绍使用最为普遍的三款 CSS 预处理器框架,分别是 Sass.L ...
- 浅谈 CSS 预处理器: 为什么要使用预处理器?
CSS 自诞生以来,基本语法和核心机制一直没有本质上的变化,它的发展几乎全是表现力层面上的提升.最开始 CSS 在网页中的作用只是辅助性的装饰,轻便易学是最大的需求:然而如今网站的复杂度已经不可同日而 ...
- 三个 CSS 预处理器(框架):Sass、LESS 和 Stylus
CSS 预处理器技术已经非常的成熟,而且也涌现出了越来越多的 CSS 的预处理器框架.本文向你介绍使用最为普遍的三款 CSS 预处理器框架,分别是 Sass.Less CSS.Stylus. 首先我们 ...
- C和指针 (pointers on C)——第十四章:预处理器
第十四章 预处理器 我跳过了先进的指针主题的章节. 太多的技巧,太学科不适合今天的我.但我真的读,读懂.假设谁读了私下能够交流一下.有的小技巧还是非常有意思. 预处理器这一章的内容.大家肯定都用过.什 ...
- <c和指针>学习笔记5动态内存分配和预处理器
1 动态内存 比如声明数组得时候,我们需要提前预估数组长度,分配大了浪费,少了就更不好操作了.从而引入动态分配,需要的时候再分配. (1)malloc和free void *malloc(size_t ...
- C和指针 第十四章 预处理器 头文件
编写一个C程序,第一个步骤称为预处理,预处理在代码编译之前,进行一些文本性质的操作,删除注释.插入被include的文件.定义替换由#define定义的符号,以及确定代码的部分内容是否应该按照条件编译 ...
随机推荐
- .NET Core开发实战(第14课:自定义配置数据源:低成本实现定制化配置方案)--学习笔记
14 | 自定义配置数据源:低成本实现定制化配置方案 这一节讲解如何定义自己的数据源,来扩展配置框架 扩展步骤 1.实现 IConfigurationSource 2.实现 IConfiguratio ...
- .NET Core开发实战(第9课:命令行配置提供程序)--学习笔记
09 | 命令行配置提供程序:最简单快捷的配置注入方法 这一节讲解如何使用命令行参数来作为配置数据源 命令行配置(提供程序的)支持三种格式的命令 1.无前缀的 key=value 模式 2.双中横线模 ...
- Kafka-生产者、broker、消费者的调优参数总结
生产环境下,为了尽可能提升Kafka的整体吞吐量,可以对Kafka的相关配置参数进行调整,以达到提升整体性能的目的. 本文主要从Kafka的不同组件出发,讲解各组件涉及的配置参数和参数含义. 一.生产 ...
- [SpringBoot][Maven]关于maven pom文件的packaging属性
关于maven pom文件的packaging属性 前几天在调试源码运行程序的时候,因为将项目中pom文件的packaging属性用错导致源码包无法引入使用而报Bean注入错误,在此进行总结整理记录. ...
- 【Flink入门修炼】1-3 Flink WordCount 入门实现
本篇文章将带大家运行 Flink 最简单的程序 WordCount.先实践后理论,对其基本输入输出.编程代码有初步了解,后续篇章再对 Flink 的各种概念和架构进行介绍. 下面将从创建项目开始,介绍 ...
- 【JS】一个思路搞定三道Promise并发编程题,手摸手教你实现一个Promise限制器
壹 ❀ 引 之前在整理手写Promise相关资料时,在文章推荐区碰巧看到了一道手写Promise并发控制调度器的笔试题(大厂可能爱考),结果今天同事又正好问了我一个关于Promise调度处理的场景问题 ...
- 西门子SIMATIC LPMLV30 库的模式和状态管理器
从基于S7-1200 / S7-1500的OMAC PackML V3.0获取到的文章内容,用于记录查看 基本信息 根据PackML_V3.0,该库包含了用于机械模式和状态管理器的功能模块. • 机械 ...
- Codeforces Round #847 (Div. 3) A-G
比赛链接 A 题意 判断输入字符串与 \(\pi\) 的最长前缀匹配,不超过 \(30\) 位. 题解 知识点:模拟. 抄样例最后一个 \(30\) 都正确的,直接匹配. 时间复杂度 \(O(1)\) ...
- NVME(学习笔记四)—概念解读
1. 综述 NVMe over PCIe协议,定义了NVMe协议的使用范围.指令集.寄存器配置规范等. 名词解释 1.1.1 Namespace Namespace是一定数量逻辑块(LB)的集合,属性 ...
- 【framework】ATMS启动流程
1 前言 ATMS 即 ActivityTaskManagerService,用于管理 Activity 及其容器(任务.堆栈.显示等).ATMS 在 Android 10 中才出现,由原来的 A ...