代码改变世界

Linux Socket学习--域和套接口简介

2012-08-16 10:21  Rollen Holt  阅读(2419)  评论(0编辑  收藏  举报

       套接口创建后,就如同一个文件描述符,我们可以使用同样的IO函数进行读写,关闭操作。其实,和引用一个已经打开的文件一样,套接口也是通过文件描述符来引用的,而且两者的文件描述符共享一个“数字空间”,比如说不能既打开一个文件描述符为4的套接口,又打开一个文件描述符为4的文件。

       套接口和已经打开的文件的区别:

       1.不能在套接口上调用函数lseek()当然对于管道也不能调用这个函数。

       2.套接口可以和网络地址关联,但是文件和管道却不可以。

       3.套接口有很多可以通过ioctl()进行查询和设置的选项。

       4.套接口必须在正确的状态写才能进行输入输出,但是已经打开的文件可以在任何的时候进行读写操作。

       调用open()函数打开一个新文件,Linux内核会返回一个当前系统可用的最小的文件描述符。比如如果关闭了标准输入(文件描述符0),又立即打开一个新文件,那么新文件的文件描述符就是0.

        另外要注意的是Linux内核在分配文件描述符的时候,并不区分这个描述符是分配给套接口还是给已经打开的文件的。      

        我们知道,使用int pipe(int fileds[2]);函数可以创建管道,如果这个函数调用成功,内核会返回两个文件描述符,fileds[0]表示管道读出端的文件描述符,fileds[1]表示管道写入端的文件描述符。所以可以看出管道是单向的,任何的反方向操作会导致错误。

        但是套接口不同,他可以进行双向通信。这个也是他和管道的根本差异。

       创建一个套接口使用以下的函数:

 

#include <sys/types.h>
#include <sys/socket.h>

int socketpair(int domain,int type,int protocol,int sv[2]);

 

下面我们来展示一个socketpair的例子:

 

#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>

int main(int argc, char const *argv[])
{
    int state;
    int s[2];

    state=socketpair(AF_LOCAL,SOCK_STREAM,0,s);
    if (state== -1)
    {
    	fprintf(stderr, "an error %s has happend.\n", strerror(errno);
    	return 1;
    }

    printf("s[0]=%d\n", s[0]);
    printf("s[1]=%d\n", s[1]);
    

	return 0;
}

 

    其中AF_LOCAL套接口又被称为是unix套接口,所以使用AF_UNIX和使用AF_LOCAL都是一样的。不过建议使用AF_LOCAL。

    下面我们来展示一下使用套接口实现I/O

    ………………UNDONE(此处代码待添加)

     我们来看看如何关闭套接口

     当我们从管道读取输入的时候,读取进程受到文件结束标识以后就知道没有后续文件了,因为文件描述符是由写进程关闭管道的写段的时候发送的。套接口的关闭类似,当在套接口发送端关闭后,接受端就会受到文件结束标识,但是这种关闭方式不适用所有的情况。close终止了数据传送的两个方向:读和写,由于TCP是全双工的,在很多的情况下,本地进程要通知远程端点已经完成了数据的发送工作,但是仍然等待从远程端点接受数据,在这种情况下,我们需要对套接口进行半关闭操作,就需要用到shutdown函数:

#include <sys/socket.h>

int shutdown(int s, int how);

其中s是需要进行半关闭的套接口,需要进行关闭的方式是how,参数how的取值请看下表:

描述
0 SHUT_RD 进程不能对指定套接口进行读操作
1 SHUT_WR 进行不能对指定套接口进行写操作
2 SHUT_RDWR 进程不能对指定套接口进行读写操作

大家可以看出,当how参数的取值为2的时候,shutdown函数和close函数一样了。下面举一个例子:

int state;
int s;

state=shutdown(s,SHUT_WR);
if(z == -1){
	perror("shundown...");
}

下面我们来看看如何处理复制套接口的情况:

   如果调用函数dup或者dup2的话,那么只有最后一个close调用才可以关闭套接口,系统这样做,是因为他认为其他复制的文件描述符仍然在使用中,如下所示:

   int s;   //已经存在的套接口

  int d;    //被复制的套接口

  d=dup(s); //复制套接口

  close(s) // 套接口并未被关闭

  close(d) //套接口被关闭

在上面的例子中,第一个close函数调用并没有关闭套接口,其实无论先关闭s还是先关闭d,没任何区别。而只有当最后一个文件描述符被关闭的时候,套接口才是被真正的关闭了。但是如果使用shutdown函数的话,就没有这个问题。:

int s

int d;

d=dup(s);

shutdown(s,SHUT_RDWR);

即使套接口s仍然通过文件描述符d打开,shutdown函数仍然立即将套接口彻底关闭。

建议:总是使用shutdown函数来代替close函数。

当调用shutdown函数的时候,如果给定的套接口不是有效的文件描述符,返回EBADF错误。

如果给定的文件描述符不是套接口,返回ENOTSOCK错误

如果套接口出于非连接状态,返回ENOTCONN错误。

注意:即便是使用参数为SHUT_RDWR的时候,函数shutdown也不会释放文件描述符,在使用close释放文件描述符之前,他一直可用。