Linux与UDP应用1:翻译软件

embedded/2025/3/5 21:38:06/

UDP应用1:翻译软件

本篇介绍

本篇基于UDP编程接口基本使用中封装的服务器和客户端进行改写,基本功能如下:

  1. 从配置文件dict.txt读取到所有的单词和意思
  2. 客户端向服务端发送英文
  3. 服务端向客户端发送英文对应的中文意思

配置文件内容

下面的内容是本次实现使用的配置文件

apple: 苹果
banana: 香蕉
cat: 猫
dog: 狗
book: 书
pen: 笔
happy: 快乐的
sad: 悲伤的
run: 跑
jump: 跳
teacher: 老师
student: 学生
car: 汽车
bus: 公交车
love: 爱
hate: 恨
hello: 你好
goodbye: 再见
summer: 夏天
winter: 冬天

设计翻译软件

根据上面对功能描述,下面对功能实现进行具体分析

设计字典类

既然是字典类,那么对应的就需要一个便于查询的结构,本次以哈希表为例

因为服务端主要是接收客户端的数据,所以服务端本身不直接处理翻译功能,而是交给一个函数进行处理,但是如果这个函数直接裸露在外部就会导致字典本身和翻译函数分开,所以本次考虑将翻译函数作为字典类的成员函数

需要注意,本次默认使用的配置文件中英文单词和中文意思使用的是一个冒号和空格进行分割,即: ,可以考虑让用户自己提供配置文件和分割符,所以类基本结构如下:

