位置: 编程技术 - 正文

Linux折腾记(六):感悟GNU C及把Vim打造成C/C++的半自动化IDE(,linux)

编辑:rootadmin

推荐整理分享Linux折腾记(六):感悟GNU C及把Vim打造成C/C++的半自动化IDE(,linux),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:linux !,linux 746,linux iv,linux iv,linux !,linux 746,linuxu,linux !,内容如对您有帮助,希望把文章链接给更多的朋友!

C语言在Linux系统中的重要性自然是无与伦比、不可替代,所以我写Linux江湖系列不可能不提C语言。C语言是我的启蒙语言,感谢C语言带领我进入了程序世界。虽然现在不靠它吃饭,但是仍免不了经常和它打交道,特别是在Linux系统下。

  Linux系统中普遍使用的是GNU-C,这里有一份Gnu-C语言手册.pdf。The GNU C Reference Manual的主页在这里:   最近为了研究X Window的底层协议,开始尝试使用XCB编程。当我打开XCB的头文件的时候,我被大量的__restrict__关键字惊呆了,好在有GNU C语言手册为我答疑解惑。__restrict__又是一个GNU扩展的关键字,后面我会详细讲解该关键字的用途。其实C语言的C标准中已经引入了restrict关键字,没有前后的下划线,但是在大量的开源代码中,使用最普遍的还是GNU的扩展,而不是C语言标准。

  和restrict关键字有相同命运的还有inline、_Complex等,它们都是在C标准中引入的关键字,但是其实在C标准出来之前,GNU C中早就有了__inline__、__complex__等扩展关键字。还记得多年前我学习Linux 0.版的源代码时,看到大量的__inline__曾经疑惑不已,不知道为什么Linus在年就能用上了如此先进的语言功能,后来才知道,这是GNU的扩展关键字。

  C语言的标准有C和C,使用GCC的时候甚至要显示指定-std=c才能全面支持C标准,所以在开源界,大家还是喜欢首选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(*sizeof(int));指针p有__restrict__关键字修饰,所以它只能在定义的时候被初始化,以后不能赋值,而没有__restrict__修饰的指针,可以随时赋值,如下:复制代码代码如下:int arr[];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,C才引入,而且需要包含<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 + %fin", __real__ c, __imag__ c); printf("a * b = %f + %fin", __real__ d, __imag__ d); printf("a / b = %f + %fin", __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(*sizeof(int)); int arr[] = {0}; printf("The size of p: %dn", sizeof(p)); printf("The size of arr: %dn", sizeof(arr)); return 0;}

这段代码的运行结果为:复制代码代码如下:The size of p: 8The size of arr:   我们经常可以使用如下的代码片段来获得一个数组中有多少个元素,如下:复制代码代码如下:int arr[];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: %dn", sizeof(arr)); return;}</p><p>int main(){ int arr[] = {0}; printf("The size of arr in main: %dn", sizeof(arr)); test_array(arr); return 0;}

这段代码的运行结果为:复制代码代码如下:The size of arr in main: The size of arr in function: 8

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

Linux折腾记(六):感悟GNU C及把Vim打造成C/C++的半自动化IDE(,linux)

  在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=%dn", p->length); printf("p->arr=%sn", p->arr);}

打造C/C++的IDE

  后面的内容展示如何将Vim打造成一个半自动的C/C++ IDE。读过我的Java博客的朋友应该知道,其实我更喜欢用Eclipse。只有在需要写非常简单的程序(比如做习题)的情况下,我才会用Vim。这在我的《打造属于自己的Vim》中有论述。在这篇文章中我展示了怎么使用Vundle管理插件以及怎么怎么阅读帮助文档,同时展示了taglist.vim的简单用法。如果要用Vim来写C/C++程序,还需要做少许扩展。

  第一,安装以下几个插件,由于使用Vundle管理插件,所以只需要把插件名写入.vimrc配置文件,然后运行:BundleInstall即可,如下图:

分别介绍一下这几个插件。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 中,只需要使用yum install ctags即可自动安装。

  第二,生成tags数据库,并将其加入到Vim中。

  我们写C程序的时候,使用到的文件主要存在于两个地方,一个是我们工作的当前目录,另外一个是/usr/include。所以要到/usr/include目录下使用ctags命令生成tags数据库文件。为了使tags数据库中包含尽可能多的信息(结构、枚举、类、函数、宏定义等等),需要指定ctags的参数,如下:

 然后将该tags文件的路径加入到.vimrc配置文件中,同时设置一个键盘映射,使得按Ctrl+F时,在工作目录中调用ctags命令。如下配置文件的最后两行:

  然后,在使用Vim写C程序的时候,如果输入了.、->这样的元素,则其成员会自动补全。如果输入的是一个字符串(比如函数名),可以按Ctrl-X Ctrl-O调用自动补全,如下图:

