进程操作篇
atexit
设置程序正常结束前调用的函数
头文件
#include <stdlib.h>
函数原型
int atexit (void (*function)(void));
说明:atexit()用来设置一个程序正常结束前调用的函数。当程序通过调用exit()或从main中返回时,参数function所指定的函数会先被调用,然后才真正由exit()结束程序。
返回值:如果执行成功则返回0,否则返回-1,失败原因存于errno中。
附加说明:
相关函数:_exit, exit, on_exit
示例
#include <stdlib.h>
void my_exit(void)
{
printf("before exit() !\n");
}
int main()
{
atexit(my_exit);
exit(0);
}
执行
before exit()!
execl
执行文件
头文件
#include <unistd.h>
函数原型
int execl(const char * path, const char * arg, ...);
说明:execl()用来执行参数path字符串所代表的文件路径,接下来的参数代表执行该文件时传递过去的argv(0)、argv[1]……,最后一个参数必须用空指针(NULL)作结束。
返回值:如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno中。
附加说明:
相关函数:fork,execle,execlp,execv,execve,execvp
示例
#include <unistd.h>
int main()
{
execl("/bin/ls", "ls", "-al", "/etc/passwd", (char * )0);
return 0;
}
执行
# 执行/bin/ls -al /etc/passwd
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd
execlp
从 PATH 环境变量中查找文件并执行
头文件
#include <unistd.h>
函数原型
int execlp(const char * file, const char * arg, ...);
说明:execlp()会从PATH 环境变量所指的目录中查找符合参数file的文件名,找到后便执行该文件,然后将第二个以后的参数当做该文件的argv[0]、argv[1]……,最后一个参数必须用空指针(NULL)作结束。
返回值:如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
错误代码:参考 execve()。
附加说明:
相关函数:fork,execl,execle,execv,execve,execvp
示例
/* 执行ls -al /etc/passwd execlp()会依PATH 变量中的/bin找到/bin/ls */
#include <unistd.h>
int main()
{
execlp("ls", "ls", "-al", "/etc/passwd", (char *)0);
return 0;
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd
execv
执行文件
头文件
#include <unistd.h>
函数原型
int execv (const char * path, char * const argv[ ]);
说明:execv()用来执行参数path字符串所代表的文件路径,与execl()不同的地方在于execve()只需两个参数,第二个参数利用数组指针来传递给执行文件。
返回值:如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
错误代码:请参考 execve()。
附加说明:
相关函数:fork,execl,execle,execlp,execve,execvp
示例
/* 执行/bin/ls -al /etc/passwd */
#include <unistd.h>
int main()
{
char * argv[ ] = {"ls", "-al", "/etc/passwd", (char *)0};
execv("/bin/ls", argv);
return 0;
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd
execve
执行文件
头文件
#include <unistd.h>
函数原型
int execve(const char * filename, char * const argv[], char * const envp[]);
说明:execve()用来执行参数filename字符串所代表的文件路径,第二个参数系利用数组指针来传递给执行文件,最后一个参数则为传递给执行文件的新环境变量数组。
返回值:如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno 中。
错误代码:
EACCES
欲执行的文件不具有用户可执行的权限。欲执行的文件所属的文件系统是
以 noexec 方式挂上。
欲执行的文件或 script 解析器非一般文件。
EPERM
进程处于被追踪模式,执行者并不具有root权限,欲执行的文件具有SUID 或SGID 位。
欲执行的文件所属的文件系统是以nosuid方式挂上,欲执行的文件具有SUID 或SGID 位元,但执行者并不具有root权限。
E2BIG
参数数组过大ENOEXEC
无法判断欲执行文件的执行文件格式,有可能是格式错误或无法在此平台执行。EFAULT
参数filename所指的字符串地址超出可存取空间范围。ENAMETOOLONG
参数filename所指的字符串太长。ENOENT
参数filename字符串所指定的文件不存在。ENOMEM
核心内存不足ENOTDIR
参数filename字符串所包含的目录路径并非有效目录EACCES
参数filename字符串所包含的目录路径无法存取,权限不足ELOOP
过多的符号连接ETXTBUSY
欲执行的文件已被其他进程打开而且正把数据写入该文件中EIO
I/O 存取错误ENFILE
已达到系统所允许的打开文件总数。EMFILE
已达到系统所允许单一进程所能打开的文件总数。EINVAL
欲执行文件的ELF执行格式不只一个PT_INTERP节区EISDIR
ELF翻译器为一目录ELIBBAD
ELF翻译器有问题。
附加说明:
相关函数:fork,execl,execle,execlp,execv,execvp
示例
#include <unistd.h>
int main()
{
char * argv[] = {"ls", "-al", "/etc/passwd", (char *)0};
char * envp[] = {"PATH=/bin", 0};
execve("/bin/ls", argv, envp);
return 0;
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd
execvp
执行文件
头文件
#include <unistd.h>
函数原型
int execvp(const char *file, char * const argv []);
说明:execvp()会从PATH 环境变量所指的目录中查找符合参数file 的文件名,找到后便执行该文件,然后将第二个参数argv传给该欲执行的文件。
返回值:如果执行成功则函数不会返回,执行失败则直接返回-1,失败原因存于errno中。
错误代码:请参考 execve()。
附加说明:
相关函数:fork,execl,execle,execlp,execv,execve
示例
/*请与execlp()范例对照*/
#include <unistd.h>
int main()
{
char * argv[] = {"ls", "-al", "/etc/passwd", 0};
execvp("ls", argv);
return 0;
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd
exit
正常结束进程
头文件
#include <stdlib.h>
函数原型
void exit(int status);
说明:exit()用来正常终结目前进程的执行,并把参数status返回给父进程,而进程所有的缓冲区数据会自动写回并关闭未关闭的文件。
返回值:无
附加说明:
相关函数:_exit,atexit,on_exit
示例
参考 wait()。
_exit
结束进程执行
头文件
#include <unistd.h>
函数原型
void _exit(int status);
说明:_exit()用来立刻结束目前进程的执行,并把参数status返回给父进程,并关闭未关闭的文件。此函数调用后不会返回,并且会传递SIGCHLD信号给父进程,父进程可以由wait函数取得子进程结束状态。
返回值:无
附加说明:_exit() 不会处理标准 I/O 缓冲区,如要更新缓冲区请使用 exit()。
相关函数:exit,wait,abort
示例
执行
vfork
建立一个新的进程
头文件
#include <unistd.h>
函数原型
pid_t vfork(void);
说明:vfork()会产生一个新的子进程,其子进程会复制父进程的数据与堆栈空间,并继承父进程的用户代码,组代码,环境变量、已打开的文件代码、工作目录和资源限制等。Linux 使用copy-on-write(COW)技术,只有当其中一进程试图修改欲复制的空间时才会做真正的复制动作,由于这些继承的信息是复制而来,并非指相同的内存空间,因此子进程对这些变量的修改和父进程并不会同步。此外,子进程不会继承父进程的文件锁定和未处理的信号。注意,Linux不保证子进程会比父进程先执行或晚执行,因此编写程序时要留意死锁或竞争条件的发生。
返回值:如果vfork()成功则在父进程会返回新建立的子进程代码(PID),而在新建立的子进程中则返回0。如果vfork 失败则直接返回-1,失败原因存于errno中。
错误代码:EAGAIN 内存不足。ENOMEM 内存不足,无法配置核心所需的数据结构空间。
附加说明:
相关函数:wait,execve
示例
#include <unistd.h>
int main()
{
if(vfork() == 0)
{
printf("This is the child process\n");
}
else {
printf("This is the parent process\n");
}
return 0;
}
执行
this is the parent process
this is the child process
getpgid
取得进程组识别码
头文件
#include <unistd.h>
函数原型
pid_t getpgid(pid_t pid);
说明:getpgid()用来取得参数pid 指定进程所属的组识别码。如果参数pid为0,则会取得目前进程的组识别码。
返回值:执行成功则返回组识别码,如果有错误则返回-1,错误原因存于errno中。
错误代码:ESRCH 找不到符合参数pid 指定的进程。
附加说明:
相关函数:setpgid,setpgrp,getpgrp
示例
/*取得init 进程(pid=1)的组识别码*/
#include <unistd.h>
int main()
{
printf("init gid = %d\n", getpgid(1));
return 0;
}
执行
init gid = 0
getpgrp
取得进程组识别码
头文件
#include <unistd.h>
函数原型
pid_t getpgrp(void);
说明:getpgrp()用来取得目前进程所属的组识别码。此函数相当于调用getpgid(0);
返回值:返回目前进程所属的组识别码。
附加说明:
相关函数:setpgid,getpgid,getpgrp
示例
#include <unistd.h>
int main()
{
printf("my gid = %d\n", getpgrp());
return 0;
}
执行
my gid = 29546
getpid
取得进程识别码
头文件
#include <unistd.h>
函数原型
pid_t getpid(void);
说明:getpid() 用来取得目前进程的进程识别码,许多程序利用取到的此值来建立临时文件,以避免临时文件相同带来的问题。
返回值:目前进程的进程识别码。
附加说明:
相关函数:fork,kill,getpid
示例
#include <unistd.h>
int main()
{
printf("pid = %d\n", getpid());
return 0;
}
执行
pid = 1494 # 每次执行结果都不一定相同
getppid
取得父进程的进程识别码
头文件
#include <unistd.h>
函数原型
pid_t getppid(void);
说明:getppid()用来取得目前进程的父进程识别码。
返回值:目前进程的父进程识别码。
附加说明:
相关函数:fork,kill,getpid
示例
#include <unistd.h>
int main()
{
printf("My parent ‘pid = %d\n", getppid());
return 0;
}
执行
My parent pid = 463
getpriority
取得程序进程执行优先权
头文件
#include <sys/time.h>
#include <sys/resource.h>
函数原型
int getpriority(int which, int who);
说明:getpriority() 可用来取得进程、进程组和用户的进程执行优先权。which有三种数值,参数who 则依which值有不同定义。
which who 代表的意义:
PRIO_PROCESS
who 为进程识别码PRIO_PGRP
who 为进程的组识别码PRIO_USER
who 为用户识别码
此函数返回的数值介于-20 至20之间,代表进程执行优先权,数值越低代表有较高的优先次序,执行会较频繁。
返回值:返回进程执行优先权,如有错误发生返回值则为-1 且错误原因存于errno。
错误代码:
ESRCH
参数which或who 可能有错,而找不到符合的进程。EINVAL
参数which 值错误。
附加说明:由于返回值有可能是-1,因此要同时检查errno是否存有错误原因。最好在调用次函数前先清除errno变量。
相关函数:setpriority,nice
示例
执行
nice
改变进程优先顺序
头文件
#include <unistd.h>
函数原型
int nice(int inc);
说明:nice()用来改变进程的进程执行优先顺序。参数inc数值越大则优先顺序排在越后面,即表示进程执行会越慢。只有超级用户才能使用负的inc 值,代表优先顺序排在前面,进程执行会较快。
返回值:如果执行成功则返回0,否则返回-1,失败原因存于errno中。
错误代码:EPERM 一般用户企图转用负的参数inc值改变进程优先顺序。
附加说明:
相关函数:setpriority,getpriority
示例
执行
on_exit
设置程序正常结束前调用的函数
头文件
#include <stdlib.h>
函数原型
int on_exit(void (* function)(int, void*), void *arg);
说明:on_exit()用来设置一个程序正常结束前调用的函数。当程序通过调用exit()或从main中返回时,参数function所指定的函数会先被调用,然后才真正由exit()结束程序。参数arg指针会传给参数function函数,详细情况请见范例。
返回值:如果执行成功则返回0,否则返回-1,失败原因存于errno中。
附加说明:
相关函数:_exit,atexit,exit
示例
#include <stdlib.h>
void my_exit(int status, void *arg)
{
printf("before exit()!\n");
printf("exit (%d)\n", status);
printf("arg = %s\n", (char*)arg);
}
int main()
{
char * str = "test";
on_exit(my_exit, (void *)str);
exit(1234);
}
执行
before exit()!
exit (1234)
arg = test
setpgid
设置进程组识别码
头文件
#include <unistd.h>
函数原型
int setpgid(pid_t pid, pid_t pgid);
说明:setpgid()将参数pid 指定进程所属的组识别码设为参数pgid 指定的组识别码。如果参数pid 为0,则会用来设置目前进程的组识别码,如果参数pgid为0,则会以目前进程的进程识别码来取代。
返回值:执行成功则返回组识别码,如果有错误则返回-1,错误原因存于errno中。
错误代码:
EINVAL
参数pgid小于0。EPERM
进程权限不足,无法完成调用。ESRCH
找不到符合参数pid指定的进程。
附加说明:
相关函数:getpgid,setpgrp,getpgrp
示例
执行
setpgrp
设置进程组识别码
头文件
#include <unistd.h>
函数原型
int setpgrp(void);
说明:setpgrp()将目前进程所属的组识别码设为目前进程的进程识别码。此函数相当于调用setpgid(0,0)。
返回值:执行成功则返回组识别码,如果有错误则返回-1,错误原因存于errno中。
附加说明:
相关函数:getpgid,setpgid,getpgrp
示例
执行
setpriority
设置程序进程执行优先权
头文件
#include <sys/time.h>
#include <sys/resource.h>
函数原型
int setpriority(int which, int who, int prio);
说明:setpriority()可用来设置进程、进程组和用户的进程执行优先权。参数which有三种数值,参数who 则依which值有不同定义 which who 代表的意义:
PRIO_PROCESS
who为进程识别码PRIO_PGRP
who 为进程的组识别码PRIO_USER
who为用户识别码
参数 prio 介于-20 至20 之间。代表进程执行优先权,数值越低代表有较高的优先次序,执行会较频繁。此优先权默认是0,而只有超级用户(root)允许降低此值。
返回值:执行成功则返回0,如果有错误发生返回值则为-1,错误原因存于errno。
错误代码:
ESRCH
参数 which 或 who 可能有错,而找不到符合的进程。EINVAL
参数 which 值错误。EPERM
权限不够,无法完成设置。EACCES
一般用户无法降低优先权。
附加说明:
相关函数:getpriority,nice
示例
执行
system
执行 shell 命令
头文件
#include <stdlib.h>
函数原型
int system(const char * string);
说明:system()会调用fork()产生子进程,由子进程来调用/bin/sh-c string来执行参数string字符串所代表的命令,此命令执行完后随即返回原调用的进程。在调用system()期间SIGCHLD 信号会被暂时搁置,SIGINT和SIGQUIT 信号则会被忽略。
返回值:如果system()在调用/bin/sh时失败则返回127,其他失败原因返回-1。若参数string为空指针(NULL),则返回非零值。如果system()调用成功则最后会返回执行shell命令后的返回值,但是此返回值也有可能为system()调用/bin/sh失败所返回的127,因此最好能再检查errno 来确认执行成功。
附加说明:在编写具有SUID/SGID权限的程序时请勿使用system(),system()会继承环境变量,通过环境变量可能会造成系统安全的问题。
相关函数:fork,execve,waitpid,popen
示例
#include <stdlib.h>
int main()
{
system("ls -al /etc/passwd /etc/shadow");
return 0;
}
执行
-rw-r--r-- 1 root root 705 Sep 3 13 :52 /etc/passwd
-r-------- 1 root root 572 Sep 2 15 :34 /etc/shadow
wait
等待子进程中断或结束
头文件
#include <sys/types.h>
#include <sys/wait.h>
函数原型
pid_t wait(int * status);
说明:wait()会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用wait()时子进程已经结束,则wait()会立即返回子进程结束状态值。子进程的结束状态值会由参数status 返回,而子进程的进程识别码也会一快返回。如果不在意结束状态值,则参数 status 可以设成 NULL。子进程的结束状态值请参考 waitpid()。
返回值:如果执行成功则返回子进程识别码(PID),如果有错误发生则返回-1。失败原因存于errno中。
附加说明:
相关函数:waitpid,fork
示例
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
int main()
{
pid_t pid;
int status,i;
if (fork() == 0) {
printf("This is the child process, pid = %d\n", getpid());
exit(5);
}
else {
sleep(1);
printf("This is the parent process, wait for child...\n");
pid = wait(&status);
i = WEXITSTATUS(status);
printf("child’s pid = %d, exit status = %d\n", pid, i);
}
}
执行
This is the child process, pid = 1501
This is the parent process, wait for child...
child’s pid = 1501, exit status = 5
waitpid
等待子进程中断或结束
头文件
#include <sys/types.h>
#include <sys/wait.h>
函数原型
pid_t waitpid(pid_t pid, int * status, int options);
说明:waitpid()会暂时停止目前进程的执行,直到有信号来到或子进程结束。如果在调用wait()时子进程已经结束,则wait()会立即返回子进程结束状态值。子进程的结束状态值会由参数status返回,而子进程的进程识别码也会一快返回。如果不在意结束状态值,则参数status可以设成NULL。
参数pid为欲等待的子进程识别码,其他数值意义如下:
pid<-1 等待进程组识别码为pid绝对值的任何子进程。
pid=-1 等待任何子进程,相当于wait()。
pid=0 等待进程组识别码与目前进程相同的任何子进程。
pid>0 等待任何子进程识别码为pid的子进程。
参数option可以为0 或下面的OR 组合
WNOHANG 如果没有任何已经结束的子进程则马上返回,不予以等待。
WUNTRACED 如果子进程进入暂停执行情况则马上返回,但结束状态不予以理会。
子进程的结束状态返回后存于status,底下有几个宏可判别结束情况
WIFEXITED(status)如果子进程正常结束则为非0值。
WEXITSTATUS(status)取得子进程exit()返回的结束代码,一般会先用WIFEXITED 来判断是否正常结束才能使用此宏。
WIFSIGNALED(status)如果子进程是因为信号而结束则此宏值为真
WTERMSIG(status)取得子进程因信号而中止的信号代码,一般会先用WIFSIGNALED 来判断后才使用此宏。
WIFSTOPPED(status)如果子进程处于暂停执行情况则此宏值为真。一般只有使用WUNTRACED 时才会有此情况。
WSTOPSIG(status)取得引发子进程暂停的信号代码,一般会先用WIFSTOPPED 来判断后才使用此宏。
返回值:如果执行成功则返回子进程识别码(PID),如果有错误发生则返回-1。失败原因存于errno中。
附加说明:
相关函数:wait,fork
示例
参考 wait()。
fprintf
格式化输出数据至文件
头文件
#include <stdio.h>
函数原型
int fprintf(FILE * stream, const char * format, ...);
说明:fprintf()会根据参数format字符串来转换并格式化数据,然后将结果输出到参数stream指定的文件中,直到出现字符串结束(’\0’)为止。
返回值:关于参数format字符串的格式请参考printf()。成功则返回实际输出的字符数,失败则返回-1,错误原因存于errno中。
附加说明:
相关函数:printf,fscanf,vfprintf
示例
#include <stdio.h>
int main()
{
int i = 150;
int j = -100;
double k = 3.14159;
fprintf(stdout, "%d %f %x\n", j, k, i);
fprintf(stdout, "%2d %*d\n", i, 2, i);
return 0;
}
执行
-100 3.141590 96
150 150
fscanf
格式化字符串输入
头文件
#include <stdio.h>
函数原型
int fscanf(FILE * stream, const char *format, ...);
说明:fscanf()会自参数stream的文件流中读取字符串,再根据参数format字符串来转换并格式化数据。格式转换形式请参考scanf()。转换后的结构存于对应的参数内。
返回值:成功则返回参数数目,失败则返回-1,错误原因存于errno中。
附加说明:
相关函数:scanf,sscanf
示例
#include <stdio.h>
int main()
{
int i;
unsigned int j;
char s[5];
fscanf(stdin, "%d %x %5[a-z] %*s %f", &i, &j, s, s);
printf("%d %d %s \n", i, j, s);
return 0;
}
执行
10 0x1b aaaaaaaaa bbbbbbbbbb # 从键盘输入
10 27 aaaaa
printf
格式化输出数据
头文件
#include <stdio.h>
函数原型
int printf(const char * format, ...);
说明:printf() 会根据参数 format 字符串来转换并格式化数据,然后将结果写出到标准输出设备,直到出现字符串结束(’\0’)为止。参数 format 字符串可包含下列三种字符类型:
一般文本,伴随直接输出。
ASCII控制字符,如\t、\n等。
格式转换字符。
格式转换为一个百分比符号(%)及其后的格式字符所组成。一般而言,每个%符号在其后都必需有一 printf() 的参数与之相呼应(只有当%%转换字符出现时会直接输出%字符),而欲输出的数据类型必须与其相对应的转换字符类型相同。 printf() 格式转换的一般形式如下:
%(flags)(width)(.prec)type
以中括号括起来的参数为选择性参数,而%与type则是必要的。底下先介绍 type 的几种形式 整数:
%d
整数的参数会被转成一有符号的十进制数字%u
整数的参数会被转成一无符号的十进制数字%o
整数的参数会被转成一无符号的八进制数字%x
整数的参数会被转成一无符号的十六进制数字,并以小写abcdef表示%X
整数的参数会被转成一无符号的十六进制数字,并以大写ABCDEF表示浮点型数%f
double 型的参数会被转成十进制数字,并取到小数点以下六位,四舍五入。%e
double型的参数以指数形式打印,有一个数字会在小数点前,六位数字在小数点后,而在指数部分会以小写的e来表示。%E
与%e作用相同,唯一区别是指数部分将以大写的E 来表示。%g
double 型的参数会自动选择以%f 或%e 的格式来打印,其标准是根据欲打印的数值及所设置的有效位数来决定。%G
与%g
作用相同,唯一区别在以指数形态打印时会选择%E 格式。 字符及字符串:%c
整型数的参数会被转成unsigned char型打印出。%s
指向字符串的参数会被逐字输出,直到出现NULL字符为止%p
如果是参数是void *
型指针则使用十六进制格式显示。 prec 有几种情况:正整数的最小位数。
在浮点型数中代表小数位数
在%g 格式代表有效位数的最大值。
在%s格式代表字符串的最大长度。
若为×符号则代表下个参数值为最大长度。
width为参数的最小长度,若此栏并非数值,而是*符号,则表示以下一个参数当做参数长度。
flags 有下列几种情况
#NAME?
+
一般在打印负数时,printf()会加印一个负号,整数则不加任何负号。此旗标会使得在打印正数前多一个正号(+)。#
此旗标会根据其后转换字符的不同而有不同含义。当在类型为o 之前(如%#o),则会在打印八进制数值前多印一个o。而在类型为x 之前(%#x)则会在打印十六进制数前多印’0x’,在型态为e、E、f、g或G 之前则会强迫数值打印小数点。在类型为g 或G之前时则同时保留小数点及小数位数末尾的零。0
当有指定参数时,无数字的参数将补上0。默认是关闭此旗标,所以一般会打印出空白字符。
返回值:成功则返回实际输出的字符数,失败则返回-1,错误原因存于errno中。
附加说明:
相关函数:scanf,snprintf
示例
#include <stdio.h>
int main()
{
int i = 150;
int j = -100;
double k = 3.14159;
printf("%d %f %x\n", j, k, i);
printf("%2d %*d\n", i, 2, i); /*参数2 会代入格式*中,而与%2d同意义*/
return 0;
}
执行
-100 3.14159 96
150 150
scanf
格式化字符串输入
头文件
#include <stdio.h>
函数原型
int scanf(const char * format, ...);
说明:scanf()会将输入的数据根据参数format字符串来转换并格式化数据。Scanf()格式转换的一般形式如下:
%[*][size][l][h]type
以中括号括起来的参数为选择性参数,而%与type则是必要的。
*
代表该对应的参数数据忽略不保存。size
为允许参数输入的数据长度。l
输入的数据数值以long int 或double型保存。h
输入的数据数值以short int 型保存。 底下介绍type的几种形式%d
输入的数据会被转成一有符号的十进制数字(int)。%i
输入的数据会被转成一有符号的十进制数字,若输入数据以”0x”或”0X”开头代表转换十六进制数字,若以”0”开头则转换八进制数字,其他情况代表十进制。%0
输入的数据会被转换成一无符号的八进制数字。%u
输入的数据会被转换成一无符号的正整数。%x
输入的数据为无符号的十六进制数字,转换后存于unsigned int型变量。%X
同%x%f
输入的数据为有符号的浮点型数,转换后存于float型变量。%e
同%f%E
同%f%g
同%f%s
输入数据为以空格字符为终止的字符串。%c
输入数据为单一字符。[]
读取数据但只允许括号内的字符。如[a-z]。[^]
读取数据但不允许中括号的^符号后的字符出现,如[^0-9].
返回值:成功则返回参数数目,失败则返回-1,错误原因存于errno中。
附加说明:
相关函数:fscanf,snprintf
示例
#include <stdio.h>
int main()
{
int i;
unsigned int j;
char s[5];
scanf("%d %x %5[a-z] %*s %f", &i, &j, s, s);
printf("%d %d %s\n", i, j, s);
return 0;
}
执行
10 0x1b aaaaaaaaaa bbbbbbbbbb
10 27 aaaaa
sprintf
格式化字符串复制
头文件
#include <stdio.h>
函数原型
int sprintf( char *str, const char * format, ...);
说明:sprintf()会根据参数format字符串来转换并格式化数据,然后将结果复制到参数str所指的字符串数组,直到出现字符串结束(’\0’)为止。关于参数format字符串的格式请参考printf()。
返回值:成功则返回参数str字符串长度,失败则返回-1,错误原因存于errno中。
附加说明:使用此函数得留意堆栈溢出,或改用 snprintf()。
相关函数:printf,sprintf
示例
#include <stdio.h>
int main()
{
char * a = "This is string A!";
char buf[80];
sprintf(buf, ">>> %s<<<\n", a);
printf("%s", buf);
return 0;
}
执行
>>>This is string A!<<<
sscanf
格式化字符串输入
头文件
#include <stdio.h>
函数原型
int sscanf (const char *str, const char * format, ...);
说明:sscanf() 会将参数 str 的字符串根据参数 format 字符串来转换并格式化数据。格式转换形式请参考 scanf()。转换后的结果存于对应的参数内。
返回值:成功则返回参数数目,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:scanf,fscanf
示例
#include <stdio.h>
int main()
{
int i;
unsigned int j;
char input[] = "10 0x1b aaaaaaaa bbbbbbbb";
char s[5];
sscanf(input, "%d %x %5[a-z] %*s %f", &i, &j, s, s);
printf("%d %d %s\n", i, j, s);
return 0;
}
执行
10 27 aaaaa
vfprintf
格式化输出数据至文件
头文件
#include <stdio.h>
#include <stdarg.h>
函数原型
int vfprintf(FILE *stream, const char * format, va_list ap);
说明:vfprintf() 会根据参数 format 字符串来转换并格式化数据,然后将结果输出到参数 stream 指定的文件中,直到出现字符串结束(’\0’)为止。关于参数 format 字符串的格式请参考 printf()。va_list 用法请参考附录 C 或 vprintf() 范例。
返回值:成功则返回实际输出的字符数,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:printf,fscanf,fprintf
示例
参考 fprintf() 及 vprintf()。
vfscanf
格式化字符串输入
头文件
#include <stdio.h>
函数原型
int vfscanf(FILE * stream, const char * format, va_list ap);
说明:vfscanf() 会自参数 stream 的文件流中读取字符串,再根据参数 format 字符串来转换并格式化数据。格式转换形式请参考 scanf()。转换后的结果存于对应的参数内。va_list 用法请参考附录 C 或vprintf()。
返回值:成功则返回参数数目,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:scanf,sscanf,fscanf
示例
参考 fscanf() 及 vprintf()。
vprintf
格式化输出数据
头文件
#include <stdio.h>
#include <stdarg.h>
函数原型
int vprintf(const char * format, va_list ap);
说明:vprintf() 作用和 printf() 相同,参数 format 格式也相同。va_list 为不定个数的参数列,用法及范例请参考附录 C。
返回值:成功则返回实际输出的字符数,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:printf,vfprintf,vsprintf
示例
#include <stdio.h>
#include <stdarg.h>
int my_printf(const char *format, ...)
{
va_list ap;
int retval;
va_start(ap, format);
printf("my_printf( ):");
retval = vprintf(format, ap);
va_end(ap);
return retval;
}
int main()
{
int i = 150, j = -100;
double k = 3.14159;
my_printf("%d %f %x\n", j, k, i);
my_printf("%2d %*d\n", i, 2, i);
return 0;
}
执行
my_printf() : -100 3.14159 96
my_printf() : 150 150
vscanf
格式化字符串输入
头文件
#include <stdio.h>
#include <stdarg.h>
函数原型
int vscanf( const char * format, va_list ap);
说明:vscanf() 会将输入的数据根据参数 format 字符串来转换并格式化数据。格式转换形式请参考 scanf()。转换后的结果存于对应的参数内。va_list 用法请参考附录 C 或 vprintf() 范例。
返回值:成功则返回参数数目,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:vsscanf,vfscanf
示例
请参考 scanf() 及 vprintf()。
vsprintf
格式化字符串复制
头文件
#include <stdio.h>
函数原型
int vsprintf(char * str, const char * format, va_list ap);
说明:vsprintf() 会根据参数 format 字符串来转换并格式化数据,然后将结果复制到参数 str 所指的字符串数组,直到出现字符串结束(’\0’)为止。关于参数 format 字符串的格式请参考 printf()。va_list 用法请参考附录 C 或 vprintf() 范例。
返回值:成功则返回参数 str 字符串长度,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:vnsprintf,vprintf,snprintf
示例
请参考 vprintf() 及 vsprintf()。
vsscanf
格式化字符串输入
头文件
#include <stdio.h>
函数原型
int vsscanf(const char * str, const char * format, va_list ap);
说明:vsscanf() 会将参数 str 的字符串根据参数 format 字符串来转换并格式化数据。格式转换形式请参考附录 C 或 vprintf() 范例。
返回值:成功则返回参数数目,失败则返回 -1,错误原因存于 errno 中。
附加说明:
相关函数:vscanf,vfscanf
示例
请参考 sscanf() 及 vprintf()。