客户端服务端交互实现

news2025/1/12 3:59:04

问题

客户端业务逻辑如何实现?

与服务设备具体交互细节如何设计?

客户端业务逻辑实现

用户输入处理

  • 字符串空格处理,分割获取命令与参数

服务信息处理

  • 字符串预处理,分割获取服务命令
  • 存储服务命令与设备地址之间的映射(命令字符串 => 地址字符串)

客户端业务逻辑实现 - 用户输入处理

 

服务端逻辑实现

查询消息处理

  • 接收广播,并回复UDP消息

服务命令处理

  • 接收 TCP 连接,通过 请求-响应 的模式进行服务

UDP响应模块设计

 

服务模块设计

TCP 响应模块设计

客户端响应接收

 

客户端服务端交互实现

main.c

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/socket.h>
#include "utility.h"
#include "addr_mgr.h"
#include "udp_point.h"
#include "tcp_client.h"
#include "type_def.h"

#define BUF_SIZE    64
#define DESC_SIZE    32
#define ADDR_SIZE    16
#define USAGE_SIZE   256

#define DIM(a)      sizeof((a)) / sizeof((*a))

typedef struct 
{
    const char* cmd;
    void (*handler)(const char*);    
} Handler;

static int GetCharCount(const char* s, int c)
{
    int ret = 0;
    
    if(s)
    {
        while(*s)
        {
            if(*s == c)
            {
                ret++;
            }
            
            s++;
        }
    }
    
    return ret;
}

static void ParseCommand(const char* s)
{
    const char* desc = s;
    const char* addr = desc + DESC_SIZE;
    const char* usage = addr + ADDR_SIZE;
    int count = 0;
    int cnt = 0;
    char** arg = NULL;
    char** cmd = NULL;
    int r = 0;
    
    printf("desc: %s\n", desc);
    printf("addr: %s\n", addr);
    
    count = GetCharCount(usage, '\n');
    arg = Malloc2d(char, count, BUF_SIZE);
    
    if(arg && (count > 0))
    {
        r = DivideByChar(usage, '\n', arg, count, BUF_SIZE);
        
        for(int i = 0; i < r; i++)
        {   
            count = GetCharCount(arg[i], ' ') + 1;
            cmd = Malloc2d(char, count, BUF_SIZE);
            
            if(cmd && (count > 0))
            {
                cnt = DivideByChar(arg[i], ' ', cmd, count, BUF_SIZE);

                for(int j = 1; j < cnt; j++)
                {
                    AddrMgr_Add(cmd[j], addr);
                    printf("%s %s\n", cmd[j], addr);
                }
            }
            
            Free2d(cmd);
        }
    }
    
    Free2d(arg);
}

void Query_Handler(const char* s)
{   
    UdpPoint* point = UdpPoint_New(8888);
    Message* msg = NULL;
    int brd = 1;
    char* remote = "255.255.255.255";
    int port = 9999;
    int i = 0;
    
    if(point)
    {
        UdpPoint_SetOpt(point, SOL_SOCKET, SO_BROADCAST, &brd, sizeof(brd));
        msg = Message_New(TYPE_QUERY, 0, 0, 0, NULL, 0);
        
        if(msg)
        {
            UdpPoint_SendMsg(point, msg, remote, port);
            
            free(msg);
            msg = NULL;
        }
        
        while(i < 3)
        {
            if(UdpPoint_Available(point) > 0)
            {
                msg = UdpPoint_RecvMsg(point, NULL, NULL);
                
                if(msg)
                {
                    if(msg->type == TYPE_RESPONSE)
                    {
                        printf("Find service!\n");
                        
                        ParseCommand(msg->payload);
                        
                        i = 0;
                        free(msg);
                        msg = NULL;
                    }
                    else if(msg->type == TYPE_ERROR)
                    {
                        printf("Can NOT find service!\n");
                    }
                }
                
            }
            else
            {
                sleep(1);
                i++;
            }
        }
    }
    
    UdpPoint_Del(point);
}

