资深C++程序员都不会对C++编程规范太陌生,C++实在太复杂,以至于所有项目都需要裁剪一个子集共项目组内使用。经过在家休息这一小段时间,我发现其实C语言更需要一个相同的规范,这就是本文的目标,最大可能规避C语言的黑暗面。

这里说的“不支持、不使用”,是指在没有明确要求的程序中,不主动使用。但在特殊场景下(如调用外部接口等),有些黑暗的角落我们还是要去了解。

大体上分成这几个部分,这些有些还是只是一个初步的想法,逐步会进一步扩充和裁剪。

1.语言标准和编译器

如果我说出建议使用C99和GCC是不是有很多人长叹一口气,不过C99事实只用到了很少一点特性,大部分编译器都支持,比如说//注释。选择GCC是因为它广范的可获得性和一致性,而主要以GCC做为一个验证标准来看待。

2.文件格式

这部分只要求两点,一是包含基准头文件,stdc99.h。C语言的头文件实在不标准。二是使用前注释,而不是行注释和后注释。

3.数据类型选择

只选用char, unsigned char, int 这几个基本类型,64位系统程序中应该还要增加一个long long 。

不使用typedef的结构体和指针,即结构体都有前缀struct 。

不使用共用体。

不使用enum类型,但使用enum来定义常量。

数组类型,需要要特别说明,不强制使用。

只在有明确要求的地方使用double类型,小心不精确的表示。

4.函数使用

一般不使用malloc,以静态分配为主,必须进行动态内存管理时,必须使用这四个一组的函数完成。

xxxalloc/xxxsize/xxxfree/xxxextend

不使用scanf/printf这种内嵌的语言。

不使用(...)式的可变参数,除非可变参数是同一类型。

5.表达式

接受使用+,-,*,/

除提领‘*’外,不接受++与其它运算符同时出现。

不使用&,|,^,~,<<,>>,见后面“位操作”

不使用?:

不使用","逗号表达式,但接受在声明和函数参数中使用。

6.位操作

由于位操作在有符号问题上操作复杂,推荐使用以下函数形式的宏,完成位操作。u/s分别表示unsigned和signed

ushfleft,ushfright,uset,uclear,uisset,umask

sshfleft,sshfright,sset,sclear,sisset,smask

7.语句

不使用do ... while/switch...case,推荐使用if else/while。

对于数组循环处理,推荐使用for (int i = 0; i < n; ++i) 这个标准结构,必要的索引通过第二个变量计算出来。

8.数组

使用数组+长度两个变量来表示一个数组。

除首地址外,建议其它元素地址使用&a[i]结构来取得。

数组循环处理使用标准结构,例如,奇数索引的元素赋值为1:

for (int i = 0; i < n; ++i) {

a[2*i + 1] = 1;

}

9.抽象数据

进程有唯一性的抽象(全局状态),定义为模块。

通过void foo_init(void)或void foo_init(int argc, char** args)这样的函数初始化。

通过void foo_close()这样的函数关闭,通常这些调用会在main函数中完成。

一般在头文件中声明为:

struct foo;

struct foo* foo_new();

void foo_dosomething(struct foo* o, void* error);

在C文中定义访问struct foo和访问内部成员的函数

#include "namespace/foo.h"

struct foo {

  成员

};

10.动态抽象数据

头文件中增加声明

struct foo_ops {

  void (*op1)(void);

void (*op2)(void);

};

struct foo* foo_new(struct foo_ops* ops, void* data);

对应的C文中,在struct foo中要增加定义

struct foo {

  struct foo_ops *ops;

  void* data;

  其它成员

};

11.错误处理机制

主机制可以通过setjmp/longjmp这样的全局状态完成,这应由main函数完成。

程序库内部应提供非侵入式的错误处理机制,例如返回错误状态,惯例如下:

int parse_options(int argc, char** args, struct error* err);

错误状态保存在err中,如果不关心错误,可以通过传入0来忽略,但函数本身应把状态记录在全局的错误对象上。 在前一个错误未清除的情况,函数应没有动作,也不修改全局错误对象。输入了错误指针时,把全局对象复制给错误指针。

12.其它杂项和风格问题

不应忽略{},总是使用{}来表明范围。

使用//注释,C99支持

使用按需声明变量,C99支持

