UNIX系统实现定义了很多幻数和常量,其中有很多已被硬编码进程序中,或用特定的技术确定。由于大量标准化工作的努力,已有若干种可移植的方法用以确定这些幻数和实现定义的限制。这非常有助于软件的可移值性。
以下两种类型的限制是必需的:
(1)编译时限制(例如:智整型的最大值是什么 ?)
(2)运行时限制(例如:文件名可以有多少个字符?)
编译时限制可在头文件中定义,程序在编译时可以包含这些头文件。但是,运行时限制则要求进程调用一个函数以获得些种限制值。
由于各种unix系统的限制不尽相同,我们需要获得这些限制。方式有三种:
(1)编 译时限制(头文件)
(2)不与文件或目录相关联的运行时限制(sysconf函数)
(3)与文件或目录相关联的运行时限制(pathconf和fpathconf函数)
使事情变得复杂的是,如果一个特定的运行时限制在一个给定的系统并不改变,则可将其静态地定义在一个头文件中。但是,如果没有将其定义在头文件中,则应用程序就必须调用三个conf 函数中的一个,以确定其运行时的值。
头文件:
#include<unistd.h>
函数原型:
long sysconf(int name);
long pathconf(const char *pathname, int name);
long fpathname(int filedes, int name);
我们需要更详细地说明这三个函数的不同返回值。(1)如果name不是一个合适的常量,则所有这三个函数都会返回-1,并将errno设置为EINVAL。
(2)有些nam可以返回变量的值(返回值>=0),或者返回-1,这表示该值是不确定的,此时并不改变errno的值。
demo :
#include <errno.h>
#include <unistd.h> /*for sysconf pathconf _SC_OPEN_MAX*/
#include <stdio.h> /*for printf*/
#include <stdarg.h> /*for va_list*/
#include <string.h> /*for strcat strlen */
#include <stdlib.h> /*for strcat exit */
#define MAXLINE 4096 /* max line length */
static void pr_sysconf(char *, int);
static void pr_pathconf(char *, char *, int);
void err_quit(const char *fmt, ...);
void err_sys(const char *fmt, ...);
void err_doit(int errnoflag, int error, const char *fmt, va_list ap);
int main(int argc, char *argv[])
{
if (argc != 2)
err_quit("usage: a.out <dirname>");
pr_sysconf("ARG_MAX =", _SC_ARG_MAX);
pr_sysconf("CHILD_MAX =", _SC_CHILD_MAX);
pr_sysconf("clock ticks/second =", _SC_CLK_TCK);
pr_sysconf("NGROUPS_MAX =", _SC_NGROUPS_MAX);
pr_sysconf("OPEN_MAX =", _SC_OPEN_MAX);
#ifdef _SC_STREAM_MAX
pr_sysconf("STREAM_MAX =", _SC_STREAM_MAX);
#endif
#ifdef _SC_TZNAME_MAX
pr_sysconf("TZNAME_MAX =", _SC_TZNAME_MAX);
#endif
pr_sysconf("_POSIX_JOB_CONTROL =", _SC_JOB_CONTROL);
pr_sysconf("_POSIX_SAVED_IDS =", _SC_SAVED_IDS);
pr_sysconf("_POSIX_VERSION =", _SC_VERSION);
pr_pathconf("MAX_CANON =", "/dev/tty", _PC_MAX_CANON);
pr_pathconf("MAX_INPUT =", "/dev/tty", _PC_MAX_INPUT);
pr_pathconf("_POSIX_VDISABLE =", "/dev/tty", _PC_VDISABLE);
pr_pathconf("LINK_MAX =", argv[1], _PC_LINK_MAX);
pr_pathconf("NAME_MAX =", argv[1], _PC_NAME_MAX);
pr_pathconf("PATH_MAX =", argv[1], _PC_PATH_MAX);
pr_pathconf("PIPE_BUF =", argv[1], _PC_PIPE_BUF);
pr_pathconf("_POSIX_NO_TRUNC =", argv[1], _PC_NO_TRUNC);
pr_pathconf("_POSIX_CHOWN_RESTRICTED =",
argv[1], _PC_CHOWN_RESTRICTED);
return 0;
}
static void
pr_sysconf(char *mesg, int name)
{
long val;
fputs(mesg, stdout);//输出第一个传数的字符串
errno = 0;
if ( (val = sysconf(name)) < 0) {//name不是一个合适的常量,name不存在
if (errno != 0)
err_sys("sysconf error");
fputs(" (not defined)\n", stdout);
} else
printf(" %ld\n", val);
}
static void
pr_pathconf(char *mesg, char *path, int name)
{
long val;
fputs(mesg, stdout);
errno = 0;
if ( (val = pathconf(path, name)) < 0) {
if (errno != 0)
err_sys("pathconf error, path = %s", path);
fputs(" (no limit)\n", stdout);
} else
printf(" %ld\n", val);
}
void err_quit(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(0, 0, fmt, ap);
va_end(ap);
exit(1);
}
/*
* Print a message and return to caller.
* Caller specifies "errnoflag".
*/
void err_doit(int errnoflag, int error, const char *fmt, va_list ap)
{
char buf[MAXLINE];
vsnprintf(buf, MAXLINE, fmt, ap);//把格式化的输出字符串生成
if (errnoflag)
snprintf(buf+strlen(buf), MAXLINE-strlen(buf), ": %s",strerror(error));//把错误翻译成描述信息
strcat(buf, "\n");//加一个换行符
fflush(stdout);//冲洗标准输出的缓存区
fputs(buf, stderr);//把错误信息输出到标准错误流
fflush(NULL);//冲洗所有输出流
}
/*
* Fatal error related to a system call.
* Print a message and terminate.
*/
void err_sys(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, errno, fmt, ap);
va_end(ap);
exit(1);
}
输出: