a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 編程 > C > 正文

C語言中const,volatile,restrict的用法總結

2020-01-26 15:43:01
字體:
來源:轉載
供稿:網友

1. const

變量聲明中帶有關鍵詞const,意味著不能通過賦值,增量或減量來修改該變量的值,這是顯而易見的一點。指針使用const則要稍微復雜點,因為不得不把讓指針本身成為const和指針指向的值成為const區別開來、下面的聲明表示pf指向的值必須是不變的

constfloat *pf;而pf則是可變的,它可以指向另外一個const或非const值;相反,下面的聲明說明pf是不能改變的,而pf所指向的值則是可以改變的:

float* const pf;

最后,當然可以有既不能改變指針的值也不能改變指針指向的值的值的聲明方式:

constfloat * const pf;

需要注意的是,還有第三種放置const關鍵字的方法:

float const * pf; //等價于constfloat * pf;

總結就是:一個位于*左邊任意位置的const使得數據成為常量,而一個位于*右邊的const使得指針本身成為const

還要注意的一點是關于const在全局數據中的使用:

使用全局變量被認為是一個冒險的方法,它使得數據在程序的任何部分都可以被錯誤地修改,如果數據是const,那么這種擔心就是多余的了不是嘛?因此對全局數據使用const是合理的。

然而,在文件之間共享const數據要格外小心,有兩個策略可以使用。一個是遵循外部變量的慣用規則,在一個文件進行定義聲明,在其他文件進行引用聲明(使用關鍵字extern)。

/*file1.c------定義一些全局常量*/

const double PI = 3.14159;

/*file2.c-----是用在其他文件中定義的全局變量*/

extern const dounle PI;

另外一個方法是把全局變量放在一個include文件里,這時候需要格外注意的是必須使用靜態外部存儲類

/*constant.h----定義一些全局常量*/

static const double PI = 3.14159;

/*file1.c-----使用其他文件定義的全局變量*/

#include”constant.h”。

/*file2.c-----使用其他文件定義的全局變量*/

#include”constant.h”

如果不使用關鍵字static,在文件file1.c和file2.c中包含constant.h將導致每個文件都有同一標識符的定義聲明ANSI標準不支持這樣做(有些編譯器確實支持)。通過使用static, 實際上給了每個文件一個獨立的數據拷貝,如果文件想使用該數據與另外一個文件通話,這樣做就不行了,因為每個文件只能看見他自己的拷貝,然而由于數據是不 可變的,這就不是問題了。使用頭文件的好處是不必惦記在一個文件中進行定義聲明,在另一個文件中進行引用聲明,缺點在于復制了數據,如果常量很大的話,這 就是個問題了。

2. volatile

限定詞volatile告訴編譯器,該變量除了可被程序改變意外還可以被其他代理改變。典型的它用于硬件地址和其他并行運行的程序共享的數據。例如,一個地址中可能保存著當前的時鐘信息。不管程序做些什么,該地址會隨時間改變。另一種情況是一個地址用來接收來自其他計算機的信息;

語法同const:

volatile int a;//a是一個易變的位置

volatile int * pf;//pf指向一個易變的位置

把volatile作為一個關鍵字的原因是它可以方便編譯器優化。

假如有如下代碼:

va= x;

//一些不使用x的代碼

vb= x;

一個聰明的編譯器可能注意到你兩次使用了x,但是沒有改變它的值,它將把x臨時存貯在一個寄存器中,接著,當vb主要x是的時候,它從寄存器而非初始的內存位置得到x的值來節省時間。這個過程被稱為緩存。通常緩存是一個好的優化方式,但是如果兩個語句中間的其他代理改變了x的值的話就不是這樣了。如果沒有規定volatile關鍵字,編譯器將無從得知這種改變是否可能發生,因此,為了安全起見,編譯器不使用緩存。那是在ANSI以前的情形,現在,如果在聲明中沒有使用volatile關鍵字,編譯器就可以假定一個值在使用過程中沒有修改,它就可以試著優化代碼??偠灾?,volatile使得每次讀取數據都是直接在內存讀取而不是緩存。

你可能會覺得奇怪,const和volatile可以同時使用,但是確實可以。例如硬件時鐘一般不能由程序改變,這使得他成為const,但他被程序以外的代理改變,這使得他成為volatile,所以你可以同時使用它們,順序是不重要的:

const volatile time;

volatile表明某個變量的值可能在外部被改變,優化器在用到這個變量時必須每次都小心地重新讀取這個變量的值,而不是使用保存在寄存器里的備份。它可以適用于基礎類 型如:int,char,long......也適用于C的結構和C++的類。當對結構或者類對象使用volatile修飾的時候,結構或者類的所有成員 都會被視為volatile.

該關鍵字在多線程環境下經常使用,因為在編寫多線程的程序時,同一個變量可能被多個線程修改,而程序通過該變量同步各個線程。

簡單示例:

復制代碼 代碼如下:

