socket 编程

网络编程:结构体struct sockaddr和struct sockaddr_in对这两个结构体进行解读。
在linux环境下,结构体struct sockaddr在/usr/include/linux/socket.h中定义,具体如下:
typedef unsigned short sa_family_t;
struct sockaddr {
        sa_family_t     sa_family;    /* address family, AF_xxx       */
        char            sa_data[14];    /* 14 bytes of protocol address */

在linux环境下,结构体struct sockaddr_in在/usr/include/netinet/in.h中定义,具体如下:
/* Structure describing an Internet socket address. */
struct sockaddr_in
{
    __SOCKADDR_COMMON (sin_);
    in_port_t sin_port;                     /* Port number. */
    struct in_addr sin_addr;            /* Internet address. */

    /* Pad to size of `struct sockaddr'. */
    unsigned char sin_zero[sizeof (struct sockaddr) -
                           __SOCKADDR_COMMON_SIZE -
                           sizeof (in_port_t) -
                           sizeof (struct in_addr)];     
                           /* 字符数组sin_zero[8]的存在是为了保证结构体struct sockaddr_in的大小和结构体struct sockaddr的大小相等 */
};
struct sockaddr是通用的套接字地址,而struct sockaddr_in则是internet环境下套接字的地址形式,二者长度一样,都是16个字节。二者是并列结构,指向sockaddr_in结构的指针也可以指向sockaddr。一般情况下,需要把sockaddr_in结构强制转换成sockaddr结构再传入系统调用函数中。

下面是struct sockaddr_in中用到两个数据类型,具体定义如下:
/* Type to represent a port. */
typedef uint16_t in_port_t;
 

struct in_addr其实就是32位IP地址
struct in_addr {
        unsigned long s_addr;
};

BSD网络软件中包含了两个函数,用来在二进制地址格式和点分十进制字符串格式之间相互转换,但是这两个函数仅仅支持IPv4。
       in_addr_t inet_addr(const char *cp);
       char *inet_ntoa(struct in_addr in);
功能相似的两个函数同时支持IPv4和IPv6
       const char *inet_ntop(int domain, const void *addr, char *str, socklen_t size);
       int inet_pton(int domain, const char *str, void *addr);

通常的用法是:
int sockfd;

struct sockaddr_in my_addr;

/*建立socket 连接*/

sockfd = socket(AF_INET, SOCK_STREAM, 0); 
/*建立 sockaddr_in 结构体相关参数*/
my_addr.sin_family = AF_INET; /* 主机字节序 */
my_addr.sin_port = htons(MYPORT); /* short, 网络字节序 */

my_addr.sin_addr.s_addr = inet_addr("192.168.0.1");
或者: my_addr.sin_addr.s_addr=INADDR_ANY;//(将本机器上的long数据转化为网络上的long数据)和任何主机通信 ,
    //INADDR_ANY 表示可以接收任意IP地址的数据,即绑定到所有的IP
bzero(&(my_addr.sin_zero), 8); /* zero the rest of the struct */
//memset(&my_addr.sin_zero, 0, 8);
/*绑定bind 函数*/

bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr));

   /*调用listen  函数*/
if(listen(socketfd,BACKLOG==-1))
{
     perror("listen error");
exit(1);
}

/*调用accetp 函数*/
sin_size=sizeof(struct sockaddr_in); // accept is wait client connect  服务器阻塞,直到客户程序建立连接 */
if((clientfd=accept(socketfd,(struct sockaddr *)(&client_socketaddr),&sin_size))==-1) //sin_size=sizeof(struct sockaddr_in); 

perror("accept error");
exit(1);
}
/*调用 recv  函数*/
//if(recvbytes=recv(socketfd,buf,MAX_BUF_SIZE,0)==-1)
if(recvbytes=recv(clientfd,buf,MAX_BUF_SIZE,0)==-1)// not is  socketfd

perror("recv error");
exit(1);

printf("rececie a connection :%s\n",buf);
close(socketfd); /* 结束通讯 */ 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值