上位机图像处理和嵌入式模块部署(上位机和下位机通信)

2024-02-10 17:20

本文主要是介绍上位机图像处理和嵌入式模块部署(上位机和下位机通信),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】

        一般情况下,如果是纯上位机开发的话,这个时候是不需要上位机和下位机进行通信的。只有上位机做好demo有必要移植到嵌入式模块,或者需要进行算法标定的时候,才需要上位机、下位机进行通信。通信的方式很多,比如232、485、usb等等。不过个人觉得比较方便的方法,还是用网络进行通信。

1、选择的协议

        如果是单纯的验证测试,那么用xmlrpc是可以的。但是,这里面存在一个问题,那就是上位机和下位机通信的内容容易被看到、被抓包。所以,从商业角度来说,比较理想的方法,还是自己设计协议、自己来实现具体的内容。当然,通信的方式当中也有可能用到xml数据,或者是json数据,这都是没有问题的。

2、windows代码

        既然是通信,那么我们可以编写一个简单的windows代码,这部分代码虽然是十多年前编写的,还是很有指导意义的。用比较新的visual studio工具编译,只需要修改一行代码scanf即可。并不是代码本身的错误,纯粹是因为函数过期了。

#include <stdio.h>
#include <Windows.h>#pragma comment(lib,"ws2_32.lib")
#define  PORT 4000
#define  IP_ADDRESS "192.168.0.97"int main(int argc, char* argv[])
{WSADATA  Ws;SOCKET ClientSocket;struct sockaddr_in ClientAddr;int Ret = 0;char SendBuffer[MAX_PATH];/* Init Windows Socket */if (WSAStartup(MAKEWORD(2, 2), &Ws) != 0){printf("Init Windows Socket Failed::%d\n", GetLastError());return -1;}/* Create Socket */ClientSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);if (ClientSocket == INVALID_SOCKET){printf("Create Socket Failed::%d\n", GetLastError());return -1;}ClientAddr.sin_family = AF_INET;ClientAddr.sin_addr.s_addr = inet_addr(IP_ADDRESS);ClientAddr.sin_port = htons(PORT);memset(ClientAddr.sin_zero, 0x00, 8);/* connect socket */Ret = connect(ClientSocket, (struct sockaddr*)&ClientAddr, sizeof(ClientAddr));if (Ret == SOCKET_ERROR){printf("Connect Error::%d\n", GetLastError());return -1;}else{printf("Connect succedded!\n");}while (1){scanf_s("%s", SendBuffer, MAX_PATH-1);/* send data to server */Ret = send(ClientSocket, SendBuffer, (int)strlen(SendBuffer), 0);if (Ret == SOCKET_ERROR){printf("Send Info Error::%d\n", GetLastError());break;}if ('q' == SendBuffer[0]){break;}}/* close socket */closesocket(ClientSocket);WSACleanup();return 0;
}

3、linux代码

        之前提及过,我们开发使用的主要代码就是linux+arm。为了方便进一步开发和调试,我们还使用到了树莓派4b。所以,这里为了和windows进行通信,我们编写了一个linux server端代码,大家可以参考一下。编译的方法就是gcc server.c -g -o server。

#include <netinet/in.h>    
#include <sys/types.h>    
#include <sys/socket.h>    
#include <stdio.h>        
#include <stdlib.h>       
#include <string.h>       #define HELLO_WORLD_SERVER_PORT    4000
#define LENGTH_OF_LISTEN_QUEUE 20
#define BUFFER_SIZE 1024int main(int argc, char **argv)
{struct sockaddr_in server_addr;int server_socket;int opt = 1;bzero(&server_addr,sizeof(server_addr)); server_addr.sin_family = AF_INET;server_addr.sin_addr.s_addr = htons(INADDR_ANY);server_addr.sin_port = htons(HELLO_WORLD_SERVER_PORT);/* create a socket */server_socket = socket(PF_INET,SOCK_STREAM,0);if( server_socket < 0){printf("Create Socket Failed!");exit(1);}/* bind socket to a specified address*/setsockopt(server_socket, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt));if( bind(server_socket,(struct sockaddr*)&server_addr,sizeof(server_addr))){printf("Server Bind Port : %d Failed!", HELLO_WORLD_SERVER_PORT); exit(1);}/* listen a socket */if(listen(server_socket, LENGTH_OF_LISTEN_QUEUE)){printf("Server Listen Failed!"); exit(1);}/* run server */while (1) {struct sockaddr_in client_addr;int client_socket;		socklen_t length;char buffer[BUFFER_SIZE];/* accept socket from client */length = sizeof(client_addr);client_socket = accept(server_socket, (struct sockaddr*)&client_addr, &length);if( client_socket < 0){printf("Server Accept Failed!\n");break;}/* receive data from client */while(1){bzero(buffer, BUFFER_SIZE);length = recv(client_socket, buffer, BUFFER_SIZE, 0);if (length < 0){printf("Server Recieve Data Failed!\n");break;}if('q' == buffer[0]){printf("Quit from client!\n");break;}printf("%s\n", buffer);}	close(client_socket);}close(server_socket);return 0;
}

