TCP/IP网络编程系列之二(初级)

Posted 天狼鼠

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了TCP/IP网络编程系列之二(初级)相关的知识,希望对你有一定的参考价值。

套接字类型与协议设置

   我们先了解一下创建套接字的那个函数 int socket(int domain,int type,int protocol);成功时返回文件描述符,失败时返回-1.其中,domain是套接字使用中的协议族(Protocol Family)信息。type套接字类型里面的数据传输类型信息。protocol计算机通信中使用的协议信息。

协议族(Protocol Family)

协议族类型有:

    PE_INET          IPV4

    PE_INET6          IPV6

          PF_LOCAL        本地通信的UNIX协议族

      PF_PACKET        底层套接字的协议族

          PF_IPX         IPX Novell协议族

套接字类型(Type)

  套接字类型指的是套接字的数据传输方式,通过socket函数的第二个参数传递,只有这样才能决定创建的套接字的数据传输方式。决定了协议族并不能同时决定数据传输方式。也就是说socket函数第一个参数的PF_INET协议族存在多重数据传输方式。

第一种:面向连接的套接字(SOCK_STREAM)

  如果给第二个参数传递SOCK_STREAM,将创建面向连接的套接字。特点:

  • 套接字连接必须一一对性。
  • 可靠的、按序传递的、基于字节的面向连接的数据传输方式的套接字。

收发数据的套接字内部有缓冲(buffer),简言之就是字节数组。通过套接字传输的数据将保存到该数组中。因此,收到数据并不意味着马上调用read函数。只要不超过数组容量,则有可能数据填充满缓冲后通过一次read函数调用全部读完,也有可能分成多次调用read函数调用进行读取。也就是说,在面向连接的套接字中,read和write函数调用多次并无太大意义,所以说面向连接的套接字编程不存在数据边界。

如果套接字缓冲已满是否意味着数据丢失?首先调用read函数从缓冲区读取部分数据,因此,缓冲并不总是满的。但如果read函数读取的速度比接收数据的速度慢,则缓冲有可能被填满,此时套接字无法在接收数据,但即使这样也不会发生数据丢失。因为传输端套接字将停止传输。也就是说,面向连接的套接字会根据接收端的状态传输数据,如果传输出错还会提供重传服务。因此,面向连接的套接字除特殊的情况不会发生数据丢失。

第二种:面向消息的套接字(SOCK_DGRAM)

  如果给第二个参数传递SOCK_DGRAM,将创建面向消息的套接字。特点:

  • 不可靠的、不按序传递的、以数据的告诉传输为目的的套接字。
  • 传输的数据具有数据边界。

套接字协议信息

第三个参数主要用在同一协议族中存在多重传输方式相同的协议,数据传输方式相同,但协议不同。此时靠第三个参数具体指定协议信息。

简单的例子:

  要求在IPV4协议族中创建面向连接的套接字。其中在TCP/IP网络编程系列之一中创建服务端和客户端的代码,服务端的代码不需要改变,只改变客户端的代码,更改read函数的调用方式。

TCPClient代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h>

#define MAXSIZE 30
void Error_Handling(char* message);

int main(int argc,char* argv[])
{
    int sock;
    struct  sockaddr_in serverAddr;
    char message[MAXSIZE];
    int str_len=0;
    int idx=0,read_len=0;
    if(argc != 3)
    {
        printf("Usage : %s <IP> <Port> \n",argv[0]);
        exit(1);
    }
    sock = socket(PF_INET,SOCK_STREAM,0);
    if(sock == -1)
    {
        Error_Handling("sock() error");
    }
    memset(&serverAddr,0,sizeof(serverAddr));
    serverAddr.sin_family=AF_INET;
    serverAddr.sin_addr.s_addr=inet_addr(argv[1]);
    serverAddr.sin_port=htons(atoi(argv[2]));
    if(connect(sock,(struct sockaddr*)&serverAddr,sizeof(serverAddr))==-1)
    {
        Error_Handling("connect() error");
    }
   
    while(( read_len = read(sock,&message[idx++],1)))
    {
        if(read_len==-1)
        {
            Error_Handling("read() error\n");
        }
        str_len+=read_len;
    }
    printf("Message from server : %s \n",message);
    printf("Function read call count:%d \n",str_len);
    close(sock);
    return 0;
}

void Error_Handling(char *message)
{
    fputs(message,stderr);
    fputc(‘\n‘,stderr);
    exit(1);
}

 

以上是关于TCP/IP网络编程系列之二(初级)的主要内容,如果未能解决你的问题,请参考以下文章

自动化运维Python系列之Socket编程

linux网络编程系列-TCP/IP模型

AOP (面向切面编程) 系列之二

Java并发编程系列之二线程基础

Java并发编程系列之二十八:CompletionService

SRv6网络编程自学系列之二 | SR报文开销与承载效率