不仅会弹出候选窗口,而且在最上面的窗口中会显示函数的完整的签名,及其所在的文件。这对于我们经常记不全函数名、记不清函数签名的人来说,已经是莫大的福音了。

  taglist.vim和OmniCppComplete插件提供的功能用起来都只需要一个命令,而c.vim提供的命令就比较多了。而且在c.vim的帮助文档中并没有列出所有功能的命令,有一个办法可以学习这些命令,那就是打开GVim,通过GVim菜单中的C/C++菜单来学习c.vim提供的功能和命令。

  相比网上其它的将Vim打造成IDE的文章,我的配置比较简单,基本上只安装了几个插件,而没有做过多的设置。当我需要某个功能的时候,我会使用命令显式地调用它,所以,称它为半自动化IDE吧。

Linux折腾记(七):硬盘GPT分区和MBR分区爬坑记 之所以想到写这篇,是因为本人在折腾Linux系统的过程中,有多次掉入硬盘分区的陷阱的经历。最近几天,再一次掉入坑中,折腾了两天才从坑中爬出来

Linux折腾记(八):使用GCC和GNU Binutils编写能在x实模式运行的位代码 不可否认,这次的标题有点长。之所以把标题写得这么详细,主要是为了搜索引擎能够准确地把确实需要了解GCC生成位实模式代码方法的朋友带到我的

Linux折腾记(九):在Ubuntu 中使用搜狗拼音输入法 在UbuntuKylin系统中,默认安装搜狗拼音输入法,但是在原生Ubuntu系统中则不是。这可以理解,毕竟搜狗输入法的Linux版有Kylin团队的不小功劳。由于搜狗

标签: ,linux

本文链接地址:https://www.jiuchutong.com/biancheng/368240.html 转载请保留说明!

上一篇:Linux折腾记(五):在Ubuntu 14.10中使用Windows字体(linux 746)

下一篇:Linux折腾记(七):硬盘GPT分区和MBR分区爬坑记(linux k)

  • 公车私车税收区别
  • 红冲上个月发票
  • 小规模纳税人每个季度超过了30万怎么办
  • 房地产企业还有未来吗
  • 外商投资企业采取发包、出租经营
  • 房地产企业拆迁安置土地增值税清算疑点
  • 应付企业负担的职工社会保险费
  • 收资料费怎么入账科目
  • 暂估的成本跨年了怎么冲销后要调整报表吗
  • 农产品加计扣除政策2023最新
  • 收到去年所得税汇算清缴退税账务处理
  • 税交多了退税走到哪一流程了问谁
  • 2月的发票入在1月可以吗
  • 固定资产报废税费怎么处理
  • 厂房租赁税率是房东交的吗
  • 收到福利费专票需要认证吗
  • 发票查询结果显示不出来
  • 定额发票2019
  • 银行异地汇款抽奖需要扣缴个人所得税吗?
  • 哪些房产免纳房产税
  • 固定资产取得方式A04代号
  • 财务软件操作指南
  • 汇算清缴申报后怎样撤销
  • 公司注销留抵税金能退税吗
  • 冲失业保险分录怎么做
  • 利润分配在利润表中如何列示
  • 房子转租扣钱吗
  • 存货折扣怎样做账
  • mac的键盘怎么打开
  • 如何修改系统时间为12/24小时制
  • windows10版本2004怎么样
  • 物资销售经营范围
  • layui表格编辑功能
  • 如何收取会员会费才合法
  • php 动态执行代码
  • vue传值inject
  • 叮咚你有一份好运请查收图片
  • uniapp零基础小白到项目实战
  • 基于车联网
  • opencv制作训练数据集
  • 应交税费中应交企业所得税为负数怎么调报表
  • 为博客园开发了什么项目
  • 购买小礼品赠送怎么做账
  • 企业合并属于非企业吗
  • 金蝶系统结账之后反结账
  • 主管盾和制单盾是是转账使用的U盾吗?
  • sql server怎么升级版本
  • 企业弥补以前年度亏损要怎么填表
  • 固定资产一次性计入费用的账务处理
  • 建筑业增值税税负率
  • 住宿费发票可以报销吗
  • 一般纳税人购买汽车会计分录
  • 税前扣除的职工福利费怎么算
  • 关于新修订的未保法
  • 发票可以冲销几次
  • 如何跨数据库查询
  • hyper-v怎么样
  • 如何在windows中对硬盘进行分区
  • linux常用小技巧
  • windows10已经阻止此软件
  • qq2440启动linux后插入u盘出现usb 1-1: device descriptor read/64, error -110,usb 1
  • ubuntu和windows文件互传
  • win7系统检测不到u盘
  • win7远程桌面连接怎么设置
  • win8系统怎么设置桌面
  • cocos2dx4.0入门
  • opengl es programming guide
  • jQuery插件封装时如要实现链式编程,需要
  • shell脚本编写 方法
  • 基于JAVASCRIPT实现的可视化工具是
  • nodejs查找文件
  • 数据库的列名是什么
  • python3.9快捷键
  • android自定义view的三大流程
  • 类库怎么用
  • android获取sha1
  • 国家税务总局税法
  • 预约怎么缴费
  • 税控盘怎么分发
  • 信用评级机构有几家
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设