刨析《C语言》【进阶】付费知识【二】

简介: 刨析《C语言》【进阶】付费知识【二】

@[toc]

计算长度

sizeof:

计算变量,数组,类型的大小,单位是字节(操作符)

#include<stdio.h>

intmain()

{

   //sizeof(数组名)-数组名表示整个数组的-计算的是整个数组的大小

   //&数组名 - 数组名表示的是整个数组,取出的是整个数组的地址

   //除此之外,所有的数组名都是数组首元素的地址

   //整形数组

   inta[]={1,2,3,4};

   printf("%d\n",sizeof(a));//16

   printf("%d\n",sizeof(a+0));//4/8 a+0是第一个元素的地址,sizeof(a+0)计算的是地址的大小

   printf("%d\n",sizeof(*a));//4 *a是数组的第一个元素,sizoef(*a)计算的是第一个元素的大小

   printf("%d\n",sizeof(a+1));//4/8  a+1是第二个元素的地址,sizeof(a+1)计算的地址的大小

   printf("%d\n",sizeof(a[1]));//4 计算的是第二个元素的大小

   printf("%d\n",sizeof(&a));// 4/8 -@a虽然数组的地址,但也是地址,sizeof(&a)计算的是一个地址的大小

   printf("%d\n",sizeof(*&a));//16 -计算的数组的大小

   //&a -- int(*p)[4]=&a;

   printf("%d\n",sizeof(&a+1));//4/8 - &a+1--数组后面的空间的地址

   printf("%d\n",sizeof(&a[0]));//4/8

   printf("%d\n",sizeof(&a[0]+1));//4/8

   //字符数组

   chararr[]={'a','b','c','d','e','f'};

   printf("%d\n",sizeof(arr));//6

   printf("%d\n",sizeof(arr+0));//4/8 -指针大小 -指针所指地址是4个字节地址

   printf("%d\n",sizeof(*arr));//1

   printf("%d\n",sizeof(arr[1]));//1

   printf("%d\n",sizeof(&arr));//4/8

   printf("%d\n",sizeof(&arr+1));//4/8

   printf("%d\n",sizeof(&arr[0]+1));//4/8

   return0;

}

#include<stdio.h>

intmain()

{

   //sizeof(数组名)-数组名表示整个数组的-计算的是整个数组的大小

   //&数组名 - 数组名表示的是整个数组,取出的是整个数组的地址

   //除此之外,所有的数组名都是数组首元素的地址

   //整形数组

   inta[]={1,2,3,4};

   printf("%d\n",sizeof(a));//16

   printf("%d\n",sizeof(a+0));//4/8 a+0是第一个元素的地址,sizeof(a+0)计算的是地址的大小

   printf("%d\n",sizeof(*a));//4 *a是数组的第一个元素,sizoef(*a)计算的是第一个元素的大小

   printf("%d\n",sizeof(a+1));//4/8  a+1是第二个元素的地址,sizeof(a+1)计算的地址的大小

   printf("%d\n",sizeof(a[1]));//4 计算的是第二个元素的大小

   printf("%d\n",sizeof(&a));// 4/8 -@a虽然数组的地址,但也是地址,sizeof(&a)计算的是一个地址的大小

   printf("%d\n",sizeof(*&a));//16 -计算的数组的大小

   //&a -- int(*p)[4]=&a;

   printf("%d\n",sizeof(&a+1));//4/8 - &a+1--数组后面的空间的地址

   printf("%d\n",sizeof(&a[0]));//4/8

   printf("%d\n",sizeof(&a[0]+1));//4/8

   //字符数组

   chararr[]={'a','b','c','d','e','f'};

   printf("%d\n",sizeof(arr));//6

   printf("%d\n",sizeof(arr+0));//4/8 -指针大小 -指针所指地址是4个字节地址

   printf("%d\n",sizeof(*arr));//1

   printf("%d\n",sizeof(arr[1]));//1

   printf("%d\n",sizeof(&arr));//4/8

   printf("%d\n",sizeof(&arr+1));//4/8

   printf("%d\n",sizeof(&arr[0]+1));//4/8

   return0;

}

intmain()

