assert断言
概念
assert介绍
请看下段代码
#include<stdio.h> #include<assert.h> int main() { int a = 10; int* p = &a; //...(此段代码省略处) //...(此段代码省略处) p = NULL; assert(p != NULL); return 0; }
代码解释
#define NDEBUG的使用
如这段代码觉得没问题,
在调试结束后,可以通过在包含 #include 的语句之前插入 #define NDEBUG 来禁用 assert 调用
注意事项
1.assert是宏,需要定义头文件
2.assert只在DEBUG下生效,调试结束后,可以通过在#include<assert.h>语句之前插入#define DEBUG来禁用assert调用
3.assert作用是先计算表达式(expression),然后判断
判断:
表达式为真,继续执行后面程序
表达式为假,vs会报错,具体到哪行
传值调用 VS 传址调用
传值调用
传的是值
举个栗子:
加法函数
看下面代码
int Add(int x, int y) { return x + y; } int main() { int a = 0, b = 0; scanf("%d %d", &a, &b); int ret = Add(a, b); printf("%d\n", ret); }
以上代码用传值调用解释:
在主函数传a,b两个值并调用Add函数;
控制台显示:
传址调用
传的是地址
举个栗子:
strlen函数模拟实现
看下面代码
#include<stdio.h> #include<string.h> #include<assert.h> size_t my_strlen(char* s) { size_t count = 0; assert(s != NULL); while (*s) { count++; s++; } return count; } int main() { char arr[] = "abcdef"; size_t len = my_strlen(arr); printf("%zd\n", len); return 0; }
控制台输出:6
代码解释:
此段代码想用strlen实现计算字符串的类型的个数
1.%zd是打印类型的
2.size_t 对应 unsigned int
3.再随着看上面👆assert断言的 注意事项就能理解这段代码了
用传址调用解释此代码:
区别
传值调用 | 传址调用 | |
传的是 | 值 | 地址 |
在什么情况下使用? | 函数调用某个数来计算 | 修改主调函数变量的值 |
请看下面两段代码:
写一个函数,交换两个整型变量的值
①段代码
#include<stdio.h> int swap1(int x, int y) { int z = 0; z = x; x = y; y = z; } int main() { int a = 0, b = 0; scanf("%d %d", &a, &b); printf("交换前:a=%d,b=%d\n", a, b); swap1(a, b); printf("交换后:a=%d,b=%d\n", a, b); return 0; }
代码解释:
涉及到函数的有关知识:
函数的实参传给形参,实参是形参的一份临时拷贝
形参是一个独立的空间,形参的修改并不会影响实参
②段代码
#include<stdio.h> int swap2(int *pa, int *pb) { int z = 0; z = *pa; *pa = *pb; *pb = z; } int main() { int a = 0, b = 0; scanf("%d %d", &a, &b); printf("交换前:a=%d,b=%d\n", a, b); swap2(&a, &b); printf("交换后:a=%d,b=%d\n", a, b); return 0; }
②段代码注意事项
1.传址调用,传的是地址,在调用函数里要写“ & ”;
2.该代码涉及到解引用操作