void Touch_Handler(const char* s)
{   
    if(s && *s)
    {
        TcpClient* client = TcpClient_New();
        char* addr = AddrMgr_Find(s);
        Message* msg = NULL;
        
        if(client && addr && TcpClient_Connect(client, addr, 8888))
        {
            msg = Message_New(TYPE_TOUCH, 0, 0, 0, s, strlen(s) + 1);
            
            if(msg)
            {
                TcpClient_SendMsg(client, msg);
                
                free(msg);
                msg = NULL;
                
                msg = TcpClient_RecvMsg(client);
                
                if(msg && (msg->type == TYPE_RESPONSE))
                {
                    printf("%s\n", msg->payload);
                    free(msg);
                    msg = NULL;
                }
            }   
        }
        
        TcpClient_Del(client);
    }
}

Handler g_handler[] = {
    {"query", Query_Handler}, 
    {"touch", Touch_Handler},
};

int main(void)
{
    char line[BUF_SIZE] = {0};
    char** arg = NULL;
    int r = 0;
    
    printf("<<<< This is client demo >>>>\n");
    
    arg = Malloc2d(char, 2, BUF_SIZE);
    
    while(arg)
    {
        fgets(line, sizeof(line), stdin);
        line[strlen(line) - 1] = 0; 
        
        if(*line)
        {
            r = DivideByChar(line, ' ', arg, 2, BUF_SIZE);
            
            for(int i = 0; (i < DIM(g_handler) && (r > 0)); i++)
            {
                if(strcmp(arg[0], g_handler[i].cmd) == 0)
                {
                    g_handler[i].handler(arg[1]);
                    break;
                }
            }
        }
    }
    
    Free2d(arg);
    
    return 0;    
}

response_task.c

#include <stdio.h>
#include "response_task.h"
#include "udp_point.h"
#include "type_def.h"

#define DESC_SIZE    32
#define ADDR_SIZE    16
#define USAGE_SIZE   256

void* Response_Task(const char* arg)
{
    UdpPoint* point = NULL;
    Message* msg = NULL;
    char remote[16] = {0};
    int port = 0;

    point = UdpPoint_New(9999);

    if(point)
    {
        printf("point = 0x%X\n", point);

        while(1)
        {
            msg = UdpPoint_RecvMsg(point, remote, &port);

            if(msg && msg->type == TYPE_QUERY)
            {
                free(msg);
                msg = NULL;
                
                msg = Message_New(TYPE_RESPONSE, 0, 0, 0, NULL, DESC_SIZE + ADDR_SIZE + USAGE_SIZE);

                if(msg)
                {
                    strncpy(msg->payload, Service_GetDesc(), DESC_SIZE);
                    strncpy(msg->payload + DESC_SIZE, Wifi_IpAddr(), ADDR_SIZE);
                    strncpy(msg->payload + DESC_SIZE + ADDR_SIZE, Service_GetUsage(), USAGE_SIZE);
                    
                    UdpPoint_SendMsg(point, msg, remote, port);
                    
                    free(msg);
                    msg = NULL;
                }
                else
                {
                    Message m = {TYPE_ERROR};
                    
                    UdpPoint_SendMsg(point, &m, remote, port);
                }
                
            }
            else
            {
                Message m = {TYPE_ERROR};
                    
                UdpPoint_SendMsg(point, &m, remote, port);
            }
        }

        UdpPoint_Del(point);
    }

    return NULL;
}

local_service.h

#ifndef LOCAL_SERVICE_H
#define LOCAL_SERVICE_H

typedef struct
{
	float    illumination;
	float    humidity;
	float    temperature;
	int      light;
} SvrData;

void Service_Init(void);
const char* Service_GetDesc(void);
const char* Service_GetUsage(void);
SvrData Service_GetData(void);
int Service_SetLight(int on);

#endif

local_service.c

#include "local_service.h"

void Service_Init(void)
{
    
}

const char* Service_GetDesc(void)
{
    return "Environment Service";
}

const char* Service_GetUsage(void)
{
    return "Illumination: Ill_Get\n"
           "Temperature: Tem_Get\n"
           "Humidity: Hum_Get\n"
           "Light: Lig_Get Lig_Set_On Lig_Set_Off\n";
}