DWORD __stdcall threadFunc(LPVOID signal)
{
int* intSignal=reinterdivt_cast(signal);
*intSignal=2;
while(*intSignal!=1)
sleep(1000);
return 0;
}

該線程啟動時將intSignal 置為2,然后循環等待直到intSignal 為1 時退出。顯然intSignal的值必須在外部被改變,否則該線程不會退出。但是實際運行的時候該線程卻不會退出,即使在外部將它的值改為1,看一下對應的偽匯編代碼就明白了:

mov ax,signal
label:
if(ax!=1)
goto label

對于C編譯器來說,它并不知道這個值會被其他線程修改。自然就把它cache在寄存器里面。C 編譯器是沒有線程概念的,這時候就需要用到volatile。volatile 的本意是指:這個值可能會在當前線程外部被改變。也就是說,我們要在threadFunc中的intSignal前面加上volatile關鍵字,這時 候,編譯器知道該變量的值會在外部改變,因此每次訪問該變量時會重新讀取,所作的循環變為如下面偽碼所示:
label:
mov ax,signal
if(ax!=1)
goto label

注意:一個參數既可以是const同時是volatile,是volatile因為它可能被意想不到地改變。它是const因為程序不應該試圖去修改它。

3.restrict

關鍵字restrict通過允許編譯器優化某幾種代碼增強了計算支持。記住,它只能用于指針,并且表明指針是訪問一個數據對象的唯一且初始的方式。為了清楚為何這樣做,我們需要看一些例子:

復制代碼 代碼如下:

intar[10];

int* restrict restar = (int*)malloc(10*sizeof(int));

int* par = ar;


這里,指針restar是訪問malloc分配的內存的唯一而且初始的方式,因此聲明為restrict。然而,par指針既不是初始的,也不是訪問數組ar中數據的唯一方式,所以不用restrict限定詞?,F在考慮下面這個更加復雜的例子,其中n是一個int
復制代碼 代碼如下:

for(n= 0;n < 10;n++)

{

par[n]+= 5;

restar[n]+= 5;

ar[n]*= 2;

par[n]+= 3;

restar[n]+= 3;

}


知道了restar是訪問它所指向的數據的唯一初始方式,編譯器就可以用具有同樣效果的一條語句來替代包含restar的兩個語句

restar[n]+= 8;/*可以替換*/

然而將兩個計算par的語句精簡為一個則會導致錯誤因為在par兩次訪問數據之間,ar改變了該數據的值。沒有關鍵字restrict,編譯器將不得不設想比較糟糕的那一種形式,而使用之后,編譯器可以放心大膽的尋找計算的捷徑。可以將關鍵字作為指針型函數參量的限定詞使用,這意味著編譯器可以假定在函數體內沒有其他標志符修改指針指向的數據,因而可以試著優化代碼,反之不然。來看一下C99標準下C庫中的兩個函數,他們從一個位置把字節復制到另一個位置

void*memcpy(void* restrict s1,const void* restrict s2,size_t n);

void*memmove(void* s1,const void * s2,size_t);

memcpy要求兩個指針的位置不能重疊,但memmove沒有這個要求。把s1,s2聲明為restrict意味著每個指針都是相應數據的唯一訪問方式,因此他們不能訪問同一數據塊。這滿足了不能有重疊的要求。

關鍵字restrict有兩個讀者:編譯器,它告訴編譯器可以自由地做一些優化的假定。另一個讀者是用戶,他告訴用戶僅使用滿足restrict要求的參數。一般,編譯器沒法檢查你是否遵循了這一限制,如果你蔑視它,也就是讓自己冒險。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 九九亚洲精品 | 久久久久国产一区二区三区四区 | 日韩一区中文字幕 | 国产乱码精品1区2区3区 | 成人国产精品久久 | 先锋影音在线 | 97中文字幕 | 国产精品视频网站 | 日本视频免费高清一本18 | 午夜av亚洲女人剧场se | 国产偷国产偷精品高清尤物 | 三级精品 | 草逼操 | 伊人免费观看视频 | 国产91视频一区二区 | 成人免费视频国产免费麻豆 | 久久精品一区二区三区四区 | 人人鲁人人莫一区二区三区 | 日韩有码在线播放 | 久久香蕉网 | 日韩h视频| 日本不卡免费新一二三区 | 亚洲欧洲一区二区 | 国产在线视频网站 | 日韩av在线中文字幕 | 日韩精品一区二区三区在线 | 国产午夜精品一区二区三区 | 久久不射电影网 | 一区二区精品 | 久久理论片 | 欧美激情在线狂野欧美精品 | 羞羞视频在线观看视频 | 成人免费视频网站在线观看 | 欧美日韩在线第一页 | 91在线精品一区二区 | 日韩免费高清 | 呦呦av在线 | 国产精品自产拍在线观看桃花 | 精品亚洲网 | 欧美美女黄色网 | 不用播放器的免费av |