> 文章列表 > C-关键字(下)

C-关键字(下)

C-关键字(下)

文章目录

      • 循环控制
        • switch-case-break-default
        • do-while-for
          • getchar()
        • break-continue
        • goto
        • void
          • void*
      • return
      • const
          • const修饰变量
          • const修饰数组
          • const修饰指针
            • 指针补充
          • const 修饰返回值
      • volatile
      • struct
        • 柔型数组
      • union联合体
        • 联合体空间开辟问题
        • 利用联合体的性质,判断机器是大端还是小端
      • enum枚举
      • typedef
          • typedef vs #define
          • `typedef static int int32_t` 行不行

循环控制

switch-case-break-default

C-关键字(下)

int main()
{int day = 0;scanf("%d",&day);switch (day)//整形(int char short)||整形表达式{case 1:		//case是用来进行判定的printf("星期一\\n");break;	//充当分支的作用,否则会将后面的都打印default:printf("不是星期一\\n");break;}return 0;
}
  • 当一个case分支时要执行多条语句或者定义变量时最好加上代码块{}或者将众多代码封装为一个函数就行.
  • 当多条分支执行一条语句时,将break删除组合落为一个就行.
int main()
{int day = 0;scanf("%d",&day);switch (day)//整形(int char short)||整形表达式{case 1:	case 2:case 3://case是用来进行判定的printf("要上课\\n");break;	//充当分支的作用,否则会将后面的都打印default:printf("输入内容有误\\n");break;}return 0;
}
  • default分支可以放在任何位置,只是为了好看并且符合语义放在最后.

  • break不要写成return,可以写但是不方便别人维护的时候好看,可以在循环的时候设置一个bool类型的标记位.

  • switch-case中绝对不能用const修饰组成的只读常量,必须是真正的常量.

int main()
{const int day = 0;scanf("%d",&day);switch (day)//整形(int char short)||整形表达式{case day:break;//编译报错case 1:	case 2:case 3://case是用来进行判定的printf("要上课\\n");break;	//充当分支的作用,否则会将后面的都打印default:printf("输入内容有误\\n");break;}return 0;
}

do-while-for

循环条件初始化,循环判定,循环条件更新.

任何C程序,在默认编译好之后,运行时都会默认打开三个输入输出流.

C-关键字(下)

perror("hello world\\n");

getchar()
  • 键盘中输入时,多读了一个\\n

C-关键字(下)

  • 为什么返回值是int类型,不是char类型

字符char类型是8字节,[0,255],ASCII码表中都是合理的值,如果也设置是返回值是char.返回成功时是一个有效字符,如果获取失败,8个bit位只能表示所有的合法字符,但是无法表示返回失败的概念.说白了,char类型的返回值无法表示失败,太小了.

  • 键盘中输入的所有的内容或者输出的字符,或者往显示器中打印的,都是字符!

C-关键字(下)

printf的返回值就是表示一共打印了多少个字符.

int a=0; scanf("%d",&a);说白了就是将键盘中的字符按照类型格式化输入到变量a中.所以会将键盘显示器啥的都叫字符设备.

break-continue

  • break对比continue

C-关键字(下)

  • continue:在while()和do while()都是跳转到条件判断处,for()循环是跳转到条件更新处.

C-关键字(下)

goto

直接跳转到标签指定处,可以向上或者向下跳转.

  • goto 语句实现循环
int main()
{int i = 0;
start:printf("[%d] goto running ...\\n", i);i++;if (i < 10){goto start;}printf("goto end...\\n");return 0;
}
  • goto语句只能在本函数块中使用,不能跨函数或者跨文件使用.

void

  • void不能定义变量.

    • void本身就被编译器定义为空类型,强制的不允许定义变量.

    • void大小是0不能开辟空间导致的,本身是空类型,理论上是不应该开空间的.即使开空间也仅仅是一个占位符.

    • 注意:不同的编译器对于void大小的规定也是不一样的.

C-关键字(下)

  • C语言函数可以不带返回值.默认的返回值就是int.避免别人误解,所以使用void告诉别人,我不想返回.只是起到一个占位符的作用,这个返回值无法接收.

C-关键字(下)

  • 告知编译器,这个函数不需要传参.

C-关键字(下)

void*
  • void* 可以定义变量只要是指针大小类型就是确定的.

  • void*可以被任何类型的指针接收.void*可以接收任意指针类型.

    • 库,系统接口的设计上,尽量设计为通用接口.例子中,既可以是int类型也可以是double类型.
    • C-关键字(下)
  • vs中void* 指针变量不能+±-,无法确定向前移动几个字节.而在Linux中是可以编译通过的,因为Linux中有确定的sizof(void)1字节大小.因为不同的平台看待void空间大小是不确定的.

  • 不能对void* 类型指针进行解引用.

return

  • 字符串类型 vs 字符串

