【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱: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端的截图,