apue读书笔记【四】:系统的限制

        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);
}

输出:

       





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值