C语言没有字符串类型,存在字符串,以\\0结尾,不是数据长度,但是占据一字节的容量大小,字符串本身是没有名字的.使用字符数组访问.

  • 计算机中删除数据释放空间是否是将数据全部清空?

    计算机中删除数据本质只需要设置该数据无效即可(文件系统部分理解).所以下载数据很慢但是删除很快.

C-关键字(下)

调用函数,形成栈帧,函数返回,栈帧空间释放.函数栈帧中的数据并不会被释放,仅仅代表这个空间是可以被覆盖的.printf()也是函数,所以使用printf的时候会形成新的函数栈帧,覆盖之前show()函数栈帧的部分.

  • 开辟栈帧时,如何确定开辟多大的空间呢?

在调用函数之前,编译器可以通过定义变量的数量和类型,是可以预估这个函数应该是多少的空间的.不同的编译器预估的方式也是不同的.

  • 递归的情况会不断创建栈帧空间,会存在栈溢出的情况.

  • 为什么临时变量具有临时性?

    函数中的变量基本都是临时变量在一个函数栈帧中,函数返回栈帧空间被释放.所有的临时变量都是依托于函数栈帧开辟空间的.

int GetData()
{int x = 0x11223344;printf("get data success!\\n");return x;
}
int main()
{int ret = GetData();printf("ret: %x\\n",ret);return 0;
}
  • GetData()函数中的变量x在函数返回时已经被销毁,是如何交给ret的呢?

函数的返回值,通过寄存器eax的方式返回给函数调用方.

  • 有ret进行接收,就将eax中内容交给ret

C-关键字(下)

  • 没有ret接收,仍然放到eax中,但是函数并不会做任何处理.
  • main函数的返回值是给谁的呢?为什么经常是0?(进程部分讲解)

const

const修饰变量

为变量添加只读属性,const修饰的变量是不可直接被修改(二次赋值).通过指针就可以间接修改.

  • const修饰变量的意义?是在编译期间的处理

让编译器进行直接修改式检查.告诉其他人不可修改.(是一种软性要求,不是强制性约束)

  • 真正意义上的不可被修改,操作系统层面的处理.

C-关键字(下)

  • 数组开辟的空间大小必须是真常量.const修饰的变量vs中是编不过的.在Linux中是可以编过的,不同的平台支持的c的版本是不同的.
  • const只能在初始化的时候赋值,二次赋值时不允许的.
const修饰数组

数组中的元素都是不可被修改的.const int arr[] = {1,2,3,4,5};

const修饰指针

地址就是指针,提高CPU内存寻址的效率.

  • 地址是数据吗?是

  • 数据可以被保存吗?4字节空间存储,这个内存空间就叫做指针变量.

  • 任何一个变量名在不同的应用场景中代表不同的含义.

    int a=10;
    int* p=&a;
    p=&b;//p的空间,变量的属性,左值
    q=p; //p中的内容,数据的属性,右值
    
指针补充
  • &

整形变量为例,对应他的指针变量中存储的是变量4字节中的最低的那个地址空间.

  • *解引用

类型相同的时候,对指针解引用,指针所指向的目标.

int main()
{int  a = 10;const int *p1 = &a;//p指向的变量不可以直接修改p1 = 200;*p1 = 100;//errorint const *p2 = &a;//p指向的变量不可以直接修改//const修饰的是指针变量p中地址内容不可直接被改变int* const p3 = &a;*p3 = 1200;int b = 20;p3 = &b;//error//const修饰的是指针变量p中地址内容不可直接被改变const int* const p4 = &a;*p4 = 200;//errorp4 = &b;//errorreturn 0;
}
  • 函数传参需要产生临时变量吗?

C中,任何函数参数都一定要产生临时变量,包括指针变量.

C-关键字(下)

const 修饰返回值

如果不想用函数返回值修改函数中变量的值时,可以选择使用const修饰返回值.

const int* GetVal()
{static int a = 10;return &a;
}
int main()
{const int* p = GetVal();//外部接收使用const修饰*p = 100;//error
}

volatile

  • 内存被覆盖的情况

不让CPU进行优化,每次都去访问内存,而不是优化被放进寄存器中数据的值,一直访问寄存器中的数据.

#include <stdio.h>
int pass = 1;
//volatile int pass = 1;
int main()
{while(pass){}return 0;
}
  • 不加volatile

C-关键字(下)

  • 加上volatile

C-关键字(下)

const volatile int a = 10;
在vs2013和gcc 4.8中都能编译通过
const是在编译期间起效果
volatile在编译期间主要影响编译器,形成不优化的代码,进而影响运行,故:编译和运行都起效果。
const要求你不要进行写入就可以。volatile意思是你读取的时候,每次都要从内存读。
两者并不冲突。
虽然volatile就叫做易变关键字,但这里仅仅是描述它修饰的变量可能会变化,要编译器注意,并不是它要求对应变量必须变化!这点要特别注意

struct

