上次忘了说,我的学习顺序是按照《Beginning Linux Programming, 4th Edition》(以后简称《BLP》)这本书来的,同时参照官方文档《The GNU C Library Reference Manual》(以后简称《GLIBC》)和 man。这次是第四章《The Linux Environment》的学习笔记。
程序参数
没什么特别,主要是 int getopt(int argc, char** argv, const char* options)
和 int getopt_long(int argc, char* const* argv, const char* shortopts, const struct option* longopts, int* indexptr)
这两个用于解析命令行参数的使用。只要按照规范定义参数,就可以很方便地进行解析。怎么 Java 就没有提供类似的方法呢?
环境变量
要设置一个环境变量,《BLP》只讲了 int putenv(char* string)
函数。根据《GLIBC》和 man 的描述,调用此函数后,string
就变成了环境的一部分,对它所做的任何更改,不论对键还是对值,都将自动反映到环境中,这就要求 string
的生命周期不能在该环境变量被删除之前结束,否则可能出错。例如:
char env[] = "a=b";
putenv(env);
printf("%s\n", getenv("a")); // 输出“b”。
env[2] = 'z';
printf("%s\n", getenv("a")); // 输出“z”。
env[0] = 'x';
// printf("%s\n", getenv("a")); // 程序崩溃……
printf("%s\n", getenv("x")); // 输出“z”。
从《GLIBC》可以查到另两个用起来更为安全可靠的函数: int setenv(const char* name, const char* value, int replace)
和 int unsetenv(const char* name)
。
可通过 extern char** environ
遍历所有的环境变量,但是最好不要直接使用此变量进行迭代,而是创建一个副本,可避免影响程序的其他部分。
时间和日期
《BLP》只讲了精确到秒的 time_t time(time_t* tloc)
。《GLIBC》指出 sys/time.h
中声明了用来获取更高精度的结构体 timeval
和函数 int gettimeofday(struct timeval* tp, struct timezone* tzp)
。timeval
定义了表示整秒数的成员 tv_sec
和表示剩余毫微秒数的成员 tv_usec
,其中 tv_usec
不超过一百万,所以两者加起来就是实际的时间。由于 timezone
结构体已被废弃,而且不再被 GNU 支持,所以 gettimeofday
的第二个参数只能设为 NULL
,否则 errno
会被置为 ENOSYS
(Function not implemented)。例如:
time_t now = time(NULL);
struct timeval tvnow;
gettimeofday(&tvnow, NULL);
printf("%jd\n", (intmax_t) now); // 输出“1304649729”。
printf("%jd.%06ju\n", (intmax_t) tvnow.tv_sec,
(uintmax_t) tvnow.tv_usec); // 输出“1304649729.644344”。
临时文件
最好不要先使用 char* tmpnam(char* s)
和 char* mktemp(char* template)
生成“随机”字符串,再创建相应文件,因为在创建文件前,有可能其他程序也生成了相同的字符串,从而导致冲突。应当使用 FILE* tmpfile(void)
或 int mkstemp(char* template)
直接生成文件——前者的好处所创建的临时文件在关闭时会被自动删除,但却无法取得文件名;后者能得到文件名,但必须手动删除。
用户信息
在我的 Debian 上,char* getlogin(void)
不能正确工作,errno
给出的原因是“No such file or directory”。具体哪个文件没找到没搞清楚,不过 man 已经指出此函数不安全,不用为妙。通过 uid_t getuid(void)
和 struct passwd* getpwuid(uid_t uid)
可以得到当前用户的详细信息。
主机信息
掌握 int uname(struct utsname* name)
就差不多了。
日志记录
遗憾的是,Linux 本身只提供了写入系统日志的函数,而没有类似 java.util.Logger
那种通用接口。
资源和限制
除了偶尔测试一下算法的性能,也许只有专业性能分析工具才用得上这些东西吧。