福州大学《嵌入式系统综合设计》实验三:多媒体开发基础编程

一、实验目的

本实验基于搭建好的开发环境和硬件环境,通过编写简单的通信实验,验证开发环境,掌握多媒体开发编程基础,包括SOCKET编程、多线程编程和线程同步知识。

二、实验内容

基于套接字、多线程、同步锁机制实现多媒体文件的收发;

发送端Ubuntu的PC机读取文件,每1024个字节组成一个包通过TCP报文发送到接收端;接收SE5上启动2个线程,线程1接收报文并将报文存入缓存;线程2通过缓存读取报文存入文件中;要求线程1和线程2之间通过同步锁进行线程同步。

三、开发环境

开发主机:Ubuntu 22.04 LTS

硬件:算能SE5

本地如果有SE5硬件,则可以PC机作为客户端,SE5作为服务器端。本地如果没有SE5硬件,只有云空间,则可以直接将客户端和服务器端都通过云空间实现,机在云空间的SE5模拟环境中实现。

四、实验器材

开发主机 + 云平台(或SE5硬件)

五、实验过程与结论

5.1 原理流程

硬件部署环境如下图所示:

如上图所示,可以利用PC作为客户端,SE5作为服务器端,将PC机的文件传送至SE5中。如果是云平台开发,可以直接将客户端和服务器端都放在云平台的模拟器中。此时,即在一台机器内既实现客户端也实现服务器端,设置服务器端的通信地址为回环地址(127.0.0.1)。

客户端程序采用TCP协议进行文件收发。客户端程序采用单线程处理,在和服务器端建立连接后,循环读取流媒体文件,并进行套接字发送。客户端运行流程包含了:

  1. 创建套接字
  2. 输入执行文件名,传输文件名,服务器地址和端口四个参数
  3. 连接服务器的ip地址及端口
  4. 读取需要发送的媒体文件
  5. 启动TCP发送文件,
  6. 循环读取流媒体文件,直到结束后断开连接。

3-1 客户端操作流程图

接收端作为服务端采用多线程进行编程。主线程用于接收连接后接收客户端发送的报文存入缓存。另起一个线程用于从缓存中读取数据包并存入文件中。服务器端的运行流程包含如下关键步骤:

  1. 创建套接字描述符
  2. 绑定ip地址和端口便于客户端接入
  3. 监听是否有客户端发出连接请求
  4. 收到连接请求后启动接收和写文线程
  5. 将接受的报文存入缓存中,同时从缓存读取报文存入文件中
  6. 传输完成后重新等待连接请求。

3-2 服务端操作流程图

5.2 关键代码解析
5.2.1 客户端

由于需要用到套接字进行编程,因此在头文件上需要包含一些必要的头文件:

#include <iostream>       
#include <errno.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>

创建套接字,可以直接利用操作系统的SOCKET接口实现,关键函数如下:

int sockfd;
struct sockaddr_in servaddr;if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)   //创建套接字并判断是否成功
{printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);return 0;
}memset(&servaddr, 0, sizeof(servaddr));    //初始化结构体
servaddr.sin_family = AF_INET;             //设置地址家族
servaddr.sin_port = htons(atoi(argv[3]));  //设置端口//发出连接请求判断是否连接成功
if (connect(sockfd, (struct sockaddr *) &servaddr, sizeof(servaddr)) < 0)        
{printf("connect error: %s(errno: %d)\n", strerror(errno), errno);return 0;
}

至此,客户端主动向服务器发送链接。

在发送端可通过fopen打开文件,通过fread函数读取流媒体文件:   

if ((fq = fopen(argv[1], "rb")) == NULL){/*判断文件是否打开*/close(serverFd);return -1;}
......
/*循环读取文件并发送*/
size_t readLen = fread(buffer, 1, sizeof(buffer), fq);

发送端启动TCP发送,这里的write函数中调用的sockfd是套接字的句柄:

while (!feof(fq)){/*循环读取文件并发送*/size_t readLen = fread(buffer, 1, sizeof(buffer), fq);if (readLen != write(serverFd, buffer, readLen)){printf("write error.\n");break;}
}
5.2.2 服务器端

服务器端由于涉及到多线程,因此需要包含多线程头文件。并且服务器端还涉及到缓冲区,本实例可以通过队列方法设计缓冲区,因此可以包含队列头文件。还有涉及到同步锁机制,因此还需要包含同步锁头文件,具体如下:

#include <stdio.h>
#include <string.h>
#include <netinet/in.h>
#include <unistd.h>
#include <thread>
#include <mutex>
#include <queue>

服务器端首先也需要创建套接字,并等待客户端发起连接,服务器端的关键代码如下:

int main(int argc, char **argv)
{int listenFd, clientFd;struct sockaddr_in servaddr;if ((listenFd = socket(AF_INET, SOCK_STREAM, 0)) < 0){/*创建套接字*/printf("create socket error\n");return -1;}memset(&servaddr, 0, sizeof(servaddr));                     //初始化结构体servaddr.sin_family = AF_INET;                              //设置地址族协议servaddr.sin_addr.s_addr = htonl(INADDR_ANY);               //设置地址servaddr.sin_port = htons(6666);                            //设置默认端口if (bind(listenFd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0){/*绑定套接字地址和端口*/printf("bind socket error\n");return -1;}if (listen(listenFd, 10) < 0){/*开启监听*/printf("listen socket error\n");return -1;}struct sockaddr_in client_addr;socklen_t size = sizeof(client_addr);if ((clientFd = accept(listenFd, (struct sockaddr *)&client_addr, &size)) < 0){/*建立连接*/printf("accept socket error\n");return -1;}std::thread write_thread(writeThread);size_t readLen = 0;while (true){/*循环读取客户端消息*/char buff[MAXBUFF] = {0};readLen = read(clientFd, buff, MAXBUFF);if (readLen <= 0)break;std::string data(buff, readLen);g_mx.lock();                                        //上锁g_dataQue.push(data);g_mx.unlock();                                      //解锁}write_thread.join();close(clientFd);close(listenFd);return 0;
}

注意,在上述函数中定义了写文件线程:

std::thread write_thread(writeThread);

并且在主线程中启动了写文件线程:

write_thread.join();

接收线程执行函数:

std::queue<std::string> g_dataQue;                   //全局队列
std::mutex g_mx;                                     //互斥锁void writeThread()
{/*写线程*/FILE *out_put = fopen("recv_data.mp4", "w+");sleep(1);                                        //休眠一秒,确保队列中有数据while (true){/*从队列中读取数据并存储*/if (g_dataQue.size() == 0)break;g_mx.lock();std::string data = g_dataQue.front();g_dataQue.pop();g_mx.unlock();fwrite((void *)data.data(), 1, data.size(), out_put);}fclose(out_put);
}

如上所示,同步锁用于进行缓冲区的读写同步。上述实例中通过std::mutex实现同步。

g_lock.lock();         //上锁
​​​​​​​g_lock.unlock();       //解锁

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

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

相关文章

循环链表3

插入函数——插入数据&#xff0c;在链表plsit的pos位置插入val数据元素 位置pos&#xff08;在无特别说明的情况下&#xff09;是从0开始计数的 要改变链表结构&#xff0c;就要依赖前驱&#xff0c;每个前驱的next存储着下一个数据结点的地址&#xff0c;也就是依靠前驱的ne…

netty整合websocket(完美教程)

websocket的介绍&#xff1a; WebSocket是一种在网络通信中的协议&#xff0c;它是独立于HTTP协议的。该协议基于TCP/IP协议&#xff0c;可以提供双向通讯并保有状态。这意味着客户端和服务器可以进行实时响应&#xff0c;并且这种响应是双向的。WebSocket协议端口通常是80&am…

敏捷需求管理

敏捷需求管理是一种以敏捷方式进行需求收集、分析和确认的方法。它强调持续不断的需求迭代和交付&#xff0c;以适应快速变化的市场和业务需求。 敏捷需求管理的主要特点包括&#xff1a; 以用户故事为核心&#xff1a;敏捷需求管理以用户故事为核心&#xff0c;将用户需求作…

FPGA——IP核 基础操作

FPGA——IP核 基础操作 IP核例化模块时钟IP核RAM IP核 IP核例化模块 找到模版 加入代码中 时钟IP核 配置模式功能 配置输入时钟 输出配置 RAM IP核

JavaScript 常用符号

JavaScript是一门基础性的编程语言&#xff0c;常用于web开发中。JS中有许多特殊的符号&#xff0c;这些符号的用法十分重要&#xff0c;直接影响代码的正确性和可读性。在日常编写中&#xff0c;我们会频繁使用以下几个符号。 一、等于号&#xff08;&#xff09; 等于号在JS…

QT之QProcess

类描述信息 QProcess允许您将进程视为顺序I/O设备。您可以写入和读取进程&#xff0c;就像使用QTcpSocket访问网络连接一样。然后&#xff0c;您可以通过调用write()写入进程的标准输入&#xff0c;并通过调用read()、readLine()和getChar()读取标准输出。由于QProcess继承了QI…

python 将str转换成list

import ast str [a,b,c] list ast.literal_eval(str) print(type(list))

Pytorch 网络冻结的三种方法区别:detach、requires_grad、with_no_grad

1、requires_grad requires_gradTrue # 要求计算梯度&#xff1b; requires_gradFalse # 不要求计算梯度&#xff1b;在pytorch中&#xff0c;tensor有一个 requires_grad参数&#xff0c;如果设置为True&#xff0c;那么它会追踪对于该张量的所有操作。在完成计算时可以通过调…

