【NDK系列2】指针、函数、预处理器

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);     0xdaf3bab4
printf("%#x\n", &p);     0xdaf3baa8

指针多少个字节?指向地址,存放的是地址

地址在 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;

printf("%d\n", i);    100
printf("%d\n", *p);   100
printf("%d\n", pv);   10

指针运算

int i1[] = {11,22,33,44,55};
int *p1 = i1;
//*p1 指向第一个数据 11,移动指针就指向第二个了
for (size_t i = 0; i < 5; i++)
{
    printf("%d\n", *p1++); 
}

指针移动.png

数组和指针

在c语言中,指针和数组名都表示地址

1、数组是一块内存连续的数据。

2、指针是一个指向内存空间的变量

int i1[] = {11,22,33,44,55};

//直接输出数组名会得到数组首元素的地址
printf("%#x\n",i1);   0xdaf3baa0

//解引用
printf("%d\n",*i1);   11
    
//将数组名赋值给一个指针,这时候指针指向数组首元素地址
int *p1 = i1;

数组指针

//二维数组类型是 int (*p)[x]

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);

 printf("%d\n", i);      55

指针数组

int *array2[2];
array2[0] = &i;
array2[1] = &j;

const

**const char *** :常量 = final

char str[] = "hello";
const char *p = str;
str[0] = 'c'; //正确
p[0] = 'c';   //错误 不能通过指针修改 const char


// 可以修改p的指向,指向其他的数据
p = "12345";

**char const ***

//性质和 const char * 一样
char const *p1;

char * const

char str[] = "hello";
//p2是一个const指针 指向char类型数据
char * const p2 = str;
p2[0] = 'd';  //正确
p2 = "12345"; //错误

char const const*

//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 addR(int num, ...) {
    va_list valist;
    int sum = 0;
    // 初始化  
    va_start(valist, num);
    for (size_t i = 0; i < num; i++) {
        //访问参数
        int j = va_arg(valist, int);
        printf("%d\n", j);
        sum += j;
    }
    //清理
    va_end(valist);
    return sum;
}

   // 调用
   int sum =  addR(3, 21, 122, 32);
   printf("sum :%d\n", sum);    175

函数指针

函数指针是指向函数的指针变量

void println(char *buffer) {
    printf("%s\n", buffer);
}

//接受一个函数作为参数
void say(void(*p)(char*), char *buffer) {
    p(buffer);
}

void(*p)(char*) = println;
p("hello");

//传递参数
say(println, "hello");

//typedef 创建别名 由编译器执行解释
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);

// 不使用别名则是这么调用
void (*p)(char *) = callback;
test(p);

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 PRINT_I(arg) if(arg) { \
 printf("%d\n",arg); \
 }
PRINT_I(dn_i);

//可变宏
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,"NDK", __VA_ARGS__);



宏函数

​ 优点:

​ 文本替换,每个使用到的地方都会替换为宏定义。

​ 不会造成函数调用的开销(开辟栈空间,记录返回地址,将形参压栈,从函数返回还要释放堆

​ 栈。)

​ 缺点:

​ 生成的目标文件大,不会执行代码检查

内联函数

​ 和宏函数工作模式相似,但是两个不同的概念,首先是函数,那么就会有类型检查同时也可以debug
在编译时候将内联函数插入。

不能包含复杂的控制语句,while、switch,并且内联函数本身不能直接调用自身。
如果内联函数的函数体过大,编译器会自动的把这个内联函数变成普通函数。

代码

https://github.com/ddssingsong/AnyNdk

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容