我最近在看UNIX网络编程,在Linux的环境下进行编辑,例子不能完全编辑,就有了这篇博客,来总结一下.
书中的例子为:
#include "unp.h"
int main(int argc, char **argv)
{
int sockfd, n;
char recvline[MAXLINE + 1];
struct sockaddr_in servaddr;
if (argc != 2)
err_quit("usage: a.out <IPaddress>");
if ( (sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
err_sys("socket error");
bzero(&servaddr, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(13); /* daytime server */
if (inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <= 0)
err_quit("inet_pton error for %s", argv[1]);
if (connect(sockfd, (SA *) &servaddr, sizeof(servaddr)) < 0)
err_sys("connect error");
while ( (n = read(sockfd, recvline, MAXLINE)) > 0) {
recvline[n] = 0;
if (fputs(recvline, stdout) == EOF)
err_sys("fputs error");
}
if (n < 0) /* null terminate */
err_sys("read error");
exit(0);
}
UNIX环境下的编辑和Linux环境下的编辑还是有一些区别的.
unp.h可以在书中的源代码中找到,当中定义了很多变量,比较庞大,所以干脆自己在网上搜寻了,发现一个比较简单的版本:
#include <errno.h> /* for syslog() */
#include <stdarg.h> /* ANSI C header file */
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#define MAXLINE 4096
static void err_doit(int, int, const char *, va_list);
/* Nonfatal error related to system call
* Print message and return */
void err_ret(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, errno, fmt, ap);
va_end(ap);
return;
}
/* Fatal error related to system call
* Print 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);
}
/* Fatal error related to system call
* Print message, dump core, and terminate */
void err_dump(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(1, errno, fmt, ap);
va_end(ap);
abort(); /* dump core and terminate */
exit(1); /* shouldn't get here */
}
/* Nonfatal error unrelated to system call
* Print message and return */
void err_msg(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(0, errno, fmt, ap);
va_end(ap);
return;
}
/* Fatal error unrelated to system call
* Print message and terminate */
void err_quit(const char *fmt, ...)
{
va_list ap;
va_start(ap, fmt);
err_doit(0, errno, fmt, ap);
va_end(ap);
exit(1);
}
/* Print message and return to caller
* Caller specifies "errnoflag" and "level" */
static void err_doit(int errnoflag, int level, const char *fmt, va_list ap)
{
int errno_save, n;
char buf[MAXLINE + 1];
errno_save = errno; /* value caller might want printed */
#ifdef HAVE_VSNPRINTF
vsnprintf(buf, MAXLINE, fmt, ap); /* safe */
#else
vsprintf(buf, fmt, ap); /* not safe */
#endif
n = strlen(buf);
if (errnoflag)
snprintf(buf + n, MAXLINE - n, ": %s", strerror(errno_save));
strcat(buf, "\n");
fflush(stdout); /* in case stdout and stderr are the same */
fputs(buf, stderr);
fflush(stderr);
return;
}
这段代码解决了err_sys和err_quit的调用,把他存为error.h文件,然后再把例子中的第一行
#include "unp.h"
换成为
#include <netinet/in.h>
#include "error.h"
struct sockaddr_in, 函数式socket和constant AF_INET, SOCK_STREAM都需要netinet/in.h的引用导入.
我使用的编辑命令为:
gcc -Wall -o a.out daytimetcpcli.c
通过编译以后要查看Linux是否开启daytime service.
如果没有要先开启daytime service.步骤为:
1).安装daytime的服务包xinetd
2). 修改/etc/xinetd.d/daytime的文件,把文件中的disable从yes改成no
3). 重启daytime service:sudo service xinet restart
daytime service就在当前的Linux环境中启动了,
最后可以执行测试,我使用的测试命令为:
~/unp/examples/daytimetcpcli$ sudo ./a.out 127.0.0.1
27 OCT 2017 17:32:06 CEST