C语言assert断言详解指针(3)

简介: C语言assert断言详解指针(3)

各位少年,大家好,我是博主那一脸阳光,今天分享assert法官的断言,指针宝箱的使用。

前言:如果你在计算机的世界中触犯了语法法规,那么编译器就要上线了,就会出现报错。然而想想我们在现实中设计到经济纠纷什么的,一些自身外部的物资这时候编译器警察就管不了了,没有权限,这时候就需要一些内部法官来管理了,今天我们介绍一个C语言中的法官assert,它是C语言中的库函数。

assert的法官的介绍和宣判

assert叫做“断言”什么是断言“呢?请看下面代码示例。

int a=10;
int b=20;
assert(a==b)//如果不为真,就报错

大家看到了吧,assert非常的霸道,通常assert在指针的时候常用到,接下来介绍一下函数原型和函数参数。

#include <assert.h>
void assert( int expression );

如果你想要使用这个函数的话,你就要使用它对应的头文件assert.h,返回返回值是void,函数的参数是整形的表达式,如果表达式结果为执行),如果为程序终止)。

assert通常用来测试代码测试运行和指针的运算往往比if语句更加简洁方便。

如果最后我们测试成功以后,该怎么屏蔽assert呢?很显然这里就需要一个了,我们在头文件处加上#define NDEBUG这样就会禁用所有assert语句了。一般我们在Debug版本上的使用,但是Release版本中选择禁用assert版本中已经优化掉了(在VS是这样的),这样在Debug版本中大大节省的时间。

指针的使用和传址调用

