刨析《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));


目录
相关文章
|
存储 程序员 开发工具
刨析《C语言》【进阶】付费知识【完结】
刨析《C语言》【进阶】付费知识【完结】
108 0
刨析《C语言》【进阶】付费知识【完结】
|
存储 算法 IDE
刨析《C语言》【进阶】付费知识【一】
刨析《C语言》【进阶】付费知识【一】
76 0
刨析《C语言》【进阶】付费知识【一】
|
8天前
|
C语言
C语言:内存函数(memcpy memmove memset memcmp使用)
C语言:内存函数(memcpy memmove memset memcmp使用)
|
2天前
|
缓存 安全 编译器
【C 言专栏】C 语言函数的高效编程技巧
【5月更文挑战第1天】本文探讨了C语言中函数的高效编程技巧,包括函数的定义与作用(如代码复用和提高可读性)、设计原则(单一职责和接口简洁)、参数传递方式(值传递、指针传递和引用传递)、返回值管理、调用约定、嵌套与递归调用,以及函数优化技巧和常见错误避免。掌握这些技巧能提升C语言代码的质量和效率。
【C 言专栏】C 语言函数的高效编程技巧
|
4天前
|
C语言
pta浙大版《C语言程序设计(第3版)》 习题6-4 使用函数输出指定范围内的Fibonacci数 (20分)
pta浙大版《C语言程序设计(第3版)》 习题6-4 使用函数输出指定范围内的Fibonacci数 (20分)
|
4天前
|
C语言
pta 浙大版《C语言程序设计(第3版)》题目集 习题6-6 使用函数输出一个整数的逆序数 (20分)
pta 浙大版《C语言程序设计(第3版)》题目集 习题6-6 使用函数输出一个整数的逆序数 (20分)
|
5天前
|
C语言
(浙大版《C语言程序设计(第3版)》 习题6-5 使用函数验证哥德巴赫猜想 (20分)
(浙大版《C语言程序设计(第3版)》 习题6-5 使用函数验证哥德巴赫猜想 (20分)
|
7天前
|
安全 C语言
【C语言】strcpy与strncpy函数的使用和模拟实现
【C语言】strcpy与strncpy函数的使用和模拟实现
5 0
|
7天前
|
C语言
【C语言】字符分类函数与字符转换函数
【C语言】字符分类函数与字符转换函数
9 1
|
7天前
|
程序员 编译器 C语言
C语言之函数与参数
C语言之函数与参数
5 0