unix环境高级编程中的err_quit,err_sys

环境

os CentOS release 6.7 (Final)

gcc 4.4.7

c语言预备知识

标准输入输出文件

在linux系统中一切设备皆文件.文件在C语言中用指针来标识.C语言定义的3个标准文件如下所示

标准文件 文件指针 对应设备
标准输入 stdin 键盘
标准输出 stdout 显示器
标准错误 stderr 显示器

下面看2个与标准输入输出有关的函数

  • char *gets(char *s) 从标准输入stdin中读取一行到s指向的缓冲区,当遇到行结束符或者EOF时读取结束.
  • int puts(const char *s) 把s和换行符写入stdout

    看下面的代码
#include <stdio.h>
#include <string.h>
int main( ) {
char str[100];
printf( "Enter a value :");
gets(str);
printf("%zu", strlen(str)); // %zu用于输出size_t,str不包括最后的回车,所以输入abc回车,这里输出为3
printf( "\nYou entered: ");
puts(str); //当输入为abc时,这里不但输出abc还多输入一个回车.
printf("aaa");
return 0;
}

再看另外一组与标准输入输出有关的函数:

  • int scanf(const char *format, ...) 从标准输入stdin中按读取内容,
  • int printf(const char *format, ...) 把内容按format指定的格式写入准备输出

flush

看下面的代码

#include <stdio.h>
#include <string.h>
#include <unistd.h> int main() {
char buff[1024];
memset(buff, '\0', sizeof(buff)); fprintf(stdout, "Going to set full buffering on\n");
setvbuf(stdout, buff, _IOFBF, 1024); fprintf(stdout, "This is tutorialspoint.com\n");
fprintf(stdout, "This output will go into buff\n");
fflush(stdout); fprintf(stdout, "and this will appear when programm\n");
fprintf(stdout, "will come after sleeping 5 seconds\n"); sleep(5); return(0);
}

setvbuf的作用是当缓冲区buff满的时候才会写出到输出流.一直到fprintf(stdout, "This output will go into buff\n");这句话缓冲区还没有满,如果程序没有结束或者没有fflush,在显示器上可能看不到前3句的输了(或者输出不完整).调用fflush则一定能看到前3句完整的输出.最后2句话后面没有fflush函数,则不会立刻看到输出.当sleep(5)结束的时候,就能看到最后2句话输出了.

函数变长参数问题

当传入参数个数不确定时可以用变长参数.不确定的参数用...表示.最常用的变长参数函数应该就是scanf和printf了.获取变长参数的参要用到C语言定义的宏,va_start,va_arg,and va_end.这些宏在头文件stdarg.h中定义.va_start初始化变长参数的列表,va_arg在变长参数的列表中区获取下一个参数,va_end清空变长参数的列表.看下面的代码如何求和

#include <stdio.h>
#include <stdlib.h>
#include <stdarg.h>
#define END -1
int va_sum (int first_num, ...) {
va_list ap;
va_start(ap, first_num); //first_num后面的参数是变长参数
int result = first_num;
int temp = 0;
while ((temp = va_arg(ap, int)) != END) {
result += temp;
}
va_end(ap);
return result;
}
int main () {
int sum_val = va_sum(1, 2, 3, 4, 5, END);
printf ("%d", sum_val);
return 0;
}

再看一个求平均值的代码:

#include <stdarg.h>
#include <stdio.h> double average (int num, ...) {
va_list arguments;
double sum = 0;
va_start(arguments, num); //num后面的参数是变长参数
for (int x = 0; x < num; x++ ) {
sum += va_arg(arguments, double);
}
va_end(arguments);
return sum / num;
} int main() {
printf("%f\n", average(3, 12.2, 22.3, 4.5));
printf("%f\n", average(5, 3.3, 2.2, 1.1, 5.5, 3.3));
}

vsnprintf

int vsnprintf (char * s, size_t n, const char * format, va_list arg )

