plainchant
  • Welcome PCT‘s Blog
  • Golang
    • golang基础
      • Go 语言 select 的实现原理
      • golang数字最大值
      • go-defer
      • Channel实现
      • go逃逸分析
      • Golang调度
  • Linux
  • Linux开发
    • 查看磁盘的UUID并挂载
    • Linux内核开发示例
    • 误删Linux内核后修复系统
    • linux补丁的创建和应用
    • Git常用命令
    • SystemV消息队列使用范例
    • ubuntu搭建全局代理
    • linux安装和配置
  • 嵌入式
    • 计算机为什么存在补码
    • 一种可靠串口协议
    • CRC校验算法
    • RasperryPi3 Ros系统安装(Debian)
  • 套接字编程
    • TCP建立连接过程分析
    • 线程的分发
    • TCP的并发处理epoll
    • TCP的并发处理select
    • 非阻塞TCP示例
    • 阻塞TCP示例
    • UDP组播
    • UDP广播
    • 非阻塞UDP示例(fcntl方式)
    • 非阻塞UDP示例
    • 局域网发现协议
    • socket通信机制浅析-前言
  • 交友网站
  • 区块链
    • Wasm虚拟机
      • wagon外部参数和内部参数的统一
      • Wagon实现log函数的第二种方法
      • Wagon实现一个log函数
      • go版本wasm解析器分析
      • 解析wasm二进制文件
      • ONT实现API的流程
      • Wasm工具安装使用
    • BCH
      • SLP代币协议
    • Cosmos
      • 区块链共识进化史
      • Tendermint 的区块构成
      • CoinEx 链 Gas 费指南
      • CoinEx交易类型收集
      • Cosmos简介和环境搭建
    • ETH
      • Geth命令详解
    • BTC
      • 助记词到地址
  • 算法
    • 动态规划
  • HTTP
    • URL 在浏览器被被输入到页面展现的过程中发生了什么
  • 运维后台
    • Docker学习笔记
  • 数据型应用系统设计
    • 数据密集型应用系统设计读书笔记
    • 数据编码与演化
      • Kafka配置
      • protobuf简介
    • MySQL
      • mysql安装和数据目录变更
      • 深入理解事务
      • MySQL事务问题验证
    • Redis
      • Redis缓存实现
      • Redis基本概念
由 GitBook 提供支持
在本页

这有帮助吗?

  1. 套接字编程

阻塞TCP示例

TCP在默认情况下就是阻塞的方式,recv会使线程挂起,这种方式在简单通信中经常使用。

服务端代码: 服务端代码中listen的数目表示并发监听数量,并不表示最大连接数量。另外程序只是做了个简单的demo,并没有做很好的重连机制,如果发送或者接收失败,recv返回0表示连接断开,次数应该做重连。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <arpa/inet.h>
#include <unistd.h> //usleep
#define checkError(ret) do{if(-1==ret){printf("[%d]err:%s\n", __LINE__, strerror(errno));exit(1);}}while(0)
int main(int argc, char const *argv[])
{
    printf("this is tcp demo\n");
    int iSocketFd = socket(AF_INET, SOCK_STREAM, 0);
    checkError(iSocketFd);


    int re = 1;
    checkError(setsockopt(iSocketFd, SOL_SOCKET, SO_REUSEADDR, &re, sizeof(re)));


    struct sockaddr_in server_addr;  
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET;  
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);        /*receive any address*/
    server_addr.sin_port = htons(7878);
    checkError(bind(iSocketFd, (struct sockaddr*)&server_addr, sizeof(server_addr)));


    checkError(listen(iSocketFd, 5));
    int iSockClient;
    struct sockaddr_in client_addr;
    memset(&client_addr, 0, sizeof(client_addr));
    socklen_t client_len = sizeof(client_addr);