#define NUM 64
struct stu
{char name[NUM / 2];int age;char sex;char addr[NUM];
}a,c,b;//struct类型就是类型,和int啥的定义变量是相同的
int main()
{struct stu x;strcpy(x.name,"张三");//x.name = "张三";//error 支持整体初始化,不支持整体赋值x.age = 18;struct stu* p = &x;printf("sut.name:%s\\n",p->name);printf("sut.name:%s\\n",(*p).name);//为什么结构体访问会存在两种形式?//结构体可能贯穿多个函数,这时候传指针就很节省空间并且效率高//在日常访问的时候使用.更方便
}
  • vs中结构体必须有一个成员,不支持空结构体.Linux中gcc可以定义,大小是0

柔型数组

  • 必须放在结构体内.

  • 首元素最好不是柔性数组,建议使用时前面最好还有一个有效的成员.

  • 柔性数组是不占用结构体空间的.所以想要动态开辟结构体大小的空间时,需要根据设定大小和类型自定义开辟.

struct data
{ int num;int arr[];
};
int main()
{struct data d;struct data* p = malloc(sizeof(struct data)+sizeof(int)*10);p->num = 10;for (int i = 0; i < p->num; i++){p->arr[i] = i;}free(p);
}

union联合体

union un
{int a;char b;
};
int main()
{union un u;u.a = 10;union un* p_un = &u;p_un->a = 20;printf("%d\\n",sizeof(union un));//4return 0;
}

联合体空间开辟问题

  • 联合体的地址和联合体中最大元素的地址是相同的

C-关键字(下)

  • 联合体中最小元素b的地址也是联合体起始地址,也就是最低地址.

C-关键字(下)

所以,联合体中的所有变量的起始地址在数字上都是相同的,取最大变量的大小开辟空间,所有变量以放射状的方式从低地址向高地址开辟空间.

C-关键字(下)

  • 开辟的空间是属于大家的,每个变量都认为自己独占所有空间,每一个元素都是第一个元素.

利用联合体的性质,判断机器是大端还是小端

C-关键字(下)

int main()
{union un u;u.a = 1;if (u.b == 1){printf("小端机器");}else{printf("大端机器");}
}

C-关键字(下)

enum枚举

存储常量.制作整合一些具有强相关性的常量.使用英文单词相对于数字,具有自描述性.

enum en
{RED,BLACK,BLUE,GREEN
};
int main()
{int a = BLUE;//如果直接用数字初始化a,需要添加注释别人才能看懂printf("%d\\n",a);//2enum en c = BLUE;printf("%d\\n",RED);//0printf("%d\\n",BLACK);//1
}

typedef

类型重命名

typedef unsigned int u_int;
struct data
{ int num;int arr[];
} a;//a 叫全局定义的一个变量
typedef struct Student
{int num;char name[10];char sex;
}Stu;//Stu 就是结构体类型typedef int a[10];//a是一个数组类型
int main()
{u_int t = -10;Stu s;a b;return 0;
}
typedef vs #define
  • 类型重命名并不是简单的某种类型替换,应该理解为一个全新的类型,而不是替换之后*先和那个变量结合.

C-关键字(下)

C-关键字(下)

  • typedef 之后不能再引入其他关键字来修饰类型

C-关键字(下)

typedef static int int32_t 行不行
  • 存储类型关键字(5个)

auto :声明自动变量,一般不使用
extern :声明变量是在其他文件中声明
register :声明寄存器变量
static :声明静态变量
typedef :用以给数据类型取别名(但是该关键字被分到存储关键字分类中,虽然看起来没什么相关性)

存储类型关键字不可以同时出现,在一个变量定义的时候只能有一个.报错:指定一个以上的存储类

  • 其他关键字(3个)

    const :声明只读变量
    sizeof :计算数据类型长度
    volatile :说明变量在程序执行中可被隐含地改变

  • 控制语句关键字(12个)

1. 循环控制(5个)
for :一种循环语句
do :循环语句的循环体
while :循环语句的循环条件
break :跳出当前循环
continue :结束当前循环,开始下一轮循环
2. 条件语句(3个)
if : 条件语句
else :条件语句否定分支
goto :无条件跳转语句
3. 开关语句 (3个)
switch :用于开关语句
case :开关语句分支
default :开关语句中的“其他”分支
4. 返回语句(1个)
return :函数返回语句(可以带参数,也看不带参数)
  • 数据类型关键字(12个)

    char :声明字符型变量或函数
    short :声明短整型变量或函数
    int : 声明整型变量或函数
    long :声明长整型变量或函数
    signed :声明有符号类型变量或函数
    unsigned :声明无符号类型变量或函数
    float :声明浮点型变量或函数
    double :声明双精度变量或函数
    struct :声明结构体变量或函数
    union :声明共用体(联合)数据类型
    enum :声明枚举类型
    void :声明函数无返回值或无参数,声明无类型指针