把va_list以格式format写到s中,前n位是有效的.

  • s 至少要n个字符,否则会有不可遇知的错误.
  • n 指定s的前n位置是有效的,后面的会被忽略.注意指定n时只能看到n-1个字符.最后一位是null
  • format 格式
  • 变长的参数列表

    看下面的例子:
#include <stdarg.h>
#include <stdio.h>
void vout(char *string, char *fmt, ...);
char fmt1 [] = "%s %s %s %s";
int main() {
char buf[100];
vout(buf, fmt1, "Sat", "Sun", "Mon", "aaa");
printf("The string is:%s\n", buf);
return 0;
}
void vout(char *buf, char *fmt, ...) {
va_list arg_ptr;
va_start(arg_ptr, fmt); //让arg_ptr指向变长参数
int n = vsnprintf(buf, 16, fmt, arg_ptr); //变长参数以fmt格式放到buf中,返回写入buf的字节数,但是只有前12个字节会输出.
printf("n=%d\n", n);
va_end(arg_ptr);
}

再看一个例子

#include <stdio.h>
#include <stdarg.h>
int mon_log(char* format, ...) {
char str_tmp[50];
int i=0;
va_list vArgList;
va_start(vArgList, format);
i = vsnprintf(str_tmp, 50, format, vArgList);
printf("%s\n", str_tmp);
va_end(vArgList);
return i;
}
int main() {
int i=mon_log("%s,%d,%d,%d","asd",2, 3, 4);
printf("%d\n", i);
return 0;
}

再看一个例子

#include <stdarg.h>
#include <stdio.h>
void vout(char *string, char *fmt, ...);
char fmt1 [] = "%s %s %s %s";
int main() {
char buf[100];
vout(buf, fmt1, "Sat", "Sun", "Mon", "aaa");
printf("The string is:%s\n", buf);
return 0;
}
void vout(char *buf, char *fmt, ...) {
va_list arg_ptr;
va_start(arg_ptr, fmt); //让arg_ptr指向变长参数
int n = vsnprintf(buf, 16, fmt, arg_ptr); //变长参数以fmt格式放到buf中,返回写入buf的字节数,但是只有前12个字节会输出.
printf("n=%d\n", n);
va_end(arg_ptr);
}

snprintf

int snprintf(char * s, size_t n, const char * format, ... );

和vsnprintf的作用差不多.不同是vsnprintf最后一个参数类型是va_list,这里是不定长参数.

看下面代码

#include <stdio.h>
int main() {
char str[10];
int ret = snprintf(str, 5, "%s", "12345678");
printf("%d\n",ret);
printf("%s\n",str);
return 0;
}

参考资料