SvrData Service_GetData(void)
{
    SvrData ret = {188, 0.33, 35.6, 1};

    return ret;
}

int Service_SetLight(int on)
{
    int ret = 1;

    printf("set light: %d\n", on);

    return ret;
}

service_task.c

#include <stdio.h>
#include "stdlib.h"
#include <string.h>
#include "service_task.h"
#include "tcp_client.h"
#include "tcp_server.h"
#include "local_service.h"
#include "type_def.h"

typedef struct 
{
    const char* cmd;
    void* data;
    char* (*handler)(void*);    
} Handler;

static char* FormatNumber(float num)
{
    char* ret = (char*)malloc(16);
    
    snprintf(ret, 16, "%.2f", num);
    
    return ret;
}

static char* Ill_Get_Handler(void* data)
{
    return FormatNumber(Service_GetData().illumination);
}

static char* Tem_Get_Handler(void* data)
{
    return FormatNumber(Service_GetData().temperature);
}

static char* Hum_Get_Handler(void* data)
{
    return FormatNumber(Service_GetData().humidity);
}

static char* Lig_Get_Handler(void* data)
{
    char* ret = (char*)malloc(4);
    
    if(Service_GetData().light)
    {
        strcpy(ret, "on");
    }
    else
    {
        strcpy(ret, "off");
    }
    
    return ret;
}

static char* Lig_Set_Handler(void* data)
{
    char* ret = (char*)malloc(4);
    
    Service_SetLight((int)data);
    
    if(data)
    {
        strcpy(ret, "on");
    }
    else
    {
        strcpy(ret, "off");
    }
    
    return ret;
}

static Handler g_handler[] = 
{
    {"Ill_Get", NULL, Ill_Get_Handler},
    {"Tem_Get", NULL, Tem_Get_Handler},
    {"Hum_Get", NULL, Hum_Get_Handler},
    {"Lig_Get", NULL, Lig_Get_Handler},
    {"Lig_Set_On", (void*)1, Lig_Set_Handler},
    {"Lig_Set_Off", (void*)0, Lig_Set_Handler},
};

static int g_handler_size = sizeof(g_handler) / sizeof(*g_handler);

static void Server_Listener_Handler(TcpClient* client, int evt)
{
    if(evt == EVT_COON)
    {
        printf("a client connect\n");
    }
    else if(evt == EVT_DATA)
    {
        Message* msg = NULL;
        char* s = NULL;

        msg = TcpClient_RecvMsg(client);

        if(msg && (msg->type == TYPE_TOUCH))
        {
            printf("service type = %s\n", msg->payload);
        
            for(int i = 0; i < g_handler_size; i++)
            {
                if(strcmp(msg->payload, g_handler[i].cmd) == 0)
                {
                    s = g_handler[i].handler(g_handler[i].data);
                    
                    break;
                }
            }

            free(msg);
            msg = NULL;
            
            if(s)
            {
                msg = Message_New(TYPE_RESPONSE, 0, 0, 0, s, strlen(s) + 1);
                free(s);
                
                if(msg)
                {
                    TcpClient_SendMsg(client, msg);
                    free(msg);
                }
                else
                {
                    Message m = {TYPE_ERROR};
                    
                    TcpClient_SendMsg(client, &m);
                }
                
                free(s);
            }
        }
        else
        {
            const char* message = "Invalid touch request";
            
            free(msg);
            
            msg = Message_New(TYPE_RESPONSE, 0, 0, 0, message, strlen(message) + 1);
            TcpClient_SendMsg(client, &m);
        }
    }
    else if(evt == EVT_CLOSE)
    {
        printf("a client left\n");
    }

    return NULL;
}

void* Service_Task(const char* arg)
{
    TcpServer* server = NULL;

    server = TcpServer_New();

    if(server)
    {
        printf("server = 0x%X\n", server);

        Service_Init();
        TcpServer_SetListener(server, Server_Listener_Handler);
        TcpServer_Start(server, 8888, 5);
        TcpServer_DoWork(server);

        TcpServer_Del(server);
    }

    return NULL;
}

实验结果如下图所示

 

课后思考

服务模块如何获取真实环境信息?

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/112492.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Java Socket实现NIO通信

