当前位置:主页 > c/c++教程 > C语言关键字

C语言学习之关键字的示例详解

发布:2023-03-04 20:00:02 59


给网友们整理相关的编程文章,网友戈元明根据主题投稿了本篇教程内容,涉及到C语言、关键字、C语言关键字相关内容,已被838网友关注,如果对知识点想更进一步了解可以在下方电子资料中获取。

C语言关键字

1. 前言

大家好,我是努力学习游泳的鱼。关键字,这名字一听,就很关键。而有些关键字,你可能不是很了解,更别谈使用。所以,这篇文章将带你见识常见的关键字,一起领略它们的风采吧。

2. 什么是关键字

C语言提供了丰富的关键字,这些关键字都是语言本身预先设定好的,

用户自己是不能创造关键字的。

大部分关键字会在其他章节介绍,这里仅介绍一些稍微有点难度的关键字。

3. extern-声明外部符号

extern可以用来声明外部符号,如外部的全局变量和函数。

如我们在test1.c里定义了全局变量aint a = 2022;

我们想在test2.c里使用,就得先用extern声明一下extern int a;

注意:一般extern是用来声明外部的全局变量的。因为如果直接写int a;就不是声明了,而是定义,会直接创建一个变量a。只有写extern int a;才是声明变量a。如果是声明外部的函数,可以省略掉extern。如直接写int Add(int, int);和写extern int Add(int, int);效果是相同的。

4. auto-自动

C语言里的局部变量,进入局部范围时自动创建,出局部范围时自动销毁。这种自动创建,自动销毁的特性,其实是由于前面省略了关键字auto。比如,int a = 0;其实编译器会处理为auto int a = 0;一般来说,auto会被省略掉。

5. typedef-类型重定义(类型重命名)

typedef关键字用于给类型起别名,相当于起了个外号。

比如unsigned int num = 10;如果我们嫌unsigned int这个类型写起来太麻烦了,可以给它起个别名叫做uint:typedef unsigned int uint;这样上面的代码就等价于uint num = 10;

6. register-寄存器

6.1 存储器

数据的存储,需要存储器。常见的存储器有:

网盘,硬盘,内存,高级缓存,寄存器。

从左到右,速度越快,从而造价越高,从而空间越小。

早期,CPU处理的数据都来自内存。当时,CPU的处理速度和内存的读写速度是差不多的。随着技术的迭代,内存的读写速度逐渐跟不上CPU的处理速度,CPU在很大程度上被闲置了。

于是就有了这么一层设计。在内存之上设置读写速度更快的高级缓存和寄存器。CPU从寄存器中拿数据,与此同时,寄存器从高级缓存中拿数据,高级缓存从内存中拿数据。如果CPU想要的数据在寄存器中没有,那就直接从高级缓存中拿数据,如果还没有再从内存中拿。由于大部分数据都能在寄存器中命中,整体上,处理数据的速度就提升了。

以上,我们能明白一点:

寄存器的读写速度是非常快的!

6.2 register关键字的作用

如果我们写int num = 10;num是放在内存中的。如果我们加了个registerregister int num = 10;此时register的作用是建议把num放在寄存器中。注意只是建议,实际是否放在寄存器中取决于编译器的处理。

7. static-静态

在C语言中,static有3种用法,分别修饰局部变量,全局变量和函数。

1.修饰局部变量-称为静态局部变量

2.修饰全局变量-称为静态全局变量

3.修饰函数-称为静态函数

7.1 static修饰局部变量

7.1.1 代码对比

下面代码的输出结果是多少呢?

#include 

void test()
{
	int a = 5;
	a++;
	printf("%d ", a);
}

int main()
{
	int i = 0;
	while (i < 10)
	{
		test();
		i++;
	}

	return 0;
}

输出结果:

10个6

为什么呢?test函数被调用了10次,每次都做了同样一件事,创建a并初始化为5,a自增变成6,打印a(即6)。本质上,每次进入test函数都会创建a,出test函数时都会销毁a。这是由于局部变量的特性:进入局部范围创建,出局部范围销毁。那么,每次进入test函数创建的都是一个新的a,和之前创建的a没有任何关系。
明白这点后,再看下面这段代码,输出的结果又是多少?

#include 

void test()
{
	static int a = 5;
	a++;
	printf("%d ", a);
}

int main()
{
	int i = 0;
	while (i < 10)
	{
		test();
		i++;
	}

	return 0;
}

答案:

输出6~15。

分析一下:第一次调用test函数时和没有static相同,创建a并初始化,自增,打印(此时a是6),但第二次调用怎么就打印7了呢?这说明,第二次调用时,a还是上次调用留下来的6,才会自增变成7!也就是说,第一次调用结束后,a并没有销毁,第二次调用时依然存在。同理,第二次调用后a也没有销毁,第三次调用时a仍是第二次调用留下来的7,然后自增变成8后打印,以此类推。

static修饰局部变量的时候,局部变量就变成了静态的局部变量,出了局部的范围,不会销毁,下一次进入函数依然存在。

7.1.2 原理分析

内存可以分为:栈区,堆区,静态区,等等。

栈区存储的是局部变量,函数参数,等等。

