C語言學習之關(guān)鍵字的示例詳解
1. 前言
大家好,我是努力學習游泳的魚。關(guān)鍵字,這名字一聽,就很關(guān)鍵。而有些關(guān)鍵字,你可能不是很了解,更別談使用。所以,這篇文章將帶你見識常見的關(guān)鍵字,一起領(lǐng)略它們的風采吧。
2. 什么是關(guān)鍵字
C語言提供了豐富的關(guān)鍵字,這些關(guān)鍵字都是語言本身預先設(shè)定好的,
用戶自己是不能創(chuàng)造關(guān)鍵字的。
大部分關(guān)鍵字會在其他章節(jié)介紹,這里僅介紹一些稍微有點難度的關(guān)鍵字。
3. extern-聲明外部符號
extern可以用來聲明外部符號,如外部的全局變量和函數(shù)。
如我們在test1.c里定義了全局變量aint a = 2022;
我們想在test2.c里使用,就得先用extern聲明一下extern int a;
注意:一般extern是用來聲明外部的全局變量的。因為如果直接寫int a;就不是聲明了,而是定義,會直接創(chuàng)建一個變量a。只有寫extern int a;才是聲明變量a。如果是聲明外部的函數(shù),可以省略掉extern。如直接寫int Add(int, int);和寫extern int Add(int, int);效果是相同的。
4. auto-自動
C語言里的局部變量,進入局部范圍時自動創(chuàng)建,出局部范圍時自動銷毀。這種自動創(chuàng)建,自動銷毀的特性,其實是由于前面省略了關(guān)鍵字auto。比如,int a = 0;其實編譯器會處理為auto int a = 0;一般來說,auto會被省略掉。
5. typedef-類型重定義(類型重命名)
typedef關(guān)鍵字用于給類型起別名,相當于起了個外號。
比如unsigned int num = 10;如果我們嫌unsigned int這個類型寫起來太麻煩了,可以給它起個別名叫做uint:typedef unsigned int uint;這樣上面的代碼就等價于uint num = 10;
6. register-寄存器
6.1 存儲器
數(shù)據(jù)的存儲,需要存儲器。常見的存儲器有:
網(wǎng)盤,硬盤,內(nèi)存,高級緩存,寄存器。
從左到右,速度越快,從而造價越高,從而空間越小。
早期,CPU處理的數(shù)據(jù)都來自內(nèi)存。當時,CPU的處理速度和內(nèi)存的讀寫速度是差不多的。隨著技術(shù)的迭代,內(nèi)存的讀寫速度逐漸跟不上CPU的處理速度,CPU在很大程度上被閑置了。
于是就有了這么一層設(shè)計。在內(nèi)存之上設(shè)置讀寫速度更快的高級緩存和寄存器。CPU從寄存器中拿數(shù)據(jù),與此同時,寄存器從高級緩存中拿數(shù)據(jù),高級緩存從內(nèi)存中拿數(shù)據(jù)。如果CPU想要的數(shù)據(jù)在寄存器中沒有,那就直接從高級緩存中拿數(shù)據(jù),如果還沒有再從內(nèi)存中拿。由于大部分數(shù)據(jù)都能在寄存器中命中,整體上,處理數(shù)據(jù)的速度就提升了。
以上,我們能明白一點:
寄存器的讀寫速度是非??斓模?/p>
6.2 register關(guān)鍵字的作用
如果我們寫int num = 10;num是放在內(nèi)存中的。如果我們加了個registerregister int num = 10;此時register的作用是建議把num放在寄存器中。注意只是建議,實際是否放在寄存器中取決于編譯器的處理。
7. static-靜態(tài)
在C語言中,static有3種用法,分別修飾局部變量,全局變量和函數(shù)。
1.修飾局部變量-稱為靜態(tài)局部變量
2.修飾全局變量-稱為靜態(tài)全局變量
3.修飾函數(shù)-稱為靜態(tài)函數(shù)
7.1 static修飾局部變量
7.1.1 代碼對比
下面代碼的輸出結(jié)果是多少呢?
#include <stdio.h>
void test()
{
int a = 5;
a++;
printf("%d ", a);
}
int main()
{
int i = 0;
while (i < 10)
{
test();
i++;
}
return 0;
}
輸出結(jié)果:
10個6
為什么呢?test函數(shù)被調(diào)用了10次,每次都做了同樣一件事,創(chuàng)建a并初始化為5,a自增變成6,打印a(即6)。本質(zhì)上,每次進入test函數(shù)都會創(chuàng)建a,出test函數(shù)時都會銷毀a。這是由于局部變量的特性:進入局部范圍創(chuàng)建,出局部范圍銷毀。那么,每次進入test函數(shù)創(chuàng)建的都是一個新的a,和之前創(chuàng)建的a沒有任何關(guān)系。
明白這點后,再看下面這段代碼,輸出的結(jié)果又是多少?
#include <stdio.h>
void test()
{
static int a = 5;
a++;
printf("%d ", a);
}
int main()
{
int i = 0;
while (i < 10)
{
test();
i++;
}
return 0;
}
答案:
輸出6~15。
分析一下:第一次調(diào)用test函數(shù)時和沒有static相同,創(chuàng)建a并初始化,自增,打印(此時a是6),但第二次調(diào)用怎么就打印7了呢?這說明,第二次調(diào)用時,a還是上次調(diào)用留下來的6,才會自增變成7!也就是說,第一次調(diào)用結(jié)束后,a并沒有銷毀,第二次調(diào)用時依然存在。同理,第二次調(diào)用后a也沒有銷毀,第三次調(diào)用時a仍是第二次調(diào)用留下來的7,然后自增變成8后打印,以此類推。
static修飾局部變量的時候,局部變量就變成了靜態(tài)的局部變量,出了局部的范圍,不會銷毀,下一次進入函數(shù)依然存在。
7.1.2 原理分析
內(nèi)存可以分為:棧區(qū),堆區(qū),靜態(tài)區(qū),等等。
棧區(qū)存儲的是局部變量,函數(shù)參數(shù),等等。
堆區(qū)是用來動態(tài)內(nèi)存開辟的,與之相關(guān)的函數(shù)有malloc,realloc,calloc和free等等。
靜態(tài)區(qū)存儲的是靜態(tài)變量和全局變量。
靜態(tài)的局部變量出了作用域依然存在,是因為它是存儲在靜態(tài)區(qū)的。
同樣存儲在靜態(tài)區(qū)的全局變量,生命周期也很長。
static修飾局部變量時,實際改變的是變量的存儲位置,本來一個局部變量是放在棧區(qū)的,被static修飾后放在了靜態(tài)區(qū),從而導致,出了作用域依然存在,生命周期并沒有結(jié)束。
注意:放在靜態(tài)區(qū)的變量出了作用域不銷毀,相當于生命周期變長了,但是作用域并沒有發(fā)生變化,也就是說,靜態(tài)的局部變量仍然只能在它的局部范圍內(nèi)使用!
靜態(tài)區(qū)中的數(shù)據(jù)的生命周期和程序的生命周期是一致的。程序結(jié)束,靜態(tài)數(shù)據(jù)的生命周期也就到了。
7.2 static修飾全局變量
7.2.1 代碼對比
我們創(chuàng)建兩個源文件,test1.c和test2.c
在test1.c里定義一個全局變量g_val
// test1.c int g_val = 2022; // 全局變量,定義在test1.c中
在test2.c內(nèi)部使用這個全局變量,由于全局變量的作用域是整個工程,所以可以跨源文件使用。但是在使用前需要使用extern聲明,否則會報編譯錯誤。
// test2.c
extern int g_val;
int main()
{
g_val = 2023;
return 0;
}
如果我們在g_val的定義前面加上static會發(fā)生什么呢?
// 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里的靜態(tài)全局變量,不能在test2.c內(nèi)部使用。看來靜態(tài)的全局變量不能跨文件使用了。
7.2.2 原理分析
一個全局變量本來是具有外部鏈接屬性的,既能在自己所在的源文件內(nèi)部使用,也能在其他文件內(nèi)部使用。
但是被static修飾之后外部鏈接屬性就變成了內(nèi)部鏈接屬性,只能在自己所在的源文件內(nèi)部使用,不能在其他文件內(nèi)部使用了。
使用上感覺作用域變小了。
7.3 static修飾函數(shù)
7.3.1 代碼對比
我們在test1.c里定義一個函數(shù)
// test1.c
int Add(int x, int y)
{
return x + y;
}
在test2.c內(nèi)部使用,同理要先聲明(此時可以省略extern),否則會報一個警告。
// test2.c
#include <stdio.h>
extern int Add(int, int); // extern可以省略
int main()
{
int sum = Add(10, 20);
printf("sum = %d\n", sum);
return 0;
}
如果在函數(shù)定義前加上static會發(fā)生什么呢?
// test1.c
static int Add(int x, int y)
{
return x + y;
}
// test2.c
#include <stdio.h>
extern int Add(int, int); // extern可以省略
int main()
{
int sum = Add(10, 20);
printf("sum = %d\n", sum);
return 0;
}
此時會報鏈接錯誤,因為Add函數(shù)是定義在test1.c內(nèi)部的靜態(tài)函數(shù),不能在test2.c內(nèi)部使用??磥韘tatic修飾函數(shù)和修飾全局變量類似,靜態(tài)的函數(shù)也不能跨文件調(diào)用。
7.3.2 原理分析
static修飾函數(shù)的作用:一個函數(shù)本來是具有外部鏈接屬性的,但是被static修飾之后,外部鏈接屬性就變成了內(nèi)部鏈接屬性,這時這個函數(shù)只能在自己所在的源文件內(nèi)部使用,其他文件是無法使用的。
使用上的感覺好像是作用域變小了。
到此這篇關(guān)于C語言學習之關(guān)鍵字的示例詳解的文章就介紹到這了,更多相關(guān)C語言關(guān)鍵字內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解析c中stdout與stderr容易忽視的一些細節(jié)
本篇文章是對在c語言中stdout與stderr容易忽視的一些細節(jié)進行了詳細的分析介紹,需要的朋友參考下2013-05-05
C++中利用cout和fstream采用非科學計數(shù)法輸出
這篇文章主要介紹了C++中利用cout和fstream采用非科學計數(shù)法輸出方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-11-11
C語言中system()執(zhí)行cmd命令打開關(guān)閉程序的方法
今天小編就為大家分享一篇C語言中system()執(zhí)行cmd命令打開關(guān)閉程序的方法。具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-05-05

