91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何把Vim打造成C/C++的半自動化IDE

發布時間:2021-09-27 13:47:42 來源:億速云 閱讀:177 作者:iii 欄目:系統運維

這篇文章主要講解了“如何把Vim打造成C/C++的半自動化IDE”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“如何把Vim打造成C/C++的半自動化IDE”吧!

感悟一:C語言標準干不過GNU擴展

   最近為了研究X Window的底層協議,開始嘗試使用XCB編程。當我打開XCB的頭文件的時候,我被大量的__restrict__關鍵字驚呆了,好在有GNU C語言手冊為我答疑解惑。__restrict__又是一個GNU擴展的關鍵字,后面我會詳細講解該關鍵字的用途。其實C語言的C99標準中已經引入了restrict關鍵字,沒有前后的下劃線,但是在大量的開源代碼中,使用最普遍的還是GNU的擴展,而不是C語言標準。

  和restrict關鍵字有相同命運的還有inline、_Complex等,它們都是在C99標準中引入的關鍵字,但是其實在C99標準出來之前,GNU C中早就有了__inline__、__complex__等擴展關鍵字。還記得多年前我學習Linux 0.11版的源代碼時,看到大量的__inline__曾經疑惑不已,不知道為什么Linus在91年就能用上了如此先進的語言功能,后來才知道,這是GNU的擴展關鍵字。

  C語言的標準有C89和C99,使用GCC的時候甚至要顯示指定-std=c99才能全面支持C99標準,所以在開源界,大家還是喜歡首選GNU的擴展關鍵字。比如__inline__、__complex__和__restrict__。總而言之,C語言標準干不過GNU擴展。

  下面來看看__restrict__的真正含義。還記得CSDN上曾經載過一篇文章《為什么有些語言會比別的快》,其中提到“很長一段時間,相同的兩個程序在Fortran和C(或者C++)中運行,Fortran會快一些,因為Fortran的優化做的更好。這是真的,就算C語言和Fortran的編譯器用了相同的代碼生成器也是一樣。這個不同不是因為Fortran的某種特性,事實上恰恰相反,是因為Fortran不具備的特性。”這是因為C語言中的指針給編譯器的優化帶來了困難,文章中繼續說道:“問題就來了。這些指針可以代替任何內存地址。更重要的是,他們可以重疊。輸出數組的內存地址也可以同時是輸入數組的。甚至可以部分重疊,輸出數組可以覆蓋一個輸入數組的一半。這對編譯器優化來說是個大問題,因為之前基于數組的優化不再適用。特別的,添加元素的順序也成問題,如果輸出重疊的數組,計算的結果會變得不確定,取決于輸出元素的動作是發生在元素被覆蓋之前還是之后。”

  有了__restrict__,C語言的該問題將不復存在。用__restrict__修飾一個指針后,①該指針只能在定義的時候被初始化;②不會再有別的指針指向該指針指向的內存,因此編譯器可以對算法進行優化。如下代碼:

代碼如下:


int * __restrict__ p = (int*)malloc(100*sizeof(int));


指針p有__restrict__關鍵字修飾,所以它只能在定義的時候被初始化,以后不能賦值,而沒有__restrict__修飾的指針,可以隨時賦值,如下:

代碼如下:


int arr[100];
int* pArr;
pArr = arr;


指針pArr沒有被__restrict__關鍵字修飾,所以可以將數組的首地址賦值給它。

比如我們定義一個函數對兩塊數據進行操作,結果放入第3塊內存,如下:

代碼如下:


void func1(void* p1, void* p2, void* p3, int size){
   for(int i=0; i<size; i++){
       p3[i] = p1[i] + p2[i];
   }
}


很顯然,由于編譯器沒辦法判斷指針p1、p2、p3指向的內存是否重疊,所以無法進行優化,加上__restrict__關鍵字后,如下:

代碼如下:


void func1(void* __restrict__ p1, void* __restrict__ p2, void* __restrict__ p3, int size){
   for(int i=0; i<size; i++){
       p3[i] = p1[i] + p2[i];
   }
}