函数调用分为两种(传址调用)(——————————(传值调用

传值调用,大家可以理解为传递的值。

#include<stdio.h>
int Add(int x, int y)
{
  return x + y;
}
int main()
{
  int a = 10;
  int b = 20;
  int ret = Add(a, b);
  printf("%d\n", ret);
  return 0;
}

当这段代码调用add函数的时候,add函数调用,传值调用,传入函数的时候,传入的是a和b的变量。

传址调用是什么呢?

传址调用:比方说有两个值,传址就是函数在调用的时候,传递的是这两个值的地址。传值调用是就是传入的两个值,就像一个展览的复品一样,一个副本。

通过传址调用模拟实现strlen函数
#include<stdio.h>
size_t my_strlen(const char *s)
{
  int count = 0;
  while (*s != '\0')
  {
    count++;
    s++;
  }
  return count;
}
int main()
{
  char arr[] = "abcdef";
  size_t len = my_strlen(arr);
  printf("%zd\n", len);
    return 0;
}

size_t原型是unsigned int无符号整形,意思是说不能有负号,只能有正数。大家这里理解就可以了。

传址调用和传值调用。

那很显然大家就疑惑了?之前我博客有一篇不需要传址调用模拟实现strlen函数的例子,

那么很显然传值调用传址调用有什么区别,有什么用途?

传址调用实际案例

大家应该还记得我们之前举例子交换两个数字的例子吧?但是我们想要在自定义函数交换?我们也是一样的思路的

但是我们需要用到传址,很显然传值调用不可能实现我们的交换。

#include<stdio.h>
void Swap(int x, int y)
{
  int z = 0;
  z = x;
  x = y;
  y = z;
}
int main()
{
  int a = 0;
  int b = 0;
  scanf("%d %d", &a, &b);
  printf("交换前:a=%d b=%d\n", a, b);
  Swap(a,b);
  printf("交换后:a=%d b=%d\n", a, b);
  return 0;
}
传参调用函数时

函数的实参传给形参时,形参是实参的一份临时拷贝 形参有自己独立的空间 对形参有自己独立的空间 对形参的修改不会影响实参! ```

#include<stdio.h>
 void Swap2(int* pa, int* pb)
{
  int z = 0;
  z = *pa;
  *pa = *pb;
  *pb = z;
}
int main()
{
  int a = 0;
  int 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;
}

我们可以看到实现成Swap2的⽅式,顺利完成了任务,这⾥调⽤Swap2函数的时候是将变量的地址传

递给了函数,这种函数调⽤⽅式叫:传址调⽤。

传址调⽤,可以让函数和主调函数之间建⽴真正的联系,在函数内部可以修改主调函数中的变量;所

以未来函数中只是需要主调函数中的变量值来实现计算,就可以采⽤传值调⽤。如果函数内部要修改

主调函数中的变量的值,就需要传址调⽤。

深入理解指针数组

数组名的理解

首先我们要明白一点数组名是数组首元素的地址,这句话是什么意思呢?

比方说

int arr[10]={0,1,2,3,4,5,6,7,8,9,10];
printf("%p",arr);
printf("%p",&arr[0]);

大家看到了,数组名是第一个元素的地址就是0的地址,我们通过上面代码,解释了数组名是首元素的地址,也就是第一个元素。

当然有例外

sizeof(数组名),sizeof中单独放数组名,这里的数组名表示整个数组,计算的是整个数组的大小,单位是字节。

int arr[10]={0,1,2,3,4,5,6,7,8,9,10);
printf("%d\n",sizeof(arr));
printf("%d\n",sizeof(arr[0]));

&数组名,这里的数组名表示整个数组,取出的是整个数组的地址(整个数组的地址和数组首元素的地址是有区别的)

printf("%d",&arr);

除此之外,任何地方使用数组名,数组名表示整个数组的地址。看下面代码例子。

#include <stdio.h>
int main()
{
 int arr[10] = { 1,2,3,4,5,6,7,8,9,10 };
 printf("%d\n",sizeof(arr));
 printf("&arr[0] = %p\n", &arr[0]);
  printf("&arr[0] = %p\n", &arr[0]+1);
 printf("arr = %p\n", arr);
  printf("arr = %p\n", arr+1);
 printf("&arr = %p\n", &arr);
  printf("&arr = %p\n", &arr+1);
 return 0;
}
&arr[0] = 0077F820
&arr[0]+1 = 0077F824
arr = 0077F820
arr+1 = 0077F824
&arr = 0077F820
&arr+1 = 0077F848

这⾥我们发现&arr[0]和&arr[0]+1相差4个字节,arr和arr+1 相差4个字节,是因为&arr[0] 和 arr 都是

⾸元素的地址,+1就是跳过⼀个元素。 但是&arr 和 &arr+1相差40个字节,这就是因为&arr是数组的地址,+1

操作是跳过整个数组的。 到这⾥⼤家应该搞清楚数组名的意义了吧。 数组名是数组⾸元素的地址

使用指针访问数组

#include<stdio.h>
int main()
{
int arr[10]={ 0 } ;
int i=0;
int sz=sizeof(arr)/sizeof(arr[0]);
int *p=arr;
for(i=0;i<sz;i++)
{
scanf("%d",p+i);
}
for(i=0;i<sz;i++)
{
printf("%d ",*(p+i));
}
return 0;

上面我们通过指针完成对数组输入输出,p每次加i,都访问数组下一个元素的地址。接下来介绍打印数组指针的不同写法i[arr]-->*(arr+i)-->arr[i]

一维数组的传参的本质

#include<stdio.h>
void test(int arr[])//大小可以不写,因为数组传参是传递的是首元素也就是第一个元素。
{
int sz=sizeof(arr)/sizeof(arr[0]);
printf("%d\n",sz);
}
int main()
{
int arr[10]={0};
test(arr);
return 0;
}
数组传参的时候,传递的是并然是数组
传递的是数组首元素的地址
还记得我们之前说过数组的地址是连续的
1,2,3。这样的+1就能找到下一个地址

接下来介绍一个算法叫做冒泡排序

冒泡排序

写一个函数,对一个整形数组的数据进行排序

排序方法很多:

1.冒泡排序

2.选择排序

3.插入排序

4.希尔排序

5.快速排序

冒泡排序(Bubble Sort)

思想:相邻的两个元素比较,如果不满足顺序就交换!![在这里插入图片描述

void BubbleSort(int *arr, int sz)
{
  int i = 0;
  for (i = 0; i < sz - 1; i++)
  {
    int j = 0;
    for (j = 0; j < sz-i-1; j++)
    {
      if (arr[j] > arr[j + 1])
      {
        int tmp = arr[j];
        arr[j] = arr[j + 1];
        arr[j + 1] = tmp;
      }
    }
  }
}
int main()
{
  int arr[] = {9,8,7,6,5,4,3,2,1,0 };
  int sz = sizeof(arr) / sizeof(arr[0]); 
  BubbleSort(arr, sz);
  return 0;
}

上面第一个for循环每一次,就代表排序成功一个数字,第二个for循环个数减去i(i已经排序数字),哪个就是1本身的数字比如说这次要排序九,减一就是减去它本身。

好了,这次我们分享到这里 下篇我会分享如何更加深入理解指针。

相关文章
|
2月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
52 0
|
10天前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
58 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
10天前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
34 9
|
10天前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
32 7
|
20天前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
81 12
|
13天前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
13天前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
43 3
|
14天前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
20天前
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
42 10
|
13天前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
29 1