文章目录一.简单介绍通道&#xff08;Channel&#xff09;多路复用器&#xff08;Selector&#xff09;二.代码实现.客户端服务端运行结果一.简单介绍 NIO 很多人也称之为 Non-block I/O&#xff0c;即非阻塞 I/O&#xff0c;因为这样叫&#xff0c;更能体现它的特点。 为什么…

kdump功能

kdump功能前言1 kdump流程2 kdump配置2.1 kexec、makedumpfile编译2.2 系统内核2.3 捕获内核3 kdump测试4 kdump的不足前言 kdump 是一种先进的基于 kexec 的内核崩溃转储机制。当系统崩溃时&#xff0c;kdump会将内存导出为vmcore保存到磁盘。 在kernel1运行的时候&#xff…

RabbitMQ——延迟队列

目录 一、延迟队列的应用场景 1. 场景&#xff1a;"订单下单成功后&#xff0c;15分钟未支付自动取消" ① 传统处理超时订单 ② RabbitMQ延时队列方案 二、延迟队列中的消息投递和消息消费 1.TTL 和 DLX ① TTL ② DLX和死信队列 ③ 延迟队列 ④ 开发步骤 …

spring mvc 通过异常封装 验证 方法

正常情况 我们先演示一下正常情况下我们验证的方法。 首先定义一个LoginBean Data public class LoginBean {// Blank 不允许保存空格&#xff0c;空格不算内容NotBlank(message "用户名不能为空")String username;// Empty 允许保留空格&#xff0c;是空格也算内容…

【Spring】核心部分之AOP:通过列举代码例子,从底层刨析,深入源码,轻轻松松理解Spring的核心AOP,AOP有这一篇足以

AOP基本概念基本原理专业术语案例演示基于注解&#xff08;重点&#xff09;基于配置文件基本概念 面向切面编程&#xff0c;也叫面向方面编程&#xff0c;利用aop可以对业务逻辑的各个部分进行隔离&#xff0c;从而使得业务逻辑各个部分之间降低耦合&#xff0c;提高程序的可…

AQS 对资源的共享方式

AQS 定义两种资源共享方式 1) Exclusive&#xff08;独占&#xff09; 只有一个线程能执行&#xff0c;如 ReentrantLock。又可分为公平锁和非公平锁,ReentrantLock 同时支持两种锁,下面以 ReentrantLock 对这两种锁的定义做介绍&#xff1a; 下面来看 ReentrantLock 中相关的…

概要设计说明书(GB8567——88)基于协同的在线表格forture-sheet

概要设计说明书 1引言 1.1编写目的 为了帮助用户更好的了解和使用本在线表格&#xff0c;提高用户与软件的亲和度。 用户手册描述配置和使用改在线表格&#xff0c;以及该软件使用过程中应该注意的一下问题。 1.2背景 说明&#xff1a; 本用户手册所描述的软件系统的名称…

医疗检测数据存储管理系统

摘要 医疗信息化的迅速发展导致了医疗数据的指数型增长&#xff0c;医疗检测数据存储管理系统给医院现有信息系统带了巨大的压力。一方面,随着各种非结构化数据的不断涌现&#xff0c;现有的医疗信息系统在存储空间&#xff0c;存储速度、存储结构上达不到医疗检测数据的要求,不…

Github惊现神作,这份算法宝典让你横扫各大厂算法面试题

时间飞逝&#xff0c;转眼间毕业七年多&#xff0c;从事 Java 开发也六年了。我在想&#xff0c;也是时候将自己的 Java 整理成一套体系。 这一次的知识体系面试题涉及到 Java 知识部分、性能优化、微服务、并发编程、开源框架、分布式等多个方面的知识点。 写这一套 Java 面试…

使用Docker搭建Nacos的持久化和集群部署

1. 准备 1.1 mysql安装 下载镜像 docker pull mysql/mysql-server:5.7 在宿主机中相关目录&#xff0c;用于挂载容器的相关数据 mkdir -p /data/mysql/{conf,data} 编写my.cnf配置文件&#xff0c;在/data/mysql/conf目录中 (或下载 直接上传即可) my.cnf.txt - 蓝奏云 / …