相當于明確告訴編譯器這幾塊內存不會重疊,所以編譯器就可以放心大膽對程序進行優化。

   另一個關鍵字是_Complex,C99才引入,而且需要包含<complex.h>頭文件。其實在GNU C中,早就有__complex__、__real__、__imag__等擴展關鍵字。如下代碼:
 

代碼如下:


#include <stdlib.h>
#include <stdio.h></p> <p>int main(){
   __complex__ a = 3 + 4i;
   __complex__ b = 5 + 6i;
   __complex__ c = a + b;
   __complex__ d = a * b;
   __complex__ e = a / b;
   printf("a + b = %f + %fi\n", __real__ c, __imag__ c);
   printf("a * b = %f + %fi\n", __real__ d, __imag__ d);
   printf("a / b = %f + %fi\n", __real__ e, __imag__ e);
   return 0;
}

  可以看到,在C語言中也可以直接對復數進行計算。數值計算再也不是Fortran的專利。

 感悟二:指針和數組還真是不一樣

  從學C語言開始,老師就教導我們說指針和數組是一樣的,它們可以用同樣的方式進行操作。而事實上,指針和數組還是有差別的。直到多年后讀《C專家編程》,才直到所謂指針和數組一樣是一個美麗的錯誤,只是因為在《The C Programming Language》這本書里,把“作為函數參數時,指針和數組一樣”這樣一句話前后分開分別印到了兩頁而已。

  比如,指針不保存數據的長度信息,而數組有,如下代碼:

代碼如下:


#include <stdlib.h>
#include <stdio.h></p> <p>int main(){
   int* p = (int*)malloc(100*sizeof(int));
   int arr[100] = {0};
   printf("The size of p: %d\n", sizeof(p));
   printf("The size of arr: %d\n", sizeof(arr));
   return 0;
}

這段代碼的運行結果為:

代碼如下:


The size of p: 8
The size of arr: 400


  我們經常可以使用如下的代碼片段來獲得一個數組中有多少個元素,如下:

代碼如下:


int arr[100];
size_t length = sizeof(arr)/sizeof(int);


  但是,當使用數組作為函數的參數的時候,數組會退化成指針。如下代碼:

代碼如下:


#include <stdlib.h>
#include <stdio.h></p> <p>void test_array(int arr[]){
   printf("The size of arr in function: %d\n", sizeof(arr));
   return;
}</p> <p>int main(){
   int arr[100] = {0};
   printf("The size of arr in main: %d\n", sizeof(arr));
   test_array(arr);
   return 0;
}

這段代碼的運行結果為:

代碼如下:


The size of arr in main: 400
The size of arr in function: 8

感悟三:C語言中的不完全類型(Incomplete Types)

   在GNU C中可以定義不完全類型,不完全類型主要有兩種,一種是空的結構,一種是空的數組,比如:

代碼如下:


struct point;
char name[0];


空的結構不能定義變量,只能使用空結構的指針。空結構可以在后面再將它補充完整,如下:

代碼如下:


struct point{
   int x,y;
};


空結構在定義鏈表的時候經常用到,如下:

代碼如下:


struct linked_list{
   struct linked_list* next;
   int x;
   /*other elements here perhaps */
}
struct linked_list* head;

  還有一種不完全類型就是將一個結構的最后一項定義為一個空的數組,這樣可以用來表示一個可變長度的結構或數組,演示該技術的代碼如下:

代碼如下:


#include <stdlib.h>
#include <stdio.h></p> <p>typedef struct {
   int length;
   int arr[0];
} incomplete_type;</p> <p>int main(){
   char hello[] = "Hello, world!";
   int length = sizeof(hello) / sizeof(char);
   incomplete_type* p = (incomplete_type*)malloc(sizeof(int) + length*sizeof(char));
   p->length = length;
   for(int i=0; i<p->length; i++){
       p->arr[i] = hello[i];
   }
   printf("p->length=%d\n", p->length);
   printf("p->arr=%s\n", p->arr);
}