堆区是用来动态内存开辟的,与之相关的函数有malloc,realloc,calloc和free等等。

静态区存储的是静态变量和全局变量。

静态的局部变量出了作用域依然存在,是因为它是存储在静态区的。

同样存储在静态区的全局变量,生命周期也很长。

static修饰局部变量时,实际改变的是变量的存储位置,本来一个局部变量是放在栈区的,被static修饰后放在了静态区,从而导致,出了作用域依然存在,生命周期并没有结束。

注意:放在静态区的变量出了作用域不销毁,相当于生命周期变长了,但是作用域并没有发生变化,也就是说,静态的局部变量仍然只能在它的局部范围内使用!
静态区中的数据的生命周期和程序的生命周期是一致的。程序结束,静态数据的生命周期也就到了。

7.2 static修饰全局变量

7.2.1 代码对比

我们创建两个源文件,test1.c和test2.c

在test1.c里定义一个全局变量g_val

// test1.c
int g_val = 2022; // 全局变量,定义在test1.c中

在test2.c内部使用这个全局变量,由于全局变量的作用域是整个工程,所以可以跨源文件使用。但是在使用前需要使用extern声明,否则会报编译错误。

// test2.c
extern int g_val;

int main()
{
	g_val = 2023;
	
	return 0;
}

如果我们在g_val的定义前面加上static会发生什么呢?

// test1.c
static int g_val = 2022; // 全局变量,定义在test1.c中

// test2.c
extern int g_val;

int main()
{
	g_val = 2023;
	
	return 0;
}

此时会报链接错误,因为g_val是定义在test1.c里的静态全局变量,不能在test2.c内部使用。看来静态的全局变量不能跨文件使用了。

7.2.2 原理分析

一个全局变量本来是具有外部链接属性的,既能在自己所在的源文件内部使用,也能在其他文件内部使用。

但是被static修饰之后外部链接属性就变成了内部链接属性,只能在自己所在的源文件内部使用,不能在其他文件内部使用了。

使用上感觉作用域变小了。

7.3 static修饰函数

7.3.1 代码对比

我们在test1.c里定义一个函数

// test1.c
int Add(int x, int y)
{
    return x + y;
}

在test2.c内部使用,同理要先声明(此时可以省略extern),否则会报一个警告。

// test2.c
#include 

extern int Add(int, int); // extern可以省略

int main()
{
	int sum = Add(10, 20);
	printf("sum = %d\n", sum);
	
	return 0;
}

如果在函数定义前加上static会发生什么呢?

// test1.c
static int Add(int x, int y)
{
	return x + y;
}

// test2.c
#include 

extern int Add(int, int); // extern可以省略

int main()
{
	int sum = Add(10, 20);
	printf("sum = %d\n", sum);
	
	return 0;
}

此时会报链接错误,因为Add函数是定义在test1.c内部的静态函数,不能在test2.c内部使用。看来static修饰函数和修饰全局变量类似,静态的函数也不能跨文件调用。

7.3.2 原理分析

static修饰函数的作用:一个函数本来是具有外部链接属性的,但是被static修饰之后,外部链接属性就变成了内部链接属性,这时这个函数只能在自己所在的源文件内部使用,其他文件是无法使用的。

使用上的感觉好像是作用域变小了。

到此这篇关于C语言学习之关键字的示例详解的文章就介绍到这了,更多相关C语言关键字内容请搜索码农之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持码农之家!


参考资料

相关文章

  • 深入理解C语言指针

    发布:2022-06-22

    为网友们分享了关于C语言的教程,关于指针,其是C语言的重点,C语言学的好坏,其实就是指针学的好坏。其实指针并不复杂,学习指针,要正确的理解指针


  • C语言实现导航功能

    发布:2022-11-03

    给大家整理了关于C语言的教程,这篇文章主要为大家详细介绍了C语言实现导航功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下


  • C语言程序中结构体的内存对齐详解

    发布:2023-03-10

    这篇文章主要为大家详细介绍了C语言程序中结构体的内存对齐的相关资料,文中的示例代码讲解详细,具有一定的参考价值,感兴趣的小伙伴可以了解一下


  • 用C语言求解一元二次方程的简单实现

    发布:2023-03-13

    这篇文章主要介绍了用C语言求解一元二次方程的简单实现方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教


  • C语言如何实现成绩等级判别

    发布:2023-03-12

    这篇文章主要介绍了C语言如何实现成绩等级判别,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教


  • C语言实现打印数字金字塔

    发布:2023-03-11

    这篇文章主要介绍了C语言实现打印数字金字塔方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教


  • C语言三种函数调用约定_cdecl与_stdcall及_fastcall详细讲解

    发布:2023-03-05

    本篇文章使用的工具是vs2010,内容可能涉及到汇编的知识,建议有一些汇编基础的再来看,不过没有汇编基础也没有关系,了解一下这三种调用约定即可


  • C语言中指针 int *p=0;和int *p;*p=0;和”&“的关系和区别详解

    发布:2022-04-13

    这篇文章主要介绍了C语言中指针 int *p=0;和int *p;*p=0;和”&“有什么关系和区别,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下


网友讨论