【考研加油】所有上岸的考研人都有一个共同的特点,就是他们都参加考试了。2023考研加油。

声明:为 2023考研的朋友加油! 2023考研加油 今明两天,将是大部分2023考研人,真正“上战场”的时候。 我想,只有经历过的人,才能对这一历程,感同身受吧! 为你们加油! 以下是在QQ空间看到的一组图,与你们共勉。 距考研还有____天! 确定目标院校中…跨考又能如何?…

阿里人在Github分享的Spring Cloud全栈笔记,你想象不到有多全

微服务到底是什么 微服务到底是什么&#xff0c;一直众说纷纭&#xff0c;我们只知道各大企业纷纷追捧和实践微服务架构&#xff0c;有的项目可能使用了Spring Cloud就算是使用微服务了&#xff0c;然后说微服务就是Spring Cloud&#xff0c;有的系统可能越做越像SOA&#xff…

RV1126笔记十六:吸烟行为检测及部署<四>

若该文为原创文章&#xff0c;转载请注明原文出处。 转换成onnx模型(windows) 一、查看pt文件 准备好训练好的pt文件,可以用Netron打开看看大概长啥样: 二、模型转换 主要的目的是想把训练好的pt文件转成onnx模型&#xff0c;为后面RV1126的部署做铺垫。 我们是在py38的con…

java之多线程的三种不同创建方式and通过多线程模拟龟兔赛跑

Process与Thread&#xff1a; 程序是指令和数据的有序集合&#xff0c;其本身没有任何运行的含义&#xff0c;是一个静态的概念&#xff0c;而进程则是执行程序的一次执行过程&#xff0c;它是一个动态的概念&#xff0c;是系统资源分配的单位&#xff0c;通常在一个进程中可以…

视频素材网,视频剪辑必备。

视频剪辑没素材&#xff0c;推荐6个网站帮你解决&#xff0c;免费可商用&#xff0c;建议收藏&#xff01; 1、菜鸟图库 视频素材下载_mp4视频大全 - 菜鸟图库 网站有超多视频素材&#xff0c;全部都是高清无水印&#xff0c;各种类型都有&#xff0c;像自然、城市、动物、科技…

自动控制原理笔记-线性系统的时域分析与校正

目录 时域法的概述&#xff1a; 时域法的作用和特点&#xff1a; 时域法常用的四个时间信号&#xff1a; 线性系统时域性能指标&#xff1a; 五个常用的性能指标&#xff1a; 一阶系统的时间响应及动态性能&#xff1a; 一阶系统动态指标的计算&#xff1a; 一阶系统的典型…

Github一夜爆火的阿里高并发技术小册究竟有什么魅力

阿里在农历2021到来之前却是又搞了一个大动作&#xff01;把阿里这一年在应对高并发流量的技术经验整合成一份技术小册开源分享供大家学习借鉴。我也是昨天才发现这份小册开源至Github上居然一夜爆火&#xff01; 看了小册之后才知道&#xff0c;原来阿里在应对高并发大流量时也…

python中的json数据和pyecharts模块入门

目录 一json数据格式 1.什么是json 2.json有什么用 3.json格式数据转化 4.python数据和json数据的相互转化 5.json总结 二.pyecharts模块入门 1.基础折线图 全局配置选项——set_global_opts方法 一json数据格式 1.什么是json JSON是一种轻量级的数据交互格式。可以按…

RabbitMQ 第一天 基础 3 RabbitMQ 快速入门 3.2 入门程序【消费者】

RabbitMQ 【黑马程序员RabbitMQ全套教程&#xff0c;rabbitmq消息中间件到实战】 文章目录RabbitMQ第一天 基础3 RabbitMQ 快速入门3.2 入门程序3.2.1 消费者3.2.2 小结第一天 基础 3 RabbitMQ 快速入门 3.2 入门程序 3.2.1 消费者 之前我们 已经完成了生产者的基本代码编…

客快物流大数据项目(九十八):ClickHouse的SQL函数

文章目录 ClickHouse的SQL函数 一、​​​​​​​​​​​​​​类型检测函数