打造C/C++的IDE

  后面的內容展示如何將Vim打造成一個半自動的C/C++ IDE。讀過我的Java博客的朋友應該知道,其實我更喜歡用Eclipse。只有在需要寫非常簡單的程序(比如做習題)的情況下,我才會用Vim。這在我的《打造屬于自己的Vim》中有論述。在這篇文章中我展示了怎么使用Vundle管理插件以及怎么怎么閱讀幫助文檔,同時展示了taglist.vim的簡單用法。如果要用Vim來寫C/C++程序,還需要做少許擴展。

  第一,安裝以下幾個插件,由于使用Vundle管理插件,所以只需要把插件名寫入.vimrc配置文件,然后運行:BundleInstall即可,如下圖:

如何把Vim打造成C/C++的半自動化IDE分別介紹一下這幾個插件。The-NERD-tree是一個瀏覽目錄和文件的插件,可以使用:help NERD_tree.txt查看它的幫助文檔。taglist.vim是瀏覽符號以及在符號之間跳轉的插件,使用:help taglist.txt查看它的幫助文檔。a.vim是在源代碼文件和頭文件之間跳轉的插件,不需要幫助文檔,它的命令就是:A。c.vim是提供IDE功能的主要插件,它提供的功能有自動注釋、反注釋、自動插入代碼塊及自動運行,如果安裝了splint,還可以對代碼進行靜態檢查,使用:help csupport.txt查看它的文檔。OmniCppComplete是一個提供自動補全功能的插件,使用:help omnicppcomplete.txt查看它的文檔。

  這些插件中,taglist.vim和OmniCppComplete需要ctags軟件的支持,所以需要安裝exuberant-ctags軟件包,在Fedora 20中,只需要使用yum install ctags即可自動安裝。

  第二,生成tags數據庫,并將其加入到Vim中。

  我們寫C程序的時候,使用到的文件主要存在于兩個地方,一個是我們工作的當前目錄,另外一個是/usr/include。所以要到/usr/include目錄下使用ctags命令生成tags數據庫文件。為了使tags數據庫中包含盡可能多的信息(結構、枚舉、類、函數、宏定義等等),需要指定ctags的參數,如下:

如何把Vim打造成C/C++的半自動化IDE

 然后將該tags文件的路徑加入到.vimrc配置文件中,同時設置一個鍵盤映射,使得按Ctrl+F12時,在工作目錄中調用ctags命令。如下配置文件的最后兩行:
如何把Vim打造成C/C++的半自動化IDE

  然后,在使用Vim寫C程序的時候,如果輸入了.、->這樣的元素,則其成員會自動補全。如果輸入的是一個字符串(比如函數名),可以按Ctrl-X Ctrl-O調用自動補全,如下圖:

如何把Vim打造成C/C++的半自動化IDE

不僅會彈出候選窗口,而且在最上面的窗口中會顯示函數的完整的簽名,及其所在的文件。這對于我們經常記不全函數名、記不清函數簽名的人來說,已經是莫大的福音了。

  taglist.vim和OmniCppComplete插件提供的功能用起來都只需要一個命令,而c.vim提供的命令就比較多了。而且在c.vim的幫助文檔中并沒有列出所有功能的命令,有一個辦法可以學習這些命令,那就是打開GVim,通過GVim菜單中的C/C++菜單來學習c.vim提供的功能和命令。

  相比網上其它的將Vim打造成IDE的文章,我的配置比較簡單,基本上只安裝了幾個插件,而沒有做過多的設置。當我需要某個功能的時候,我會使用命令顯式地調用它,所以,稱它為半自動化IDE吧。

感謝各位的閱讀,以上就是“如何把Vim打造成C/C++的半自動化IDE”的內容了,經過本文的學習后,相信大家對如何把Vim打造成C/C++的半自動化IDE這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

富平县| 旌德县| 图木舒克市| 登封市| 贵州省| 桓台县| 平乡县| 许昌县| 灌南县| 宝兴县| 平顺县| 吉木乃县| 酒泉市| 建平县| 遵义市| 永胜县| 海宁市| 孝感市| 延津县| 湘阴县| 宜春市| 麻城市| 祥云县| 临清市| 东阳市| 时尚| 平利县| 双流县| 南召县| 鲁山县| 利津县| 沽源县| 城口县| 获嘉县| 云林县| 米脂县| 靖宇县| 沾益县| 葫芦岛市| 竹山县| 台北县|