阿里云服务器公网带宽升级的三种方法

阿里云服务器公网带宽不够用有哪些解决方法&#xff1f;可以更改带宽或带宽临时升级&#xff0c;更改带宽是永久公网带宽&#xff0c;带宽临时升级可以选择升级时间段&#xff0c;也可以绑定弹性公网EIP来修改公网带宽&#xff0c;阿里云服务器网aliyunfuwuqi.com分享阿里云服务…

海外IP代理科普——API代理是什么?怎么用?

随着互联网的不断发展&#xff0c;越来越多的企业开始使用API&#xff08;应用程序接口&#xff09;来实现数据的共享和交流。而在API使用中&#xff0c;海外代理IP也逐渐普及。那么&#xff0c;什么是API代理IP呢&#xff1f;它有什么作用&#xff1f;API接口有何用处&#xf…

从0开始学习JavaScript--JavaScript 函数

JavaScript中的函数是编写可维护、模块化代码的关键。本文将深入研究JavaScript函数的各个方面&#xff0c;包括基本语法、函数作用域、闭包、高阶函数、箭头函数等&#xff0c;并通过丰富的示例代码来帮助读者更好地理解和应用这些概念。 函数的基本语法 函数是一段可被重复…

git merge指定的文件

如果你只想在合并时包含特定类型的文件&#xff08;例如&#xff0c;只合并.java文件&#xff09;&#xff0c;可以使用git checkout命令和git merge命令的组合来实现。以下是一个可能的步骤&#xff1a; 确保在目标分支上&#xff1a; 在执行合并之前&#xff0c;请确保你已经…

openGauss学习笔记-129 openGauss 数据库管理-参数设置-查看参数值

文章目录 openGauss学习笔记-129 openGauss 数据库管理-参数设置-查看参数值129.1 操作步骤129.2 示例 openGauss学习笔记-129 openGauss 数据库管理-参数设置-查看参数值 openGauss安装后&#xff0c;有一套默认的运行参数&#xff0c;为了使openGauss与业务的配合度更高&…

nvim 配置教程

1. 主角&#xff1a; NeoVim sudo apt install -y ninja-build gettext cmake unzip curl sudo apt install -y universal-ctags cscope #函数跳转用到的依赖 git clone https://github.com/neovim/neovim.git --depth1 cd neovim make CMAKE_BUILD_TYPERelWithDebInfo sudo…

C#学习相关系列之Linq用法---where和select用法(二)

一、select用法 Linq中的select可以便捷使我们的对List中的每一项进行操作&#xff0c;生成新的列表。 var ttlist.select(p>p10); //select括号内为List中的每一项&#xff0c;p10即为对每一项的操作&#xff0c;即对每项都加10生成新的List 用法实例&#xff1a; 1、la…

C++ 20类型转换指南:使用场景与最佳实践

C 20类型转换指南&#xff1a;使用场景与最佳实践 类型转换 (Casts) C 提供了五种特定的类型转换&#xff1a;const_cast<>()、static_cast<>()、reinterpret_cast<>()、dynamic_cast<>() 和 C20 引入的 std::bit_cast<>()。 请注意&#xff…

阿里云服务器带宽可以修改吗?不够用怎么办?

阿里云服务器公网带宽不够用有哪些解决方法&#xff1f;可以更改带宽或带宽临时升级&#xff0c;更改带宽是永久公网带宽&#xff0c;带宽临时升级可以选择升级时间段&#xff0c;也可以绑定弹性公网EIP来修改公网带宽&#xff0c;阿里云服务器网aliyunfuwuqi.com分享阿里云服务…

NoSQL 与传统数据库的集成

数据库集成势在必行 随着数据格局以前所未有的复杂性和规模发展&#xff0c;围绕数据库的叙述已经发生了巨大的变化。NoSQL 数据库已成为传统关系数据库的引人注目的替代品&#xff0c;在可扩展性、灵活性和数据模型多样性方面提供了显着的优势。然而&#xff0c;由于其 ACID …

SpringCloud原理-OpenFeign篇(三、FeignClient的动态代理原理)

文章目录 前言正文一、前戏&#xff0c;FeignClientFactoryBean入口方法的分析1.1 从BeanFactory入手1.2 AbstractBeanFactory#doGetBean(...)中对FactoryBean的处理1.3 结论 FactoryBean#getObject() 二、FeignClientFactoryBean实现的getObject()2.1 FeignClientFactoryBean#…

oepnpnp - 自己出图做开口扳手

文章目录 oepnpnp - 自己出图做开口扳手概述笔记做好的一套扳手实拍美图工程图END oepnpnp - 自己出图做开口扳手 概述 我的openpnp设备顶部相机安装支架, 由于结构限制, 螺柱的安装位置和机械挂壁的距离太近了. 导致拧紧(手工或者工具)很困难. 也不能重新做相机支架, 因为将…