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

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

相关文章

Python的time模块一些常用功能(各种与时间相关的函数)

《Python的time模块一些常用功能(各种与时间相关的函数)》Python的time模块提供了各种与时间相关的函数,包括获取当前时间、处理时间间隔、执行时间测量等,:本文主要介绍Python的... 目录1. 获取当前时间2. 时间格式化3. 延时执行4. 时间戳运算5. 计算代码执行时间6. 转换为指

Python正则表达式语法及re模块中的常用函数详解

《Python正则表达式语法及re模块中的常用函数详解》这篇文章主要给大家介绍了关于Python正则表达式语法及re模块中常用函数的相关资料,正则表达式是一种强大的字符串处理工具,可以用于匹配、切分、... 目录概念、作用和步骤语法re模块中的常用函数总结 概念、作用和步骤概念: 本身也是一个字符串,其中

Python中的getopt模块用法小结

《Python中的getopt模块用法小结》getopt.getopt()函数是Python中用于解析命令行参数的标准库函数,该函数可以从命令行中提取选项和参数,并对它们进行处理,本文详细介绍了Pyt... 目录getopt模块介绍getopt.getopt函数的介绍getopt模块的常用用法getopt模

python logging模块详解及其日志定时清理方式

《pythonlogging模块详解及其日志定时清理方式》:本文主要介绍pythonlogging模块详解及其日志定时清理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录python logging模块及日志定时清理1.创建logger对象2.logging.basicCo

Qt spdlog日志模块的使用详解

《Qtspdlog日志模块的使用详解》在Qt应用程序开发中,良好的日志系统至关重要,本文将介绍如何使用spdlog1.5.0创建满足以下要求的日志系统,感兴趣的朋友一起看看吧... 目录版本摘要例子logmanager.cpp文件main.cpp文件版本spdlog版本:1.5.0采用1.5.0版本主要

Spring Boot项目部署命令java -jar的各种参数及作用详解

《SpringBoot项目部署命令java-jar的各种参数及作用详解》:本文主要介绍SpringBoot项目部署命令java-jar的各种参数及作用的相关资料,包括设置内存大小、垃圾回收... 目录前言一、基础命令结构二、常见的 Java 命令参数1. 设置内存大小2. 配置垃圾回收器3. 配置线程栈大小

opencv图像处理之指纹验证的实现

《opencv图像处理之指纹验证的实现》本文主要介绍了opencv图像处理之指纹验证的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学... 目录一、简介二、具体案例实现1. 图像显示函数2. 指纹验证函数3. 主函数4、运行结果三、总结一、

tomcat多实例部署的项目实践

《tomcat多实例部署的项目实践》Tomcat多实例是指在一台设备上运行多个Tomcat服务,这些Tomcat相互独立,本文主要介绍了tomcat多实例部署的项目实践,具有一定的参考价值,感兴趣的可... 目录1.创建项目目录,测试文China编程件2js.创建实例的安装目录3.准备实例的配置文件4.编辑实例的

SpringBoot配置Ollama实现本地部署DeepSeek

《SpringBoot配置Ollama实现本地部署DeepSeek》本文主要介绍了在本地环境中使用Ollama配置DeepSeek模型,并在IntelliJIDEA中创建一个Sprin... 目录前言详细步骤一、本地配置DeepSeek二、SpringBoot项目调用本地DeepSeek前言随着人工智能技

通过Docker Compose部署MySQL的详细教程

《通过DockerCompose部署MySQL的详细教程》DockerCompose作为Docker官方的容器编排工具,为MySQL数据库部署带来了显著优势,下面小编就来为大家详细介绍一... 目录一、docker Compose 部署 mysql 的优势二、环境准备与基础配置2.1 项目目录结构2.2 基