// 默认配置文件位置和配置文件
const std::string default_path = "./";
const std::string default_file = "dict.txt";// 默认分隔符
const std::string default_sep = ": ";class Dictionary
{
public:Dictionary(const std::string &path = default_path, const std::string &file = default_file, const std::string &sep = default_sep): _path(path), _file(file), _sep(sep){// 1. 读取配置文件// 2. 分割字符串并将key和value添加到哈希表}// 翻译std::string translate(const std::string &word){}~Dictionary(){}private:std::unordered_map<std::string, std::string> _dict; // 字典哈希表std::string _path;                                  // 配置文件路径std::string _file;                                  // 配置文件std::string _sep;                                   // 分隔符
};

读取配置文件并分割

首先是读取配置文件,本次考虑将从配置文件中读取到的字符串通过分割成keyvalue依次存入到一个哈希表中。并且因为是字典类,所以考虑在创建字典类对象时就完成前面的读取和存储操作

根据上面的思路需要考虑两个步骤:

  1. 获取并读取配置文件
  2. 分割字符串获取到keyvalue存储到哈希表

读取配置文件的操作就是打开指定的文件并读取其中的内容,在本次配置文件的内容中,一行为一组数据,所以在读取文件内容时需要按照行读取

读到一行数据后,需要将该行数据按照指定的分隔符进行切割存入keyvalue中,这里有两种思路:

  1. 自行实现分割逻辑
  2. 使用库函数

本次考虑使用第一种思路,那么对于分割逻辑来说,基本思路如下:

  1. 找到:第一次出现的位置,以该位置为终点,以字符串第一个字符为起点,切出第一个子串作为key
  2. 找到分隔符的下一个字符,以该位置为起点,以字符串结尾为终点,切出第二个子串作为value

将获取到的keyvalue存入到哈希表中

重复上面的步骤直到读取完毕文件,将文件关闭,代码如下:

// 1. 读取配置文件
std::string dictPath = default_path + default_file;
// 1.1 打开文件
std::fstream out(dictPath);// 2. 分割字符串并将key和value添加到哈希表
std::string message;
while (std::getline(out, message))
{// 分割字符串auto word_end = message.find(_sep, 0);std::string key = message.substr(0, word_end);std::string value = message.substr(word_end + sep.size(), message.size());// 存储到哈希表中_dict.insert({key, value});
}out.close();

翻译函数

所谓翻译就是在哈希表中根据指定字符串查找对应的value并返回,基本代码如下:

// 翻译
std::string translate(const std::string &word)
{auto pos = _dict.find(word);if (pos == _dict.end())return "无指定单词对应的中文意思";return pos->second;
}

修改服务端类

本次服务端做的任务就是接收客户端发送的数据,再调用字典类的翻译函数将翻译结果返回给客户端,所以实际上需要修改的就是两点:

  1. 服务端需要拿到翻译函数
  2. 执行翻译功能并返回结果给客户端

首先修改第一点,因为服务端本身没有翻译函数,所以需要外界传递,此时可以考虑在构造服务端时要求外部传递翻译函数,对应地服务端就需要一个成员用于接收这个函数:

using task_t = std::function<std::string(const std::string &)>;class UdpServer
{
public:UdpServer(task_t t, uint16_t port = default_port): // ...,_translate(t){// ...}// ...
private:// ...task_t _translate; // 翻译函数
};

需要注意,在构造函数中,没有默认参数的形式参数一定要写在有默认参数的形式参数之前,具体原因见C++中的缺省参数
接着修改第二点,执行翻译函数并将结果返回给客户端,这一步主要涉及到服务端的任务部分。在上一节中,主要任务是将客户端发送的信息再发给客户端,这次就是调用翻译函数再将结果返回给客户端,所以代码如下:

// 启动服务器
void start()
{if (!_isRunning){_isRunning = true;while (true){// 1. 接收客户端信息// ...if (ret > 0){// ...// 翻译std::string ret = _translate(buffer);ssize_t n = sendto(_socketfd, ret.c_str(), ret.size(), 0, &temp, temp.getLength());// ...}}}
}

修改服务端主函数

主要修改的地方就是创建服务端对象部分,因为需要传递一个翻译任务函数,所以需要先创建一个字典类对象,再调用字典类对象中的翻译方法,需要注意的是,不能直接将字典类的翻译函数作为参数传递给服务端对象的构造函数,因为该函数的参数列表除了需要显示传递的字符串对象外,还存在一个字典类对象,这并不符合服务端类构造函数要求的函数类型,这里提供两个方法:

  1. 使用绑定,将字典类对象作为固定参数绑定给翻译函数
  2. 使用lambda表达式,在表达式中调用翻译函数

即:

=== “绑定”

// 创建字典类对象
std::shared_ptr<Dictionary> dict;
// 创建UdpServerModule对象
std::shared_ptr<UdpServer> udp_server = = std::make_shared<UdpServer>(std::bind(&Dictionary::translate, dict.get(), std::placeholders::_1));

=== “lambda表达式”

// 创建字典类对象
std::shared_ptr<Dictionary> dict;
// 创建UdpServerModule对象
std::shared_ptr<UdpServer> udp_server = = std::make_shared<UdpServer>([&dict](const std::string word){ dict->translate(word); 
});

测试

以绑定为例,服务端整体代码如下:

#include "udp_server.hpp"
#include "dictionary.hpp"
#include <memory>using namespace UdpServerModule;
using namespace DictionaryModule;int main(int argc, char *argv[])
{// 创建字典类对象std::shared_ptr<Dictionary> dict;// 创建UdpServerModule对象std::shared_ptr<UdpServer> udp_server;if (argc == 1){// 绑定udp_server = std::make_shared<UdpServer>(std::bind(&Dictionary::translate, dict.get(), std::placeholders::_1));}else if (argc == 2){// std::string ip = argv[1]; 去除uint16_t port = std::stoi(argv[1]);udp_server = std::make_shared<UdpServer>(std::bind(&Dictionary::translate, dict.get(), std::placeholders::_1), port);}else{LOG(LogLevel::ERROR) << "错误使用,正确使用:" << argv[0] << " 端口(或者不写)";exit(4);}udp_server->start();return 0;
}

客户端代码保持和上一节一样,此处不再展示,运行结果如下:
在这里插入图片描述


http://www.ppmy.cn/embedded/170299.html

相关文章

大白话React Hooks(如 useState、useEffect)的使用方法与原理

啥是 React Hooks 在 React 里&#xff0c;以前我们写组件主要用类&#xff08;class&#xff09;的方式&#xff0c;写起来有点复杂&#xff0c;尤其是处理状态和副作用的时候。React Hooks 就是 React 16.8 之后推出的新特性&#xff0c;它能让我们不用写类&#xff0c;直接…

211.SpringSecurity:认证、授权概念,自定义数据源认证,验证码认证

目录 一、权限管理概念 1.基本概念 2.常用权限管理解决方案 (1)Shiro (2)Spring Security 二、整体架构 1.认证:Authentication (1)AuthenticationManager (2)Authentication (3)SecurityContextHolder 2.授权:Authorization (1)AccessDecisionManag…

面试常问的压力测试问题

性能测试作为软件开发中的关键环节&#xff0c;确保系统在高负载下仍能高效运行。压力测试作为性能测试的重要类型&#xff0c;旨在通过施加超出正常负载的压力&#xff0c;观察系统在极端条件下的表现。面试中&#xff0c;相关问题常被问及&#xff0c;包括定义、重要性、与负…

计算机毕设JAVA——某高校宿舍管理系统(基于SpringBoot+Vue前后端分离的项目)

文章目录 概要项目演示图片系统架构技术运行环境系统功能简介 概要 网络上许多计算机毕设项目开发前端界面设计复杂、不美观&#xff0c;而且功能结构十分单一&#xff0c;存在很多雷同的项目&#xff1a;不同的项目基本上就是套用固定模板&#xff0c;换个颜色、改个文字&…

大模型的实践应用36-基于AI Agent和通义千问大模型,支持多轮问答的智能问数和数据分析的应用场景

大家好,我是微学AI,今天给大家介绍一下大模型的实践应用36基于AI Agent和通义千问大模型,支持多轮问答的智能问数和数据分析的应用场景。 文章目录 一、整体框架功能概述(一)深度解析多轮对话在数据查询需求确定中的关键作用(二)明确查询相关要素的重要性框架结构(一)…

Redis数据库面试——数据结构类型知识

大家好&#xff0c;这里是Good Note&#xff0c;关注 公主号&#xff1a;Goodnote&#xff0c;专栏文章私信限时Free。本文详细介绍Redis 提供的5种基本数据结构类型和4种特殊类型&#xff0c;除此之外&#xff0c;还有8种底层数据结构&#xff0c;每种结构类型有其特点和适用场…

19.4-STM32接收数据-状态显示在屏幕

功能介绍放开头&#xff0c; 使用便捷无需愁 这是全网最详细、性价比最高的STM32实战项目入门教程&#xff0c;通过合理的硬件设计和详细的视频笔记介绍&#xff0c;硬件使用STM32F103主控资料多方便学习&#xff0c;通过3万字笔记、12多个小时视频、20多章节代码手把手教会你…

c++中什么时候应该使用extern关键字?

目录 1. 共享全局变量 2. 共享常量&#xff08;const&#xff09;变量 3. 与C语言交互 4. 显式模板实例化声明 5. 动态库符号管理 注意事项 关键总结 在C中&#xff0c;extern关键字主要用于声明变量、函数或模板的外部链接性&#xff0c;表明其定义存在于其他编译单元中…