reconnect:
    iSockClient = accept(iSocketFd, (struct sockaddr*)&client_addr, &client_len);
    checkError(iSockClient);    
    printf("client ipaddr:%s\n", inet_ntoa(client_addr.sin_addr));


    const char *aSend = "This is tcp server";
    char aRecv[2048] = {0}; 
    while(1)
    {
        printf("wait client data...\n");
        int irecv = recv(iSockClient, aRecv, sizeof(aRecv), 0);
        if(-1 == irecv){
            printf("recv err:%s\n", strerror(errno));
            if(errno == EAGAIN){
                usleep(100);
                continue;
            } else {
                exit(1);
            }
        } else if (0 == irecv){
            printf("disconnect with client\n");
            close(iSockClient);
            goto reconnect;
        }
        printf("recv client ip:%s, data:%s\n", inet_ntoa(client_addr.sin_addr), aRecv);
        checkError(send(iSockClient, aSend, strlen(aSend), 0));
        sleep(1);
    }
    return 0;
}

客户端代码: 客户端代码中并不需要bind,它的重连需要重新建立Socket描述字,因为TCP是双向通信,服务器断开后并没有将整个Socket断开,它只是处于半连接状态,按照《TCP/IP协议》的说法,此时还可以向服务端发送数据,但是服务端程序因为没有去监控此Socket,所以数据全部会丢失,想重连服务器就得将Socket关掉重建。

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
#include <arpa/inet.h>
#include <unistd.h> //usleep
#include <signal.h>
#define checkError(ret) do{if(-1==ret){printf("[%d]err:%s\n", __LINE__, strerror(errno));exit(1);}}while(0)
int main(int argc, char const *argv[])
{
    printf("this is a tcp client demo\n");
    signal(SIGPIPE, SIG_IGN);//ingnore signal interference
    int iSocketFd = 0;
reconnect: 
    iSocketFd = socket(AF_INET, SOCK_STREAM, 0);
    checkError(iSocketFd);


    int re = 1;
    checkError(setsockopt(iSocketFd, SOL_SOCKET, SO_REUSEADDR, &re, sizeof(re)));


    struct sockaddr_in server_addr;  
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET;  
    server_addr.sin_addr.s_addr = htonl(INADDR_ANY);        /*receive any address*/
    //server_addr.sin_addr.s_addr = inet_addr("10.128.118.43");
    server_addr.sin_port = htons(7878);
    while(1){
        printf("conncet to server...\n");
        if(-1 != connect(iSocketFd, (struct sockaddr *)&server_addr, sizeof(server_addr))){
            break;
        }   
        sleep(1);
    }


    char aRecv[2048] = {0}; 
    const char *aSend = "This is tcp client";
    while(1)
    {
        checkError(send(iSocketFd, aSend, strlen(aSend), 0));

        printf("wait server data...\n");
        int irecv = recv(iSocketFd, aRecv, sizeof(aRecv), 0);
        if(-1 == irecv){
            printf("recv err[%d]:%s\n", errno, strerror(errno));
            if(errno == EAGAIN){
                usleep(100);
                continue;
            } else {
                exit(1);
            }
        } else if (0 == irecv){
            printf("disconnect with server\n");
            close(iSocketFd);
            goto reconnect;
        }
        printf("server ip:%s, data:%s\n", inet_ntoa(server_addr.sin_addr), aRecv);
        sleep(1);
    }
    return 0;
}

================================================================== 上面代码运行起来应该是什么样的结果呢?下面是在同一台机器上运行的结果:

客户端:

this is a tcp client demo
wait server data...
server ip:0.0.0.0, data:This is tcp server

服务端:

this is tcp demo
client ipaddr:127.0.0.1
wait client data...
recv client ip:127.0.0.1, data:This is tcp client

注意到IP了么,Linux的网络Socket并没有出本机,或者说它可能没有走TCP/IP的标准流程,直接在内部做了转发,所以在本机通信,可以不必用本地Socket,直接用网络Socket即可。 但是这个特性需要将Socket的地址设成INADDR_ANY,如果改成下面形式就会走IP层了:

    //server_addr.sin_addr.s_addr = htonl(INADDR_ANY);        /*receive any address*/
    server_addr.sin_addr.s_addr = inet_addr("10.128.118.43");

输出如下:

this is a tcp client demo
wait server data...
server ip:10.128.118.43, data:This is tcp server
this is tcp demo
client ipaddr:10.128.118.43
wait client data...
recv client ip:10.128.118.43, data:This is tcp client
上一页非阻塞TCP示例下一页UDP组播

最后更新于4年前

这有帮助吗?