posts - 297,  comments - 15,  trackbacks - 0
本文介绍Linux系统网络编程的内容,如套接口的概念与使用、网络编程的结构等。
1 什么是套接口
简单地说,套接口就是一种使用UNIX系统中的文件描述符和系统进程通信的一种方法。
因为在UNIX系统中,所有的I/O操作都是通过读写文件描述符而产生的。文件描述符就是
一个和打开的文件相关连的整数。但文件可以是一个网络连接、一个FIFO、一个管道、一个
终端、一个真正存储在磁盘上的文件或者UNIX系统中的任何其他的东西。所以,如果你希望
通过Internet和其他的程序进行通信,你只有通过文件描述符。
使用系统调用socket(),你可以得到socket()描述符。然后你可以使用send() 和recv()调用而
与其他的程序通信。你也可以使用一般的文件操作来调用read() 和write()而与其他的程序进行
通信,但send() 和recv()调用可以提供一种更好的数据通信的控制手段。下面我们讨论Internet
套接口的使用方法。
2 两种类型的Internet套接口
有两种最常用的Internet 套接口,“数据流套接口”和“数据报套接口”,以后我们用
“SOCK_STREAM” 和“SOCK_DGRAM”分别代表上面两种套接口。数据报套接口有时也
叫做“无连接的套接口”。
数据流套接口是可靠的双向连接的通信数据流。如果你在套接口中以“ 1, 2”的顺序放入
两个数据,它们在另一端也会以“1, 2”的顺序到达。它们也可以被认为是无错误的传输。
经常使用的telnet应用程序就是使用数据流套接口的一个例子。使用HTTP的WWW浏览器
也使用数据流套接口来读取网页。事实上,如果你使用telnet 登录到一个WWW站点的8 0端口,
然后键入“GET 网页名”,你将可以得到这个HTML页。数据流套接口使用TCP得到这种高质
量的数据传输。数据报套接口使用UDP,所以数据报的顺序是没有保障的。数据报是按一种应
答的方式进行数据传输的。
3 网络协议分层
由于网络中的协议是分层的,所以上层的协议是依赖于下一层所提供的服务的。也就是说,
你可以在不同的物理网络中使用同样的套接口程序,因为下层的协议对你来说是透明的。
UNIX系统中的网络协议是这样分层的:
• 应用层( telnet、ftp等)。
• 主机到主机传输层( TCP、UDP )。
• Internet层( IP和路由)。
• 网络访问层(网络、数据链路和物理层)。
4 数据结构
下面我们要讨论使用套接口编写程序可能要用到的数据结构。
首先是套接口描述符。一个套接口描述符只是一个整型的数值: i n t。
第一个数据结构是struct sockaddr,这个数据结构中保存着套接口的地址信息。
struct sockaddr {
   unsigned short sa_family;    /* address family, AF_xxx */
   char sa_data[14];               /* 14 bytes of protocol address */
} ;
sa_family 中可以是其他的很多值,但在这里我们把它赋值为“ A F _ I N E T”。s a _ d a t a包括一
个目的地址和一个端口地址。
你也可以使用另一个数据结构s o c k a d d r _ i n,如下所示:
struct sockaddr_in {
   short int sin_family;                   /* Address family */
   unsigned short int sin_port;       /* Port number */
   struct in_addr sin_addr;           /* Internet address */
   unsigned char sin_zero[8];       /* Same size as struct sockaddr */
} ;
这个数据结构使得使用其中的各个元素更为方便。要注意的是sin_zero应该使用bzero() 或
者memset()而设置为全0。另外,一个指向sockaddr_in数据结构的指针可以投射到一个指向数
据结构sockaddr的指针,反之亦然。
5 IP地址和如何使用IP地址
有一系列的程序可以使你处理I P地址。
首先,你可以使用inet_addr()程序把诸如“132.241.5.10”形式的IP地址转化为无符号的整
型数。
ina.sin_addr.s_addr = inet_addr("132.241.5.10");
如果出错,inet_addr()程序将返回- 1。
也可以调用inet_ntoa()把地址转换成数字和句点的形式:
printf ("%s", inet_ntoa (ina.sin_addr));
这将会打印出I P地址。它返回的是一个指向字符串的指针。
5.1 socket()
我们使用系统调用socket()来获得文件描述符:
#include <sys/types.h>
#include <sys/socket.h>
int socket(int domain, int type, int protocol);
第一个参数domain设置为“AF_INET”。第二个参数是套接口的类型:SOCK_STREAM 或
SOCK_DGRAM。第三个参数设置为0。
系统调用socket()只返回一个套接口描述符,如果出错,则返回- 1。
5.2 bind()
一旦你有了一个套接口以后,下一步就是把套接口绑定到本地计算机的某一个端口上。但
如果你只想使用connect()则无此必要。
下面是系统调用bind()的使用方法:
#include <sys/types.h>
#include <sys/socket.h>
int bind(int sockfd, struct sockaddr *my_addr, int addrlen);
第一个参数sockfd 是由socket()调用返回的套接口文件描述符。第二个参数my_addr 是指向
数据结构sockaddr的指针。数据结构sockaddr中包括了关于你的地址、端口和I P地址的信息。
第三个参数addrlen可以设置成sizeof(struct sockaddr)。
下面是一个例子:
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#define MYPORT 3490
m a i n ( )
{
int sockfd;
struct sockaddr_in my_addr;
sockfd = socket(AF_INET, SOCK_STREAM, 0); /* do some error checking! */
m y _ a d d r.sin_family = AF_INET; /* host byte order */
m y _ a d d r.sin_port = htons(MYPORT); /* short, network byte order */
m y _ a d d r. s i n _ a d d r.s_addr = inet_addr("132.241.5.10");
b z e r o ( & ( m y _ a d d r.sin_zero), 8); /* zero the rest of the struct */
/* don't forget your error checking for bind(): */
bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr));
如果出错,bind() 也返回- 1。
如果你使用connect()系统调用,那么你不必知道你使用的端口号。当你调用connect()时,
它检查套接口是否已经绑定,如果没有,它将会分配一个空闲的端口。
5.3 connect()
系统调用connect()的用法如下:
#include <sys/types.h>
#include <sys/socket.h>
int connect(int sockfd, struct sockaddr *serv_addr, int addrlen);
第一个参数还是套接口文件描述符,它是由系统调用socket()返回的。第二个参数是
serv_addr是指向数据结构sockaddr的指针,其中包括目的端口和I P地址。第三个参数可以使用
sizeof(struct sockaddr)而获得。下面是一个例子:
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#define DEST_IP "132.241.5.10"
#define DEST_PORT 23
main()
{
int sockfd;
struct sockaddr_in dest_addr; /* will hold the destination addr */
sockfd = socket(AF_INET, SOCK_STREAM, 0); /* do some error checking! */
d e s t _ a d d r.sin_family = AF_INET; /* host byte order */
d e s t _ a d d r.sin_port = htons(DEST_PORT); /* short, network byte order */
d e s t _ a d d r. s i n _ a d d r.s_addr = inet_addr(DEST_IP);
b z e r o ( & ( d e s t _ a d d r.sin_zero), 8); /* zero the rest of the struct */
/* don't forget to error check the connect()! */
connect(sockfd, (struct sockaddr *)&dest_addr, sizeof(struct sockaddr));
同样,如果出错, c o n n e c t ( )将会返回- 1。
5.4 listen()
如果你希望不连接到远程的主机,也就是说你希望等待一个进入的连接请求,然后再处理
它们。这样,你通过首先调用l i s t e n ( ),然后再调用a c c e p t ( )来实现。
系统调用l i s t e n ( )的形式如下:
int listen(int sockfd, int backlog);
第一个参数是系统调用s o c k e t ( )返回的套接口文件描述符。第二个参数是进入队列中允许
的连接的个数。进入的连接请求在使用系统调用a c c e p t ( )应答之前要在进入队列中等待。这个
值是队列中最多可以拥有的请求的个数。大多数系统的缺省设置为2 0。你可以设置为5或者1 0。
当出错时,l i s t e n ( )将会返回- 1值。
当然,在使用系统调用l i s t e n ( )之前,我们需要调用b i n d ( )绑定到需要的端口,否则系统内
核将会让我们监听一个随机的端口。所以,如果你希望监听一个端口,下面是应该使用的系统
调用的顺序:
s o c k e t ( ) ;
b i n d ( ) ;
l i s t e n ( ) ;
/* accept() goes here */
5.5 accept()
系统调用a c c e p t ( )比较起来有点复杂。在远程的主机可能试图使用c o n n e c t ( )连接你使用
l i s t e n ( )正在监听的端口。但此连接将会在队列中等待,直到使用a c c e p t ( )处理它。调用a c c e p t ( )
之后,将会返回一个全新的套接口文件描述符来处理这个单个的连接。这样,对于同一个连接
来说,你就有了两个文件描述符。原先的一个文件描述符正在监听你指定的端口,新的文件描
述符可以用来调用s e n d ( )和r e c v ( )。
调用的例子如下:
#include <sys/socket.h>
int accept(int sockfd, void *addr, int *addrlen);
第一个参数是正在监听端口的套接口文件描述符。第二个参数a d d r是指向本地的数据结构
s o c k a d d r _ i n的指针。调用c o n n e c t ( )中的信息将存储在这里。通过它你可以了解哪个主机在哪个
端口呼叫你。第三个参数同样可以使用sizeof(struct sockaddr_in)来获得。
如果出错,a c c e p t ( )也将返回- 1。下面是一个简单的例子:
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#define MYPORT 3490 /* the port users will be connecting to */
#define BACKLOG 10 /* how many pending connections queue will hold */
m a i n ( )
{
int sockfd, new_fd; /* listen on sock_fd, new connection on new_fd */
struct sockaddr_in my_addr; /* my address information */
struct sockaddr_in their_addr; /* connector's address information */
int sin_size;
sockfd = socket(AF_INET, SOCK_STREAM, 0); /* do some error checking! */
m y _ a d d r.sin_family = AF_INET; /* host byte order */
m y _ a d d r.sin_port = htons(MYPORT); /* short, network byte order */
m y _ a d d r. s i n _ a d d r.s_addr = INADDR_ANY; /* auto-fill with my IP */
b z e r o ( & ( m y _ a d d r.sin_zero), 8); /* zero the rest of the struct */
/* don't forget your error checking for these calls: */
bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr));
listen(sockfd, BACKLOG);
sin_size = sizeof(struct sockaddr_in);
new_fd = accept(sockfd, &their_addr, &sin_size);
下面,我们将可以使用新创建的套接口文件描述符n e w _ f d来调用s e n d ( )和r e c v ( )。
5.6 send() 和recv()
系统调用s e n d ( )的用法如下:
int send(int sockfd, const void *msg, int len, int flags);
第一个参数是你希望给发送数据的套接口文件描述符。它可以是你通过s o c k e t ( )系统调用
返回的,也可以是通过a c c e p t ( )系统调用得到的。第二个参数是指向你希望发送的数据的指针。
第三个参数是数据的字节长度。第四个参数标志设置为0。
下面是一个简单的例子:
char *msg = "Beej was here!";
int len, bytes_sent;
len = strlen(msg);
bytes_sent = send(sockfd, msg, len, 0);
系统调用s e n d ( )返回实际发送的字节数,这可能比你实际想要发送的字节数少。如果返回
的字节数比要发送的字节数少,你在以后必须发送剩下的数据。当s e n d ( )出错时,将返回- 1。
系统调用r e c v ( )的使用方法和s e n d ( )类似:
int recv(int sockfd, void *buf, int len, unsigned int flags);
第一个参数是要读取的套接口文件描述符。第二个参数是保存读入信息的地址。第三个参
数是缓冲区的最大长度。第四个参数设置为0。
系统调用r e c v ( )返回实际读取到缓冲区的字节数,如果出错则返回- 1。
这样使用上面的系统调用,你可以通过数据流套接口来发送和接受信息。
5.7 sendto() 和recvfrom()
因为数据报套接口并不连接到远程的主机上,所以在发送数据包之前,我们必须首先给出
目的地址,请看:
int sendto(int sockfd, const void *msg, int len, unsigned int flags,
const struct sockaddr *to, int tolen);
除了两个参数以外,其他的参数和系统调用s e n d ( )时相同。参数t o是指向包含目的I P地址
和端口号的数据结构s o c k a d d r的指针。参数t o l e n可以设置为sizeof(struct sockaddr)。
系统调用s e n d t o ( )返回实际发送的字节数,如果出错则返回- 1。
系统调用r e c v f r o m ( )的使用方法也和r e c v ( )的十分近似:
int recvfrom(int sockfd, void *buf, int len, unsigned int flags
struct sockaddr *from, int *fromlen);
参数f r o m是指向本地计算机中包含源I P地址和端口号的数据结构s o c k a d d r的指针。参数
f r o m l e n设置为sizeof(struct sockaddr)。
系统调用r e c v f r o m ( )返回接收到的字节数,如果出错则返回- 1。
5.8 close() 和shutdown()
你可以使用c l o s e ( )调用关闭连接的套接口文件描述符:
c l o s e ( s o c k f d ) ;
这样就不能再对此套接口做任何的读写操作了。
使用系统调用s h u t d o w n ( ),可有更多的控制权。它允许你在某一个方向切断通信,或者切
断双方的通信:
int shutdown(int sockfd, int how);
第一个参数是你希望切断通信的套接口文件描述符。第二个参数h o w值如下:
0—Further receives are disallowed
1—Further sends are disallowed
2—Further sends and receives are disallowed (like close())
shutdown() 如果成功则返回0,如果失败则返回- 1。
5.9 getpeername()
这个系统的调用十分简单。它将告诉你是谁在连接的另一端:
#include <sys/socket.h>
int getpeername(int sockfd, struct sockaddr *addr, int *addrlen);
第一个参数是连接的数据流套接口文件描述符。第二个参数是指向包含另一端的信息的数
据结构s o c k a d d r的指针。第三个参数可以设置为sizeof(struct sockaddr)。
如果出错,系统调用将返回- 1。
一旦你获得了它们的地址,你可以使用inet_ntoa() 或者g e t h o s t b y a d d r ( )来得到更多的信息。
5.10 gethostname()
系统调用g e t h o s t n a m e ( )比系统调用g e t p e e r n a m e ( )还简单。它返回程序正在运行的计算机的
名字。系统调用g e t h o s t b y n a m e ( )可以使用这个名字来决定你的机器的I P地址。
下面是一个例子:
#include <unistd.h>
int gethostname(char *hostname, size_t size);
如果成功,g e t h o s t n a m e将返回0。如果失败,它将返回- 1。
6 DNS
DNS 代表“Domain Name Service”,即域名服务器。它可以把域名翻译成相应的I P地址。
你可以使用此I P地址调用b i n d ( )、c o n n e c t ( )、s e n d t o ( )或者用于其他的地方。
系统调用g e t h o s t b y n a m e ( )可以完成这个函数:
#include <netdb.h>
struct hostent *gethostbyname(const char *name);
它返回一个指向数据结构h o s t e n t的指针,数据结构h o s t e n t如下:
struct hostent {
char *h_name;
char **h_aliases;
int h_addrtype;
int h_length;
char **h_addr_list;
} ;
#define h_addr h_addr_list[0]
h_name —主机的正式名称。
h_aliases —主机的别名。
h _ a d d r t y p e—将要返回的地址的类型,一般是A F _ I N E T。
h _ l e n g t h—地址的字节长度。
h _ a d d r _ l i s t—主机的网络地址。
h_addr —h _ a d d r _ l i s t中的第一个地址。
系统调用g e t h o s t b y n a m e ( )返回一个指向填充好的数据结构h o s t e n t的指针。当发生错误时,
则返回一个N U L L指针。下面是一个实际例子:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
int main(int argc, char *argv[])
{
struct hostent *h;
if (argc != 2) { /* error check the command line */
f p r i n t f ( s t d e r r,"usage: getip address\n");
e x i t ( 1 ) ;
}
if ((h=gethostbyname(argv[1])) == NULL) { /* get the host info */
h e r r o r ( " g e t h o s t b y n a m e " ) ;
e x i t ( 1 ) ;
}
printf("Host name : %s\n", h->h_name);
printf("IP Address : %s\n",inet_ntoa(*((struct in_addr *)h->h_addr)));
return 0;
}
在使用g e t h o s t b y n a m e ( )时,你不能使用p e r r o r ( )来打印错误信息。你应该使用的是系统调用
h e r r o r ( )。
7 客户机/服务器模式
在网络上大部分的通信都是在客户机/服务器模式下进行的。例如t e l n e t。当你使用t e l n e t连
接到远程主机的端口2 3时,主机上的一个叫做t e l n e t d的程序就开始运行。它处理所有进入的
t e l n e t连接,为你设置登录提示符等。
应当注意的是客户机/服务器模式可以使用S O C K _ S T R E A M、S O C K _ D G R A M或者任何其
他的方式。例如t e l n e t / t e l n e t d、f t p / f t p d和b o o t p / b o o t p d。每当你使用f t p时,远程计算机都在运
行一个f t p d为你服务。
一般情况下,一台机器上只有一个服务器程序,它通过使用f o r k ( )来处理多个客户端程序
的请求。最基本的处理方法是:服务器等待连接,使用a c c e p t ( )接受连接,调用f o r k ( )生成一个
子进程处理连接。
8 简单的数据流服务器程序
此服务器程序所作的事情就是通过一个数据流连接发送字符串“ Hello, Wo r l d ! \ n”。你可以
在一个窗口上运行此程序,然后在另一个窗口使用t e l n e t:
$ telnet remotehostname 3490
其中,r e m o t e h o s t n a m e是你运行的机器名。下面是此程序的代码:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#define MYPORT 3490 /* the port users will be connecting to */
#define BACKLOG 10 /* how many pending connections queue will hold */
m a i n ( )
{
int sockfd, new_fd; /* listen on sock_fd, new connection on new_fd */
struct sockaddr_in my_addr; /* my address information */
struct sockaddr_in their_addr; /* connector's address information */
int sin_size;
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
p e r r o r ( " s o c k e t " ) ;
e x i t ( 1 ) ;
}
m y _ a d d r.sin_family = AF_INET; /* host byte order */
m y _ a d d r.sin_port = htons(MYPORT); /* short, network byte order */
m y _ a d d r. s i n _ a d d r.s_addr = INADDR_ANY; /* auto-fill with my IP */
b z e r o ( & ( m y _ a d d r.sin_zero), 8); /* zero the rest of the struct */
if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) \
== -1) {
p e r r o r ( " b i n d " ) ;
e x i t ( 1 ) ;
}
if (listen(sockfd, BACKLOG) == -1) {
p e r r o r ( " l i s t e n " ) ;
e x i t ( 1 ) ;
}
while(1) { /* main accept() loop */
sin_size = sizeof(struct sockaddr_in);
if ((new_fd = accept(sockfd, (struct sockaddr *)&their_addr, \
&sin_size)) == -1) {
p e r r o r ( " a c c e p t " ) ;
c o n t i n u e ;
}
printf("server: got connection from %s\n", \
i n e t _ n t o a ( t h e i r _ a d d r. s i n _ a d d r ) ) ;
if (!fork()) { /* this is the child process */
if (send(new_fd, "Hello, world!\n", 14, 0) == -1)
perror("send");
close(new_fd);
exit(0);
}
close(new_fd); /* parent doesn't need this */
while(waitpid(-1,NULL,WNOHANG) > 0); /* clean up child processes */
}
}
你也可以使用下面的客户机程序从服务器上得到字符串。
9 简单的数据流客户机程序
客户机所做的是连接到你在命令行中指定的主机的3 4 9 0端口。它读取服务器发送的字符
串。
下面是客户机程序的代码:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <netdb.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#define PORT 3490 /* the port client will be connecting to */
#define MAXDATASIZE 100 /* max number of bytes we can get at once */
int main(int argc, char *argv[])
{
int sockfd, numbytes;
char buf[MAXDATA S I Z E ] ;
struct hostent *he;
struct sockaddr_in their_addr; /* connector's address information */
if (argc != 2) {
f p r i n t f ( s t d e r r,"usage: client hostname\n");
e x i t ( 1 ) ;
}
if ((he=gethostbyname(argv[1])) == NULL) { /* get the host info */
h e r r o r ( " g e t h o s t b y n a m e " ) ;
e x i t ( 1 ) ;
}
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) == -1) {
p e r r o r ( " s o c k e t " ) ;
e x i t ( 1 ) ;
}
t h e i r _ a d d r.sin_family = AF_INET; /* host byte order */
t h e i r _ a d d r.sin_port = htons(PORT); /* short, network byte order */
t h e i r _ a d d r.sin_addr = *((struct in_addr *)he->h_addr);
b z e r o ( & ( t h e i r _ a d d r.sin_zero), 8); /* zero the rest of the struct */
if (connect(sockfd, (struct sockaddr *)&their_addr, \
sizeof(struct sockaddr)) == -1) {
p e r r o r ( " c o n n e c t " ) ;
e x i t ( 1 ) ;
}
if ((numbytes=recv(sockfd, buf, MAXDATASIZE, 0)) == -1) {
p e r r o r ( " r e c v " ) ;
e x i t ( 1 ) ;
}
buf[numbytes] = '\0';
printf("Received: %s",buf);
c l o s e ( s o c k f d ) ;
return 0;
}
如果你在运行服务器程序之前运行客户机程序,则将会得到一个“ Connection refused”的
信息。
10 数据报套接口
程序l i s t e n e r在机器中等待端口4 9 5 0到来的数据包。程序t a l k e r向指定的机器的4 9 5 0端口发
送数据包。
下面是l i s t e n e r. c的代码:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <sys/socket.h>
#include <sys/wait.h>
#define MYPORT 4950 /* the port users will be sending to */
#define MAXBUFLEN 100
m a i n ( )
{
int sockfd;
struct sockaddr_in my_addr; /* my address information */
struct sockaddr_in their_addr; /* connector's address information */
int addr_len, numbytes;
char buf[MAXBUFLEN];
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1) {
perror("socket");
exit(1);
}
my_addr.sin_family = AF_INET; /* host byte order */
my_addr.sin_port = htons(MYPORT); /* short, network byte order */
my_a d d r.s i n_addr.s_addr = INADDR_ANY; /* auto-fill with my IP */
bzero(&(my_addr.sin_zero), 8); /* zero the rest of the struct */
if (bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr)) \
== -1) {
perror("bind ") ;
exit(1);
}
addr_len = sizeof(struct sockaddr);
if ((numbytes=recvfrom(sockfd, buf, MAXBUFLEN, 0, \
(struct sockaddr *)&their_addr, &addr_len)) == -1) {
perror( "recvfrom" );
exit(1) ;
}
printf("got packet from %s\n",inet_ntoa(their_addr. s i n _ a d d r ) ) ;
printf("packet is %d bytes long\n",numbytes);
buf[numbytes] = '\0';
printf("packet contains \"%s\"\n",buf);
close(sockfd);
}
下面是t a l k e r. c的代码:
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <netdb.h>
#include <sys/socket.h>
#include <sys/wait.h>
#define MYPORT 4950 /* the port users will be sending to */
int main(int argc, char *argv[])
{
int sockfd;
struct sockaddr_in their_addr; /* connector's address information */
struct hostent *he;
int numbytes;
if (argc != 3) {
fprintf (stderr, "usage: talker hostname message\n");
exit(1);
}
if ((he=gethostbyname(argv[1])) == NULL) { /* get the host info */
h e r r o r ( " g e t h o s t b y n a m e " ) ;
e x i t ( 1 ) ;
}
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) == -1) {
perror("socket") ;
exit(1);
}
their_addr.sin_family = AF_INET; /* host byte order */
their_addr.sin_port = htons(MYPORT); /* short, network byte order */
their_addr.sin_addr = *((struct in_addr *)he->h_addr);
bzero(&(their_addr.sin_zero), 8); /* zero the rest of the struct */
if ((numbytes=sendto(sockfd, argv[2], strlen(argv[2]), 0, \
(struct sockaddr *)&their_addr, sizeof(struct sockaddr))) == -1) {
perror ("s e n d t o");
exit(1);
}
printf("sent %d bytes to %s\n",numbytes,inet_ntoa(their_addr. s i n _ a d d r ) ) ;
close(sockfd);
return 0;
}
你可以在一台机器上运行l i s t e n e r程序,在另一台机器上运行t a l k e r程序,然后观察它们之
间的通信。
21.11 阻塞
当使用上面的listener程序时,此程序在等待直到一个数据包到来。这是因为它调用了
recvform(),如果没有数据,recvform ( )就一直阻塞,直到有数据到来。
很多函数都有阻塞。系统调用accept ( )阻塞,所有的类似recv*()的函数也可以阻塞。它们
之所以可以阻塞是因为系统内核允许它们阻塞。当你第一次创建一个套接口文件描述符时,系
统内核将它设置为可以阻塞。如果你不希望套接口阻塞,你可以使用系统调用fcntl():
#include <unistd.h>
#include <fcntl.h>
.
sockfd = socket(AF_INET, SOCK_STREAM, 0);
fcntl(sockfd, F_SETFL, O_NONBLOCK);
.
如果你设置为不阻塞,那么就得频繁地询问套接口以便检查有无信息到来。如果你试图读
取一个没有阻塞的套接口,同时它又没有数据,那么你将得到- 1。
询问套接口以检查有无信息得到来可能会占用太多的C P U时间。另一个可以使用的方法是
select()。
select()用于同步I / O多路复用。这个系统调用十分有用。考虑一下下面的情况:你是一个
服务器,你希望监听进入的连接,同时还一直从已有的连接中读取信息。
也许你认为可以使用一个accept()调用和几个recv()调用。但如果调用accept()阻塞了怎么
办?如果在这时你希望调用recv()接受数据呢?
系统调用select()使得你可以同时监视几个套接口。它可以告诉你哪一个套接口已经准备好
了以供读取,哪一个套接口已经可以写入。
下面是select()的用法:
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
int select(int numfds, fd_set *readfds, fd_set *writefds,
fd_set *exceptfds, struct timeval *timeout);
此函数监视几个文件描述符,特别是r e a d f d s、w r i t e f d s和e x c e p t f d s。如果你希望检查是否
可以从标准输入中和一些其他的套接口文件描述符s o c k f d中读取数据,只需把文件描述符0和
s o c k f d添加到r e a d f d s中。参数n u m f d s应该设置为最高的文件描述符的值加1。
当s e l e c t ( )返回时,r e a d f d s将会被修改以便反映你选择的那一个文件描述符已经准备好了以
供读取。你可以使用F D _ I S S E T ( )测试。
FD_ZERO(fd_set *set)—清除文件描述符集。
FD_SET(int fd, fd_set *set)—把fd 添加到文件描述符集中。
FD_CLR(int fd, fd_set *set)—把fd 从文件描述符中移走。
FD_ISSET(int fd, fd_set *set)—检测fd 是否在文件描述符集中。
数据结构t i m e v a l包含下面的字段:
struct timeval {
int tv_sec; /* seconds */
int tv_usec; /* microseconds */
} ;
把t v _ s e c设置成需要等待的时间秒数,t v _ u s e c设置成需要等待的微秒数。一秒中包括1 000
0 0 0 μ s 。下面的程序等待2 . 5 s:
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
#define STDIN 0 /* file descriptor for standard input */
m a i n ( )
{
struct timeval tv;
fd_set readfds;
t v.tv_sec = 2;
t v.tv_usec = 500000;
F D _ Z E R O ( & r e a d f d s ) ;
FD_SET(STDIN, &readfds);
/* don't care about writefds and exceptfds: */
select(STDIN+1, &readfds, NULL, NULL, &tv);
if (FD_ISSET(STDIN, &readfds))
printf("A key was pressed!\n");
else
printf( " Timed out.\n");
摘自:
Linux系统高级编程-china pub
posted on 2009-11-15 21:40 chatler 阅读(719) 评论(0)  编辑 收藏 引用 所属分类: Socket

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   博问   Chat2DB   管理


<2010年11月>
31123456
78910111213
14151617181920
21222324252627
2829301234
567891011

常用链接

留言簿(10)

随笔分类(307)

随笔档案(297)

algorithm

Books_Free_Online

C++

database

Linux

Linux shell

linux socket

misce

  • cloudward
  • 感觉这个博客还是不错,虽然做的东西和我不大相关,觉得看看还是有好处的

network

OSS

  • Google Android
  • Android is a software stack for mobile devices that includes an operating system, middleware and key applications. This early look at the Android SDK provides the tools and APIs necessary to begin developing applications on the Android platform using the Java programming language.
  • os161 file list

overall

搜索

  •  

最新评论

阅读排行榜

评论排行榜