4、代码分析

        从上述的代码来看,主要就是linux启动一个server,然后accept到的socket和client进行通信。通信的过程当中,一直是client发送数据,server显示数据,如果client断掉,server会give up,接着去accept另外一个socket。这就是整个代码的处理流程,虽然比较简单,但是也算是实现了上位机、下位机的处理流程算法。

        测试的时候,首先server端启动,然后client启动,接着client端发送数据,检查server端是不是可以收到数据。这是client端的截图,

        这是server端的截图,

这篇关于上位机图像处理和嵌入式模块部署(上位机和下位机通信)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/697667

相关文章

MySQL 主从复制部署及验证(示例详解)

《MySQL主从复制部署及验证(示例详解)》本文介绍MySQL主从复制部署步骤及学校管理数据库创建脚本,包含表结构设计、示例数据插入和查询语句,用于验证主从同步功能,感兴趣的朋友一起看看吧... 目录mysql 主从复制部署指南部署步骤1.环境准备2. 主服务器配置3. 创建复制用户4. 获取主服务器状态5

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

如何在Ubuntu 24.04上部署Zabbix 7.0对服务器进行监控

《如何在Ubuntu24.04上部署Zabbix7.0对服务器进行监控》在Ubuntu24.04上部署Zabbix7.0监控阿里云ECS服务器,需配置MariaDB数据库、开放10050/1005... 目录软硬件信息部署步骤步骤 1:安装并配置mariadb步骤 2:安装Zabbix 7.0 Server

Python中re模块结合正则表达式的实际应用案例

《Python中re模块结合正则表达式的实际应用案例》Python中的re模块是用于处理正则表达式的强大工具,正则表达式是一种用来匹配字符串的模式,它可以在文本中搜索和匹配特定的字符串模式,这篇文章主... 目录前言re模块常用函数一、查看文本中是否包含 A 或 B 字符串二、替换多个关键词为统一格式三、提

SQLite3 在嵌入式C环境中存储音频/视频文件的最优方案

《SQLite3在嵌入式C环境中存储音频/视频文件的最优方案》本文探讨了SQLite3在嵌入式C环境中存储音视频文件的优化方案,推荐采用文件路径存储结合元数据管理,兼顾效率与资源限制,小文件可使用B... 目录SQLite3 在嵌入式C环境中存储音频/视频文件的专业方案一、存储策略选择1. 直接存储 vs

嵌入式数据库SQLite 3配置使用讲解

《嵌入式数据库SQLite3配置使用讲解》本文强调嵌入式项目中SQLite3数据库的重要性,因其零配置、轻量级、跨平台及事务处理特性,可保障数据溯源与责任明确,详细讲解安装配置、基础语法及SQLit... 目录0、惨痛教训1、SQLite3环境配置(1)、下载安装SQLite库(2)、解压下载的文件(3)、

一文深入详解Python的secrets模块

《一文深入详解Python的secrets模块》在构建涉及用户身份认证、权限管理、加密通信等系统时,开发者最不能忽视的一个问题就是“安全性”,Python在3.6版本中引入了专门面向安全用途的secr... 目录引言一、背景与动机:为什么需要 secrets 模块?二、secrets 模块的核心功能1. 基

RabbitMQ工作模式中的RPC通信模式详解

《RabbitMQ工作模式中的RPC通信模式详解》在RabbitMQ中,RPC模式通过消息队列实现远程调用功能,这篇文章给大家介绍RabbitMQ工作模式之RPC通信模式,感兴趣的朋友一起看看吧... 目录RPC通信模式概述工作流程代码案例引入依赖常量类编写客户端代码编写服务端代码RPC通信模式概述在R

在Spring Boot中实现HTTPS加密通信及常见问题排查

《在SpringBoot中实现HTTPS加密通信及常见问题排查》HTTPS是HTTP的安全版本,通过SSL/TLS协议为通讯提供加密、身份验证和数据完整性保护,下面通过本文给大家介绍在SpringB... 目录一、HTTPS核心原理1.加密流程概述2.加密技术组合二、证书体系详解1、证书类型对比2. 证书获