Minus-C 一个最小化的C语言规范的更多相关文章

  1. 实现iOS图片等资源文件的热更新化(四): 一个最小化的补丁更新逻辑

    简介 以前写过一个补丁更新的文章,此处会做一个更精简的最小化实现,以便于集成.为了使逻辑具有通用性,将剥离对AFNetworking和ReativeCocoa的依赖.原来的文章,可以先看这里: htt ...

  2. 关于吴恩达机器学习支持向量机的问题,讲到对偶前有一个最小化f(w)使用拉格朗日求解时转化成一个最大的相等式的理解和一些困惑

    (纯属个人理解) 参考: https://www.zhihu.com/question/267482928 https://www.cnblogs.com/90zeng/p/Lagrange_dual ...

  3. 一个最小化的SpringBoot项目

    项目结构 项目基于Maven管理,注意使用了父pom <parent> <groupId>org.springframework.boot</groupId> &l ...

  4. 分享我自己的一个最小化安装CentOS6的初始化脚本

    在自己的虚拟机上使用的基于CentOS6的系统初始化脚本 #!/bin/bash # #Filename:system_init.sh #Description:系统安装完成后,对系统进行一些配置,以 ...

  5. Effective Java 第三版——15. 使类和成员的可访问性最小化

    Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...

  6. 手把手带你安装最小化suse 12 linux

    当然. 你需要现有一个vmware,还要有一个suse的镜像,suse镜像可以直接去官网获取,只需要注册一个suse的账号就可以了,官网下载会有点慢,可以使用迅雷下载 F2 可以设置安装界面的语言,可 ...

  7. C# 隐藏最大化、最小化和关闭三个按钮

    在Windows的窗体编程中,基本上每一个窗体都是一个最小化.最大化和关闭按钮的. 一.禁用最大化和最小化 对于最大化和最小化按钮,在C#窗体开发时,各一个属性来启用或禁用这两个按钮. this.Ma ...

  8. Monthly Expense(二分--最小化最大值)

    Farmer John is an astounding accounting wizard and has realized he might run out of money to run the ...

  9. centos6最小化安装默认没有 NetworkManager服务

    转载Centos6最小化安装中设置网卡默认启动   Centos 6.0版本提供了一个"最小化"(Minimal)安装的选项.这是一个非常好的改进,因为系统中再也不会存在那些不必要 ...

随机推荐

  1. XE7 - 程序图标及启动画面图片的注意事项

    还是继续昨晚写的,年前已经解决了这个问题,现在补记下.启动画面失真是本篇笔记的重点.搜索了很多文章,基本上大同小异,几乎都没怎么提及启动画面失真的问题.不知道是不是我的操作不对头,. Project ...

  2. 【英语】Bingo口语笔记(69) - 脏话的表达

  3. SOAP+WSDL

    一 SOAPSOAP最开始是用作RPC机制的,后来XML的出现使其应用非常广泛.它与HTTP一样是一种应用级协议,使用他可以在不同的应用程序之间进行数据交换.SOAP可以基于HTTP,也可以基于HTT ...

  4. acess() 判断目录是否存在

    acess()功能描述: 检查调用进程是否可以对指定的文件执行某种操作. <pre lang="c" escaped="true">#include ...

  5. 微信 ua

    Mozilla/5.0 (Linux; U; Android 2.3.6; zh-cn; GT-S5660 Build/GINGERBREAD) AppleWebKit/533.1 (KHTML, l ...

  6. order by多个字段对索引的影响

    某前台sql语句,简化后如下SELECT products_name,products_viewed FROM `products_description` ORDER BY products_vie ...

  7. Java反射基本玩法

    三个主要的反射类 Class反射对象描述类语义结构,可以从Class对象中获取构造函数.成员变量.方法类等元素的反射对象,并以编程的方式通过这些反射对象对目标类对象进行操作.这些反射对象类在java. ...

  8. hadoop——配置eclipse下的map-reduce运行环境 1

    1.通过修改实例模板程序来实现自己的map-reduce: 为了让示例程序run起来: 1)安装eclipse 2)安装map-reduce的eclipse插件 eclipse的map-reduce插 ...

  9. RIA+REST架构实现完美WEB开发

    记得第一次看到REST的身影,是在InfoQ上的一篇介绍,随后又翻阅了后面的参考文章和Developerwork上一些资料,甚至随手翻了翻Roy博士的论文.所幸,在不少人还在体会REST到底是何方神圣 ...

  10. 在Ubuntu中用root帐号登录

    一.其实我个人认为这没有多大必要,因为当你需要 root 的权限时,使用 sudo 便可以了.如果你实在需要在 Ubuntu 中启用 root 帐号的话,那么不妨执行下面的操作: 1.重新设置 roo ...