{

   inta[3][4] = { 0 };

   printf("%d\n", sizeof(a));//48 = 3*4*sizeof(int)

   printf("%d\n", sizeof(a[0][0]));//4 - a[0][0] - 是第一行第一个元素

   printf("%d\n", sizeof(a[0]));//16

   printf("%d\n", sizeof(a[0] +1));//4 解释:a[0]作为数组名并没有单独放在sizeof内部,

                                   //也没取地址,所以a[0]就是第一行第一个算的地址

                                   //a[0]+1,就是第一行第二个元素的地址

   printf("%d\n", sizeof(*(a[0] +1)));//4 - 解释:*(a[0] + 1)是第一行第二个元素

   printf("%d\n", sizeof(a+1));//4 - 解释:a是二维数组的数组名,并没有取地址

   //也没有单独放在sizeof内部,所以a就表示二维数组首元素的地址,即:第一行的地址

   //a + 1就是二维数组第二行的地址

   printf("%d\n", sizeof(*(a+1)));//16 解释:a+1是第二行的地址,所以*(a+1)表示第二行

   //所以计算的就是第2行的大小

   printf("%d\n", sizeof(&a[0] +1));//4 解释:a[0]是第一行的数组名,

   //&a[0]取出的就是第一行的地址,&a[0]+1 就是第二行的地址

   printf("%d\n", sizeof(*(&a[0] +1)));//&a[0]+1 就是第二行的地址

   //*(&a[0]+1) 就是第二行,所以计算的第二行的地址

   printf("%d\n", sizeof(*a));//16 解释:a作为二维数组的数组名,没有&,没有单独放在sizeof内部

   //a就是首元素的地址,即第一行的地址,所以*a就是第一行,计算的是第一行的大小

   printf("%d\n", sizeof(a[3]));//16 解释:a[3]其实是第四行的数组名(如果有的话)

   //所以其实不存在,也能通过类型计算大小的

   printf("%d\n", sizeof(a[-1]));

   return0;

}

strlen

strlen:是求字符串长度的,只能对字符串长度(库函数-使用得引用头文件)

#include<stdio.h>

#include<string.h>

intmain()

{

   chararr[]={'a','b','c','d','e','f'};

   printf("%d\n",strlen(arr));//随机值  -遇到‘\0’结束

   printf("%d\n",strlen(arr+0));//随机值

   //printf("%d\n",strlen(*arr));//err

   //printf("%d\n",strlen(arr[1]));//err

   printf("%d\n",strlen(&arr));//随机值

   printf("%d\n",strlen(&arr+1));//随机值- 6

   printf("%d\n",strlen(&arr[0]+1));//随机值- 1

   return0;

}

因为strlen只对字符串求长度,对字符会产生随机值

指针

指针变量的大小

32位计算机系统 整形指针占4个字节,实参传字符形参也是4个字节

void test1 (char ch)//char *ch{

printf("%d\n",sizeof(ch));//4个字节,因为传入的是字符的首地址,也就是指针char *ch ,指针长度为4,所以char字符类型的传参是传的指针字节

}char arr[10]={0};printf("%d\n",sizeof(char));//10test1(ch);//字符数组首元素

  • 只要在32位操作环境下,不管是什么类型,都是4个字节

  • 在64位环境下

声明指针

int* a,b,c;事实上只声明了变量a是指针类型如果要声明三个指针:int a ,b, *c;

结构体

  1. . :结构体变量.成员
  2.    -> :结构体指针->成员

#include<stdio.h>

#include<string.h>

structBook

{

   charbook_name[20];

   intprice;

};

intmain()

{

   structBookb={"c语言程序设计",55};

   structBook*p=&b;

   //更改价格

   (*p).price=19;//等同于p->price

   printf("%d\n",b.price);

   //更改书名

   //使用库函数字符串拷贝函数

   //b1.name="c++";//error

   strcpy(p->book_name,"C++");//因为book_name是字符型的数组名,数组本身是个地址,而price是变量

   printf("%s\n",(*p).book_name);

   printf("%s\t %d\n",p->book_name,p->price);

   printf("%s\t%d\n",(*p).book_name,(*p).price);//(*p).book_name,(*p).price等同于p->book_name,p->price

   printf("%s\n",b.book_name);

   printf("%d\n",b.price);

   return0;

}

数组元素地址

1.sizeof(数组名),计算整个数组的大小,sizeof内部单独放一个数组名,数组名表示整个数组2.&数组名,取出的数组的地址。&数组名,数组名表示整个数组。

除此1,2两种情况之外,所以的数组名都表示数首元素的地址

字符串

字符串的比较

stract(str1,str1); //err,因为自己追加自己会把'\0'覆盖掉,导致没有一直都没有'\0'反复循环

不能用两个字符串比较两个字符串相等,应该使用字符串例:

charpassword[20]={0};

sacnf("%s",password);

//if(pwssword == "123456")//err

if(strcmp(password,"123456")==0)

printf("相同")

字符串的拷贝