unix环境高级编程中的err_quit,err_sys用到的知识点的更多相关文章

  1. UNIX环境高级编程--#include "apue.h"

    apue.h头文件为作者自己编写而非系统自带,故需要自行添加! 第一:打开网站 http://www.apuebook.com/第二:选择合适的版本(一共有三个版本,根据书的版本选择)下载源码sour ...

  2. unix环境高级编程附录 B 通用代码

    0.说明: 在测试 unix 环境高级编程中的代码时,需要一些作者事先写好的代码, 如: apue.h 包含某些标准系统头文件,定义许多常量及函数原型 还有两个作者自编的函数来对错误进行处理 1.ep ...

  3. multiple definition of `err_sys' 《UNIX环境高级编程》

    本文地址:http://www.cnblogs.com/yhLinux/p/4079930.html 问题描述: [点击此处直接看解决方案] 在练习<UNIX环境高级编程>APUE程序清单 ...

  4. Unix环境高级编程第三版中实例代码如何在自己的linux上运行的问题

    学习Linux已经有2个月了,最近被期末考试把进度耽误了,前几天把Unix环境高级编程看了两章,感觉对Linux的整体有了一些思路,今天尝试着对第一章涉及到的一个简单的交互式shell编译运行一下,结 ...

  5. apue.h头文件(UNIX环境高级编程)

    在看UNIX环境高级编程是,碰到一个头文件"apue.h",搜一下别人的帖子,其实apue.h是作者自己写的一个文件,包含了常用的头文件,系统不自带.其中包含了常用的头文件,以及出 ...

  6. UNIX环境高级编程(第三版)关于apue.h的用法

    UNIX环境高级编程(第三版)中的例子用到apue.h这个头文件,但是书里面写的地址已经不能访问. 经过一番查找之后,找到如下解决方案: 1.到www.apuebook.com上下载第2版的源码,也可 ...

  7. 关于UNIX/Linux下安装《UNIX环境高级编程》源代码的问题

    <UNIX环境高级编程(第三版)>是一本广为人知的unix系统编程书籍. 但是,书中的代码示例,要想正确的编译运行,要先做好准备工作: 1.下载源代码 传送门:http://apueboo ...

  8. 【UNIX环境高级编程】文件I/O

    [UNIX环境高级编程]文件I/O大多数文件I/O只需要5个函数: open.read.write.lseek以及close 不带缓冲的I/O: 每个read和write都调用内核中的一个系统调用 1 ...

  9. 《unix环境高级编程》学习笔记【原创】

    本文基于unix环境高级编程的学习的笔记,写的比较简如有不对,欢迎指点. 简单的描述下面函数的功能改变ctr+c信号原本的作用终止程序,在按下中断键的时候输出一句话. while循环主要读取用户的输入 ...

随机推荐

  1. 【algo&ds】1.时间复杂度和空间复杂度分析

    1.时间复杂度分析O(f(n)) 分析方法 只关注循环执行次数最多的一段代码 加法原则 乘法原则 高优先级原则 常见时间复杂度量级 多项式量级和非多项式量级.其中,非多项式量级只有两个:O(2^n) ...

  2. GitHub和Git

    GitHub托管项目代码 首先一些基本概念: repository(仓库) 用来存放项目代码,每个项目代表一个仓库,开一个项目就意味着你有一个仓库. star(收藏) 收藏方便下次查找. fork(复 ...

  3. nyoj 125-盗梦空间 (数学ans += temp * 60 * pow(0.05, cnt))

    125-盗梦空间 内存限制:64MB 时间限制:3000ms 特判: No 通过数:8 提交数:10 难度:2 题目描述: <盗梦空间>是一部精彩的影片,在这部电影里,Cobb等人可以进入 ...

  4. python:类1——类和对象基础

    一.OO = Object Oriented 面向对象 OOP面向对象编程.OOA面向对象分析.OOD面向对象设计 二.属性+方法——>类(数据和函数) class Turtle(): #类名约 ...

  5. gitbook怎么操作

    首先我先说一下什么是GitBook,它和Git没半毛钱关系,定义如下: GitBook 是一个基于 Node.js 的命令行工具,支持 Markdown 和 AsciiDoc 两种语法格式,可以输出 ...

  6. [springboot 开发单体web shop] 8. 商品详情&评价展示

    上文回顾 上节 我们实现了根据搜索关键词查询商品列表和根据商品分类查询,并且使用到了mybatis-pagehelper插件,讲解了如何使用插件来帮助我们快速实现分页数据查询.本文我们将继续开发商品详 ...

  7. 元数据管理的重要性 - xms

    什么是元数据?引用百科的描述就是:元数据(Metadata),又称中介数据.中继数据,为描述数据的数据(data about data),主要是描述数据属性(property)的信息: 看起来有点抽象 ...

  8. node.js安装express框架(1)

    一.全局安装express 使用express首先确保你的node.js已经安装好了环境变量配置成功,安装了npm或者cnpm 你可以在终端上面输入node -v查看你的node版本号 打开cmd终端 ...

  9. MySql 表索引设计原则

    索引的优点 1.加快数据的检索速度,这是创建索引的最主要的原因; 2.通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性; 3.加速表和表之间的连接; 4.在使用分组和排序子句进行数据检索时,可 ...

  10. Java基础面试题及答案(三)

    多线程 35. 并行和并发有什么区别? 并行是指两个或者多个事件在同一时刻发生:而并发是指两个或多个事件在同一时间间隔发生. 并行是在不同实体上的多个事件,并发是在同一实体上的多个事件. 在一台处理器 ...