ios底层Socket编程理解

前言

Socket网络编程在任何一门编程语言中都很重要,而且socket底层是纯C语言,跨平台,了解并熟悉底层交互是提高自己编程水平重要的一步.环环在此稍加总结,如果有童鞋要面试还能用的上,结尾附有demo案例(IOS).

正文

  • 首先明确Socket在网络模型中哪里:是应用层与传输层之间的桥梁


    ios底层Socket编程理解_第1张图片
    image
  • 回顾一下网络模型: OSI七层网络模型:
    1.应用层.2.表示层.3.会话层.4.传输层.5.网络层.6.数据链路层.7.物理层
    TCP/IP四层网络模型:应用层.传输层.网络层.网络接入层

  • HTTP协议:属于应用层面向对象的协议(超文本传输协议),常基于TCP连接方式, 特点是:
    1.支持客户/服务端模式
    2.简单快捷灵活
    3.客户端发送的每次请求都需要服务器回送响应,请求结束后主动释放连接.俗称”短连接"

  • TCP协议:传输控制协议,提供面向连接.可靠的字节流服务,提供超时重发,丢弃重复数据,检验数据,流量控制等功能。在正式收发数据前,必须建立可靠的连接,也即:三次握手.

第一次握手:客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状态,等待服务器确认;
第二次握手:服务器收到syn包,必须确认客户的SYN(ack=j+1),同时自己也发送一个SYN包(syn=k),即SYN+ACK包,此时服务器进入SYN_RECV状态;
第三次握手:客户端收到服务器的SYN+ACK包,向服务器发送确认包ACK(ack=k+1),此包发送完毕,客户端和服 务器进入ESTABLISHED状态,完成三次握手。

  • UDP协议:用户数据报协议,面向非连接,不保证可靠性的数据传输服务,没有超时重发等机制,故而传输速度很快.

特点:它不与对方建立连接,而是直接就把数据包发送过去, UDP适用于一次只传送少量数据、对可靠性要求不高的应用环境。

  • Socket:又称”套接字”,应用程序通过”套接字”向网络发送请求或应答,它是一个针对TCP和UDP编程的接口,借助它建立TCP/UDP连接。socket连接就是所谓的长连接,理论上客户端和服务器端一旦建立起连接将不会主动断掉.

  • HTTP协议—Socket连接--TCP连接关系:

1.HTTP协议提供了封装或者显示数据的具体形式;
2.Socket连接提供了网络通信的能力;
3.TCP连接提供如何在网络中传输;
4.socket是纯C语言的,跨平台;
5.HTTP协议是基于socket的,底层使用的就是socket;
6.创建Socket连接时,可以指定使用的传输层协议(TCP或UDP),当使用TCP协议进行连接时,该Socket连接就是一个TCP连接。

  • TCP和UDP区别

1.基于连接和无连接
2.对系统资源要求(TCP较多,UDP较少)
3.UDP程序结构较简单
4.TCP是流模式,UDP是数据报模式
5.可靠性:TCP保证数据正确性,UDP可能丢包,不保证数据准确性

  • Socket通信流程图
    
ios底层Socket编程理解_第2张图片
image

以socket客户端编程为例:
0.导入头文件

#import 
#import 
#import 

1.创建socket

@implementation ViewController {
    int _clientSocket;
}

  /*
     1.AF_INET: ipv4 执行ip协议的版本
     2.SOCK_STREAM:指定Socket类型,面向连接的流式socket 传输层的协议
     3.IPPROTO_TCP:指定协议。 IPPROTO_TCP 传输方式TCP传输协议
     返回值 大于0 创建成功
     */
_clientSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);

2.建立连接(与服务器)

  /*
     终端里面 命令模拟服务器 netcat  nc -lk 12345
     参数一:套接字描述符
     参数二:指向数据结构sockaddr的指针,其中包括目的端口和IP地址
     参数三:参数二sockaddr的长度,可以通过sizeof(struct sockaddr)获得
     返回值 int -1失败 0 成功
  */
    struct sockaddr_in addr;
   /* 填写sockaddr_in结构*/
    addr.sin_family = AF_INET;
    addr.sin_port=htons(12345);
    addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    int connectResult= connect( _clientSocket, (const struct sockaddr *)&addr, sizeof(addr));

3.发送数据(到服务器)

   /*
      第一个参数指定发送端套接字描述符;
      第二个参数指明一个存放应用程式要发送数据的缓冲区;
      第三个参数指明实际要发送的数据的字符数;
      第四个参数一般置0。
      成功则返回实际传送出去的字符数,失败返回-1,
   */
      char * str = "itcast";
      ssize_t sendLen = send( _clientSocket, str, strlen(str), 0);

4.接送数据(从服务器)

     /*
      第一个参数socket
      第二个参数存放数据的缓冲区
      第三个参数缓冲区长度。
      第四个参数指定调用方式,一般置0
      返回值 接收成功的字符数
     */
      char *buf[1024];
      ssize_t recvLen = recv( _clientSocket, buf, sizeof(buf), 0);
      NSLog(@"---->%ld",recvLen);

5.关闭连接

       close( _clientSocket);

6.demo封装方法:

//建立连接
- (void)connectToServer:(NSString *)ip port:(int)port {
    _clientSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    struct sockaddr_in addr;
    /* 填写sockaddr_in结构*/
    addr.sin_family = AF_INET;
    addr.sin_port = htons(port);
    addr.sin_addr.s_addr = inet_addr(ip.UTF8String);
    int connectResult = connect(_clientSocket, (const struct sockaddr *)&addr, sizeof(addr));
    if (connectResult == 0) {
        NSLog(@"conn ok");
    }
}
//发送数据并等待返回数据
- (NSString *)sentAndRecv:(NSString *)msg {
   const char *str = msg.UTF8String;
//发消息
    ssize_t sendLen = send(_clientSocket, str, strlen(str), 0);
//收消息
    char *buf[1024];
    ssize_t recvLen = recv(_clientSocket, buf, sizeof(buf), 0);
    NSString *recvStr = [[NSString alloc] initWithBytes:buf length:recvLen encoding:NSUTF8StringEncoding];
    return recvStr;
}

案例效果图:

ios底层Socket编程理解_第3张图片
image

案例一:多线程实现服务端与客户端简单的交互,我的demo地址:
服务端: https://github.com/zhonghphuan/ServerSocket.git
客户端: https://github.com/zhonghphuan/ClientSocket.git

案例二:利用Socket发送HTTP格式的请求并且通过浏览器监控:
https://github.com/FieldsOfHope/Socket_Interactive.git

如果有其他问题,请私信我:[email protected]

github Demo已更新【20190228】:域名与IP问题转换
感谢@十二月青 提出的问题。


ios底层Socket编程理解_第4张图片
图片.png

你可能感兴趣的:(ios底层Socket编程理解)