把字符串拷贝到目标地址,调试我们发现,遇到'\0'结束拷贝

//更改书名

   //使用库函数字符串拷贝函数

   //b1.name="c++";//error

   strcpy(p->book_name,"C++");//因为book_name是字符型的数组名,数组本身是个地址

当拷贝的不是'\0'结束,程序运行出错

  • 源字符串必须以 '\0' 结束。
  • 会将源字符串中的 '\0' 拷贝到目标空间。
  • 目标空间必须足够大,以确保能存放源字符串。
  • 目标空间必须可变。
  • 学会模拟实现。注意:源字符必须是字符数组或者是一个指向动态分配内存的数组的指针,不能使用字符串常量!

结构体

内存对齐

总体来说:

结构体的内存对齐是拿空间来换取时间的做法。起。

S1和S2类型的成员一模一样,但是S1和S2所占空间的大小有了一些区别。

//例如:

struct S1

{

char c1;

int i;

char c2;

};

struct S2

{

char c1;

char c2;

int i;

};

修改默认对齐数

之前我们见过了 #pragma 这个预处理指令,这里我们再次使用,可以改变我们的默认对齐数。

//例如:

struct S1

{

char c1;

int i;

char c2;

};

struct S2

{

char c1;

char c2;

int i;

};

#include <stdio.h>

#pragma pack(8)//设置默认对齐数为8

struct S1

{

char c1;

int i;

char c2;

};


#pragma pack()//取消设置的默认对齐数,还原为默认

#pragma pack(1)//设置默认对齐数为1

struct S2

{

char c1;

int i;

char c2;

};

#pragma pack()//取消设置的默认对齐数,还原为默认

int main()

{

   //输出的结果是什么?

   printf("%d\n", sizeof(struct S1));

   printf("%d\n", sizeof(struct S2));


目录
相关文章
|
存储 算法 IDE
刨析《C语言》【进阶】付费知识【一】
刨析《C语言》【进阶】付费知识【一】
104 0
刨析《C语言》【进阶】付费知识【一】
|
存储 程序员 开发工具
刨析《C语言》【进阶】付费知识【完结】
刨析《C语言》【进阶】付费知识【完结】
147 0
刨析《C语言》【进阶】付费知识【完结】
|
2月前
|
C语言 C++
C语言 之 内存函数
C语言 之 内存函数
38 3
|
15天前
|
存储 C语言 开发者
【C语言】字符串操作函数详解
这些字符串操作函数在C语言中提供了强大的功能,帮助开发者有效地处理字符串数据。通过对每个函数的详细讲解、示例代码和表格说明,可以更好地理解如何使用这些函数进行各种字符串操作。如果在实际编程中遇到特定的字符串处理需求,可以参考这些函数和示例,灵活运用。
37 10
|
15天前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
37 9
|
15天前
|
存储 Unix Serverless
【C语言】常用函数汇总表
本文总结了C语言中常用的函数,涵盖输入/输出、字符串操作、内存管理、数学运算、时间处理、文件操作及布尔类型等多个方面。每类函数均以表格形式列出其功能和使用示例,便于快速查阅和学习。通过综合示例代码,展示了这些函数的实际应用,帮助读者更好地理解和掌握C语言的基本功能和标准库函数的使用方法。感谢阅读,希望对你有所帮助!
30 8
|
15天前
|
C语言 开发者
【C语言】数学函数详解
在C语言中,数学函数是由标准库 `math.h` 提供的。使用这些函数时,需要包含 `#include <math.h>` 头文件。以下是一些常用的数学函数的详细讲解,包括函数原型、参数说明、返回值说明以及示例代码和表格汇总。
39 6
|
15天前
|
存储 C语言
【C语言】输入/输出函数详解
在C语言中,输入/输出操作是通过标准库函数来实现的。这些函数分为两类:标准输入输出函数和文件输入输出函数。
92 6
|
15天前
|
存储 缓存 算法
【C语言】内存管理函数详细讲解
在C语言编程中,内存管理是至关重要的。动态内存分配函数允许程序在运行时请求和释放内存,这对于处理不确定大小的数据结构至关重要。以下是C语言内存管理函数的详细讲解,包括每个函数的功能、标准格式、示例代码、代码解释及其输出。
47 6
|
15天前
|
C语言 开发者
【C语言】断言函数 -《深入解析C语言调试利器 !》
断言(assert)是一种调试工具,用于在程序运行时检查某些条件是否成立。如果条件不成立,断言会触发错误,并通常会终止程序的执行。断言有助于在开发和测试阶段捕捉逻辑错误。
24 5