Chinaunix首页 | 论坛 | 博客
  • 博客访问: 4842392
  • 博文数量: 930
  • 博客积分: 12070
  • 博客等级: 上将
  • 技术积分: 11448
  • 用 户 组: 普通用户
  • 注册时间: 2008-08-15 16:57
文章分类

全部博文(930)

文章存档

2011年(60)

2010年(220)

2009年(371)

2008年(279)

分类: LINUX

2010-08-19 17:42:01

 

使用套接字出了可以实现不同网络间主机的通信外,还可以实现同一主机的不同进程间的通信,且建立的通信时双向的通信。这里所指的使用套接字实现进程间通信,是由将通信域指定为PF_UNIX来实现的。下面是socket函数的具体消息定义,该函数的形式如下:

                            int socket(int domain, int type, int protocol);

socket函数中的domain用于指定通信域,domain参数取PF_UNIX时,表示创建UNIX域的套接字。使用PF_UNIX域的套接字可以实现同一机器上的不同进程间的通信。

调用bind函数现实了套接字与地址(这里是文件名)的绑定。bing函数的具体信息如下:

                             int bind(int sockfd, const struct sockaddr *my_addr, socklen_t addrlen);

其参数my_addr为指向结构体sockaddr_un的指针,该结构体的定义如下:

struct sockaddr_un{

          sa_family_t   sun_family; /* PF_UNIX或AF_UNIX*/

           char sun_path[108]; /*路径名*/

};

在该结构体中,sun_family 为AF_UNIX,sun_path是套接字在文件系统中的路径名。

1.1服务器端实现

程序serv.c为使用套接字在UNIX域内实现进程间通信的服务器端程序。首先,程序通过调用socket函数,建立了监听连接的套接字,然后调用bind函数,将套接字与地址信息关联起来。调用listen函数实现对该端口的监听,当有连接请求时,通过调用accept函数建立与客户端的连接,最后,通过调用read函数来读取客户机发送过来的消息,当然也可以使用recv函数实现相同的功能。serv.c具体代码如下:

//UNIX域通信代码实例,服务器端

#include

#include

#incude

#inlcude

//定义用于通信的文件名

#define UNIX_DOMAIN "/tmp/UNIX.domain"

int main()

{

int listen_fd;

int com_fd;

int ret;

int i;

static char recv_buf[1024];

int len;

struct sockaddr_un clt_addr;

socklen_t clt_addr_len;

struct sockaddr_un srv_addr;

//创建用于通信的套接字,通信域为UNIX通信域

listen_fd = socket(PF_UNIX, SOCK_STREAM, 0);

if(listen_fd<0){

perror("cannot creat listening socket");

return 1;

}

//设置服务器地址参数

srv_addr.sun_family = AF_UNIX;

strncpy(srv_addr.sun_path,UNIX_DOMAIN,sizeof(srv_addr.sun_path)-1);

unlink(UNIX_DOMAIN);

//绑定套接字与服务器地址信息

ret = bind(listen_fd,(struct sockaddr *)&srv_addr,sizeof(serv_addr));

if(ret ==-1){

perror("cannot bind server socket");

close(listen_fd);

unlink(UNIX_DOMAIN);

return 1;

}

//对套接字进行监听

ret = listen(listen_fd,1);

if(ret == -1){

perror("cannot listen the client connect request");

close(listen_fd);

unlink(UNIX_DOMAIN);

return 1;

}

printf("begin listening");

//当有连接请求时,调用accept函数建立服务器与客户机之间的连接

len = sizeof(clt_addr);

com_fd = accept(listen_fd,(sruct sockaddr*)&clt_addr,&len);

if(com_fd < 0){

perror("cannot accept client connect request");

close(listen_fd);

unlink(UNIX_DOMAIN);

return 1;

}

//读取并输出客户端发送过来的连接信息

printf("\n========info=========\n");

for(i=0;i<4;i++){

memset(recv_buf,0 ,1024);

int num = read(com_fd,recv_buf,sizeof(recv_buf);

printf("message from client(%d)):%s\n",num,recv_buf);

}

close(com_fd);

close(listen_fd);

unlink(UNIX_DOMAIN);

return 0;

}

1.2 客户端实现

clt.c为使用套接字在UNIX域内实现进程间通信的客户端程序。相比服务器端的程序,客户端较为简单程序首先通过调用socket函数建立通信所需的套接字,然后,调用connect函数来连接服务器,在成功建立连接后,通过调用write函数向服务器发送指定的消息,具体代码如下:

//clt.c UNIX域通信代码示例,客户端

#include

#include

#incude

#include

#define UNIX_DOMAIN "/tmp/UNIX.domain"

int main()

{

int connect_fd;

int ret;

int snd_buf[1024];

int i;

static struct sockaddr_un srv_addr;

connect_fd = socket(PF_UNIX,SOCK_STREAM, 0);

if (connect_fd <0){

perror("cannnot creat communication socket");

return 1;

}

srv_addr.sun_family = AF_UNIX;

strcpy(&srv_addr.sun_path, UNIX_DOMAIN);

ret = connect(connect_fd,(struct sockaddr*)&srv_addr,sizeof(srv_addr);

if(ret == -1){

perror("cannot connect to the server");

return 1;

}

memset(snd_buf, 0, 1024);

strcpy(snd_buf,"client");

for(i=0;i<4;i++)

write(connect_fd,snd_buf,sizeof(snd_buf);

close(connect_fd);

return 0;

}

1.3程序运行结果分析

使用gcc编译srv.c和clt.c,获得名为srv和clt的可执行文件。先执行srv程序,然后运行clt程序,具体输出如下:

[program@localhost charter13]$ gcc clt.c -o clt
[program@localhost charter13]$ gcc -o srv srv.c

[program@localhost charter13]$ ./srv &
[1] 13450
[program@localhost charter13]$ ./clt &
[2] 13451
[program@localhost charter13]$
=====info=====
Message from client (1024)) :message from client
Message from client (1024)) :message from client
Message from client (1024)) :message from client
Message from client (1024)) :message from client

[1]- Done                    ./srv
[2]+ Done                    ./clt
[program@localhost charter13]$

当运行srv程序后,该程序将处于监听状态。这时,可以通过netstat命令查看程序与你想那个情况:

netstat -an|grep /tmp

阅读(1373) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~