【在Linux世界中追寻伟大的One Piece】Socket编程UDP

news2024/10/23 11:01:20

目录

1 -> UDP网络编程

1.1 -> V1版本 -echo server

1.2 -> V2版本 -DictServer

1.3 -> V2版本 -DictServer(封装版)


1 -> UDP网络编程

1.1 -> V1版本 -echo server

简单的回显服务器和客户端代码。

备注:代码中会用到地址转换函数

nocopy.hpp

#pragma once
#include <iostream>

class nocopy
{
public:
	nocopy() {}
	nocopy(const nocopy&) = delete;
	const nocopy& operator = (const nocopy&) = delete;
	~nocopy() {}
};

UdpServer.hpp

#pragma once
#include <iostream>
#include <string>
#include <cerrno>
#include <cstring>
#include <unistd.h>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include "nocopy.hpp"
#include "Log.hpp"
#include "Comm.hpp"
#include "InetAddr.hpp"

const static uint16_t defaultport = 8888;
const static int defaultfd = -1;
const static int defaultsize = 1024;

class UdpServer : public nocopy
{
public:
	UdpServer(uint16_t port = defaultport)
		: _port(port), _sockfd(defaultfd)
	{
	}

	void Init()
	{
		// 1. 创建 socket,就是创建了文件细节
		_sockfd = socket(AF_INET, SOCK_DGRAM, 0);
		if (_sockfd < 0)
		{
			lg.LogMessage(Fatal, "socket errr, %d : %s\n", errno,
				strerror(errno));
			exit(Socket_Err);
		}

		lg.LogMessage(Info, "socket success, sockfd: %d\n",
			_sockfd);

		// 2. 绑定,指定网络信息
		struct sockaddr_in local;

		bzero(&local, sizeof(local)); // memset
		local.sin_family = AF_INET;
		local.sin_port = htons(_port);
		local.sin_addr.s_addr = INADDR_ANY; // 0

		// local.sin_addr.s_addr = inet_addr(_ip.c_str()); // 1. 4字节 IP 2. 变成网络序列
		// 结构体填完,设置到内核中了吗??没有
		int n = ::bind(_sockfd, (struct sockaddr*)&local,
			sizeof(local));
		if (n != 0)
		{
			lg.LogMessage(Fatal, "bind errr, %d : %s\n", errno,
				strerror(errno));
			exit(Bind_Err);
		}
	}

	void Start()
	{
		// 服务器永远不退出
		char buffer[defaultsize];
		for (;;)
		{
			struct sockaddr_in peer;

			socklen_t len = sizeof(peer); // 不能乱写
			ssize_t n = recvfrom(_sockfd, buffer, sizeof(buffer) -
				1, 0, (struct sockaddr*)&peer, &len);
			if (n > 0)
			{
				InetAddr addr(peer);
				buffer[n] = 0;

				std::cout << "[" << addr.PrintDebug() << "]# " <<
					buffer << std::endl;

				sendto(_sockfd, buffer, strlen(buffer), 0, (struct
					sockaddr*)&peer, len);
			}
		}
	}

	~UdpServer()
	{
	}

private:
	// std::string _ip; // 后面要调整
	uint16_t _port;
	int _sockfd;
};

InetAddr.hpp

#pragma once
#include <iostream>
#include <string>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

class InetAddr
{
public:
	InetAddr(struct sockaddr_in& addr) :_addr(addr)
	{
		_port = ntohs(_addr.sin_port);
		_ip = inet_ntoa(_addr.sin_addr);
	}

	std::string Ip() 
	{ 
		return _ip; 
	}

	uint16_t Port() 
	{ 
		return _port; 
	};

	std::string PrintDebug()
	{
		std::string info = _ip;
		info += ":";
		info += std::to_string(_port); // "127.0.0.1:4444"

		return info;
	}

	~InetAddr() {}

private:
	std::string _ip;
	uint16_t _port;
	struct sockaddr_in _addr;
};

Comm.hpp

#pragma once

enum 
{
	Usage_Err = 1,
	Socket_Err,
	Bind_Err
};
  • 云服务器不允许直接bind公有IP,我们也不推荐编写服务器的时候,bind明确的IP,推荐直接写成INADDR_ANY。
/* Address to accept any incoming messages. */

#define INADDR_ANY ((in_addr_t) 0x00000000)

在网络编程中,当一个进程需要绑定一个网络端口以进行通信时,可以使用INADDR_ANY作为IP地址参数。这样做意味着该端口可以接受来自任何IP地址的连接请求,无论是本地主机还是远程主机。例如,如果服务器有多个网卡(每个网卡上有不同的IP地址),使用INADDR_ANY可以省去确定数据是从服务器上具体哪个网卡/IP地址上面获取的。

UdpClient.hpp

#include <iostream>
#include <cerrno>
#include <cstring>
#include <string>
#include <unistd.h>
#include <sys/types.h> /* See NOTES */
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>

void Usage(const std::string& process)
{
	std::cout << "Usage: " << process << " server_ip server_port"
		<< std::endl;
}

// ./udp_client server_ip server_port
int main(int argc, char* argv[])
{
	if (argc != 3)
	{
		Usage(argv[0]);

		return 1;
	}

	std::string serverip = argv[1];
	uint16_t serverport = std::stoi(argv[2]);

	// 1. 创建 socket
	int sock = socket(AF_INET, SOCK_DGRAM, 0);
	if (sock < 0)
	{
		std::cerr << "socket error: " << strerror(errno) <<
			std::endl;

		return 2;
	}

	std::cout << "create socket success: " << sock << std::endl;
	// 2. client 要不要进行 bind? 一定要 bind 的!!
	// 但是,不需要显示 bind,client 会在首次发送数据的时候会自动进行bind
	// 为什么?server 端的端口号,一定是众所周知,不可改变的,client 需要 port,bind 随机端口.
	// 为什么?client 会非常多.
	// client 需要 bind,但是不需要显示 bind,让本地 OS 自动随机 bind,选择随机端口号
	// 2.1 填充一下 server 信息
	struct sockaddr_in server;
	memset(&server, 0, sizeof(server));
	server.sin_family = AF_INET;
	server.sin_port = htons(serverport);
	server.sin_addr.s_addr = inet_addr(serverip.c_str());
	while (true)
	{
		// 我们要发的数据
		std::string inbuffer;
		std::cout << "Please Enter# ";
		std::getline(std::cin, inbuffer);

		// 我们要发给谁呀?server
		ssize_t n = sendto(sock, inbuffer.c_str(),
			inbuffer.size(), 0, (struct sockaddr*)&server, sizeof(server));
		if (n > 0)
		{
			char buffer[1024];
			//收消息
			struct sockaddr_in temp;
			socklen_t len = sizeof(temp);
			ssize_t m = recvfrom(sock, buffer, sizeof(buffer) - 1,
				0, (struct sockaddr*)&temp, &len); // 一般建议都是要填的.
			if (m > 0)
			{
				buffer[m] = 0;
				std::cout << "server echo# " << buffer <<
					std::endl;
			}
			else
				break;
		}
		else
			break;
	}

	close(sock);

	return 0;
}

1.2 -> V2版本 -DictServer

实现一个简单的英译汉的网络字典

dict.txt

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

Dict.hpp

#pragma once
#include <iostream>
#include <string>
#include <fstream>
#include <unordered_map>

const std::string sep = ": ";

class Dict
{
private:
	void LoadDict()
	{
		std::ifstream in(_confpath);
		if (!in.is_open())
		{
			std::cerr << "open file error" << std::endl; // 后面可以用日志替代打印
				return;
		}

		std::string line;
		while (std::getline(in, line))
		{
			if (line.empty()) 
				continue;

			auto pos = line.find(sep);
			if (pos == std::string::npos) 
				continue;

			std::string key = line.substr(0, pos);
			std::string value = line.substr(pos + sep.size());
			_dict.insert(std::make_pair(key, value));
		}

		in.close();
	}

public:
	Dict(const std::string& confpath) :_confpath(confpath)
	{
		LoadDict();
	}

	std::string Translate(const std::string& key)
	{
		auto iter = _dict.find(key);
		if (iter == _dict.end()) 
			return std::string("Unknown");

		else 
			return iter->second;
	}

	~Dict()
	{}

private:
	std::string _confpath;
	std::unordered_map<std::string, std::string> _dict;
};

UdpServer.hpp

#pragma once
#include <iostream>
#include <string>
#include <cerrno>
#include <cstring>
#include <unistd.h>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unordered_map>
#include <functional>
#include "nocopy.hpp"
#include "Log.hpp"
#include "Comm.hpp"
#include "InetAddr.hpp"

const static uint16_t defaultport = 8888;
const static int defaultfd = -1;
const static int defaultsize = 1024;

using func_t = std::function<void(const std::string& req,
	std::string* resp)>;

class UdpServer : public nocopy
{
public:
	UdpServer(func_t func, uint16_t port = defaultport)
		: _func(func), _port(port), _sockfd(defaultfd)
	{
	}

	void Init()
	{
		// 1. 创建 socket,就是创建了文件细节
		_sockfd = socket(AF_INET, SOCK_DGRAM, 0);
		if (_sockfd < 0)
		{
			lg.LogMessage(Fatal, "socket errr, %d : %s\n", errno,
				strerror(errno));
			exit(Socket_Err);
		}

		lg.LogMessage(Info, "socket success, sockfd: %d\n",
			_sockfd);

		// 2. 绑定,指定网络信息
		struct sockaddr_in local;
		bzero(&local, sizeof(local)); // memset
		local.sin_family = AF_INET;
		local.sin_port = htons(_port);
		local.sin_addr.s_addr = INADDR_ANY; // 0

		// local.sin_addr.s_addr = inet_addr(_ip.c_str()); // 1. 4字节 IP 2. 变成网络序列
		// 结构体填完,设置到内核中了吗??没有
		int n = ::bind(_sockfd, (struct sockaddr*)&local,
			sizeof(local));
		if (n != 0)
		{
			lg.LogMessage(Fatal, "bind errr, %d : %s\n", errno,
				strerror(errno));
			exit(Bind_Err);
		}
	}

	void Start()
	{
		// 服务器永远不退出
		char buffer[defaultsize];
		for (;;)
		{
			struct sockaddr_in peer;
			socklen_t len = sizeof(peer); // 不能乱写
			ssize_t n = recvfrom(_sockfd, buffer, sizeof(buffer) -
				1, 0, (struct sockaddr*)&peer, &len);
			if (n > 0)
			{
				InetAddr addr(peer);
				buffer[n] = 0;

				std::cout << "[" << addr.PrintDebug() << "]# " <<
					buffer << std::endl;

				std::string value;
				_func(buffer, &value); // 回调业务翻译方法
				sendto(_sockfd, value.c_str(), value.size(), 0,
					(struct sockaddr*)&peer, len);
			}
		}
	}

	~UdpServer()
	{
	}

private:
	// std::string _ip; // 后面要调整
	uint16_t _port;
	int _sockfd;
	func_t _func;
};

Main.cc

#include "UdpServer.hpp"
#include "Comm.hpp"
#include "Dict.hpp"
#include <memory>

void Usage(std::string proc)
{
	std::cout << "Usage : \n\t" << proc << " local_port\n" <<
		std::endl;
}

Dict gdict("./dict.txt");
void Execute(const std::string& req, std::string* resp)
{
	*resp = gdict.Translate(req);
}

// ./udp_server 8888
int main(int argc, char* argv[])
{
	if (argc != 2)
	{
		Usage(argv[0]);

		return Usage_Err;
	}

	// std::string ip = argv[1];
	uint16_t port = std::stoi(argv[1]);
	std::unique_ptr<UdpServer> usvr =
		std::make_unique<UdpServer>(Execute, port);

	usvr->Init();
	usvr->Start();

	return 0;
}

1.3 -> V2版本 -DictServer(封装版)

udp_socket.hpp

#pragma once
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <cassert>
#include <string>
#include <unistd.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

typedef struct sockaddr sockaddr;
typedef struct sockaddr_in sockaddr_in;

class UdpSocket 
{
public:
	UdpSocket() : fd_(-1) 
	{
	}

	bool Socket() 
	{
		fd_ = socket(AF_INET, SOCK_DGRAM, 0);
		if (fd_ < 0) 
		{
			perror("socket");

			return false;
		}

		return true;
	}

	bool Close() 
	{
		close(fd_);

		return true;
	}

	bool Bind(const std::string& ip, uint16_t port) 
	{
		sockaddr_in addr;

		addr.sin_family = AF_INET;
		addr.sin_addr.s_addr = inet_addr(ip.c_str());
		addr.sin_port = htons(port);

		int ret = bind(fd_, (sockaddr*)&addr, sizeof(addr));
		if (ret < 0) 
		{
			perror("bind");

			return false;
		}

		return true;
	}

	bool RecvFrom(std::string* buf, std::string* ip = NULL,
		uint16_t* port = NULL) 
	{
		char tmp[1024 * 10] = { 0 };
		sockaddr_in peer;

		socklen_t len = sizeof(peer);
		ssize_t read_size = recvfrom(fd_, tmp,
			sizeof(tmp) - 1, 0,
			(sockaddr*)&peer, &len);
		if (read_size < 0) 
		{
			perror("recvfrom");

			return false;
		}

		// 将读到的缓冲区内容放到输出参数中
		buf->assign(tmp, read_size);
		if (ip != NULL) 
		{
			*ip = inet_ntoa(peer.sin_addr);
		}

		if (port != NULL) 
		{
			*port = ntohs(peer.sin_port);
		}

		return true;
	}

	bool SendTo(const std::string& buf, const std::string& ip,
		uint16_t port) 
	{
		sockaddr_in addr;

		addr.sin_family = AF_INET;
		addr.sin_addr.s_addr = inet_addr(ip.c_str());
		addr.sin_port = htons(port);

		ssize_t write_size = sendto(fd_, buf.data(), buf.size(), 0,
			(sockaddr*)&addr, sizeof(addr));
		if (write_size < 0) 
		{
			perror("sendto");

			return false;
		}

		return true;
	}

private:
	int fd_;
};

UDP通用服务器

udp_server.hpp

#pragma once
#include "udp_socket.hpp"

// C 式写法
// typedef void (*Handler)(const std::string& req, std::string*resp);
// C++ 11 式写法, 能够兼容函数指针, 仿函数, 和 lambda

#include <functional>

typedef std::function<void(const std::string&, std::string*
	resp)> Handler;

class UdpServer 
{
public:
	UdpServer() 
	{
		assert(sock_.Socket());
	}

	~UdpServer() 
	{
		sock_.Close();
	}

	bool Start(const std::string& ip, uint16_t port, Handler
		handler) 
	{
		// 1. 创建 socket
		// 2. 绑定端口号
		bool ret = sock_.Bind(ip, port);
		if (!ret) 
		{
			return false;
		}

		// 3. 进入事件循环
		for (;;) 
		{
			// 4. 尝试读取请求
			std::string req;
			std::string remote_ip;

			uint16_t remote_port = 0;
			bool ret = sock_.RecvFrom(&req, &remote_ip, &remote_port);
			if (!ret) 
			{
				continue;
			}

			std::string resp;
			// 5. 根据请求计算响应
			handler(req, &resp);
			// 6. 返回响应给客户端
			sock_.SendTo(resp, remote_ip, remote_port);
			printf("[%s:%d] req: %s, resp: %s\n", remote_ip.c_str(),
				remote_port,
				req.c_str(), resp.c_str());
		}
		sock_.Close();

		return true;
	}

private:
	UdpSocket sock_;
};

实现英译汉服务器

以上代码是对udp服务器进行通用接口的封装。基于以上封装,实现一个查字典的服务器就很容易了。

dict_server.cc

#include "udp_server.hpp"
#include <unordered_map>
#include <iostream>

std::unordered_map<std::string, std::string> g_dict;

void Translate(const std::string& req, std::string* resp) 
{
	auto it = g_dict.find(req);
	if (it == g_dict.end()) 
	{
		*resp = "未查到!";

		return;
	}

	*resp = it->second;
}

int main(int argc, char* argv[]) 
{
	if (argc != 3) 
	{
		printf("Usage ./dict_server [ip] [port]\n");

		return 1;
	}

	// 1. 数据初始化
	g_dict.insert(std::make_pair("hello", "你好"));
	g_dict.insert(std::make_pair("world", "世界"));
	g_dict.insert(std::make_pair("c++", "最好的编程语言"));
	g_dict.insert(std::make_pair("bit", "特别 NB"));

	// 2. 启动服务器
	UdpServer server;
	server.Start(argv[1], atoi(argv[2]), Translate);

	return 0;
}

UDP通用客户端

udp_client.hpp

#pragma once
#include "udp_socket.hpp"

class UdpClient 
{
public:
	UdpClient(const std::string& ip, uint16_t port) : ip_(ip),
		port_(port) 
	{
		assert(sock_.Socket());
	}

	~UdpClient() 
	{
		sock_.Close();
	}

	bool RecvFrom(std::string* buf) 
	{
		return sock_.RecvFrom(buf);
	}

	bool SendTo(const std::string& buf) 
	{
		return sock_.SendTo(buf, ip_, port_);
	}

private:
	UdpSocket sock_;
	// 服务器端的 IP 和 端口号
	std::string ip_;
	uint16_t port_;
};

实现英译汉客户端

#include "udp_client.hpp"
#include <iostream>

int main(int argc, char* argv[]) 
{
	if (argc != 3) 
	{
		printf("Usage ./dict_client [ip] [port]\n");

		return 1;
	}

	UdpClient client(argv[1], atoi(argv[2]));
	for (;;) 
	{
		std::string word;
		std::cout << "请输入您要查的单词: ";
		std::cin >> word;
		if (!std::cin) 
		{
			std::cout << "Good Bye" << std::endl;

			break;
		}

		client.SendTo(word);
		std::string result;
		client.RecvFrom(&result);
		std::cout << word << " 意思是 " << result << std::endl;
	}

	return 0;
}

感谢各位大佬支持!!!

互三啦!!!

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

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

相关文章

Java面试题库——多线程

1.并行和并发有什么区别&#xff1f; 并行&#xff1a;是指两个或多个事件在同一时刻发生&#xff0c;是在不同实体上的多个事件&#xff1b; 并发&#xff1a;是指两个或多个事件在同一时间间隔发生&#xff0c;是同一实体上的多个事件。2.线程和进程的区别&#xff1f; 根本…

数据结构修炼——常见的排序算法:插入/希尔/选择/堆排/冒泡/快排/归并/计数

目录 一、常见的排序算法二、常见排序算法的实现2.1 排序算法回顾2.1.1 冒泡排序2.1.2 堆排序 2.2 直接插入排序2.3 希尔排序2.4 选择排序2.5 快速排序2.5.1 快速排序&#xff08;霍尔法&#xff09;2.5.2 快速排序&#xff08;挖坑法&#xff09;2.5.3 快速排序&#xff08;前…

极客wordpress模板

这是一个展示WordPress主题的网页设计。页面顶部有一个导航栏&#xff0c;包含多个选项&#xff0c;如“关于我们”、“产品中心”、“案例展示”、“新闻动态”、“联系我们”和“技术支持”。页面中间部分展示了多个产品&#xff0c;每个产品都有一个图片和简短的描述。页面下…

【Linux】冯诺依曼体系结构 OS的概念

&#x1fa90;&#x1fa90;&#x1fa90;欢迎来到程序员餐厅&#x1f4ab;&#x1f4ab;&#x1f4ab; 主厨&#xff1a;邪王真眼 主厨的主页&#xff1a;Chef‘s blog 所属专栏&#xff1a;青果大战linux 总有光环在陨落&#xff0c;总有新星在闪烁 前言废话&#xff1a…

动态链接过程分析

目录 一、前言二、示例程序三、动态库的加载过程1、动态链接器加载动态库2、动态库的加载地址 四、符号重定位1、全局符号表2、全局偏移表 GOT3、liba.so 动态库文件的布局4、liba.so 动态库的虚拟地址5、GOT 表的内部结构6、反汇编 liba.so 代码 五、补充1、延迟绑定 plt 上文…

【ARM】ARM架构参考手册_Part B 内存和系统架构(5)

目录 5.1关于缓存和写缓冲区 5.2 Cache 组织 5.2.1 集联性&#xff08;Set-associativity&#xff09; 5.2.2 缓存大小 5.3 缓存类型 5.3.1 统一缓存或分离缓存 5.3.2 写通过&#xff08;Write-through&#xff09;或写回&#xff08;Write-back&#xff09;缓存 5.3.3…

基于R语言机器学习遥感数据处理与模型空间预测技术及实际项目案例分析

随机森林作为一种集成学习方法&#xff0c;在处理复杂数据分析任务中特别是遥感数据分析中表现出色。通过构建大量的决策树并引入随机性&#xff0c;随机森林在降低模型方差和过拟合风险方面具有显著优势。在训练过程中&#xff0c;使用Bootstrap抽样生成不同的训练集&#xff…

Linux环境配置(学生适用)

1.挑选最便宜的云服务器 如腾讯云服务器&#xff0c;华为云服务器&#xff0c;百度云服务器等等…… 2.找到你的云服务器实例&#xff0c;然后找到你的公网IP。 3.云服务器实例 ---更多 --- 重置root密码 (一定要重置&#xff09; 4. 下载并安装 xshell 或者其他登陆软件 xshel…

12. 命令行

Hyperf 的命令行默认由 hyperf/command 组件提供&#xff0c;而该组件本身也是基于 symfony/console 的抽象。 一、安装 通常来说该组件会默认存在&#xff0c;但如果您希望用于非 Hyperf 项目&#xff0c;也可通过下面的命令依赖 hyperf/command 组件。 composer require hype…

告别ELK,APO提供基于ClickHouse开箱即用的高效日志方案——APO 0.6.0发布

ELK一直是日志领域的主流产品&#xff0c;但是ElasticSearch的成本很高&#xff0c;查询效果随着数据量的增加越来越慢。业界已经有很多公司&#xff0c;比如滴滴、B站、Uber、Cloudflare都已经使用ClickHose作为ElasticSearch的替代品&#xff0c;都取得了不错的效果&#xff…

【Golang】Go语言中如何创建Cron定时任务

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

MySQL【知识改变命运】11

联合查询 6. ⼦查询6.1 语法6.2 单⾏⼦查询6.3 多⾏⼦查询6.4 多列⼦查询6.5 在from⼦句中使⽤⼦查询 7. 合并查询7.1 创建新表并初始化数据7.2 Union7.3 Union all 8. 插⼊查询结果8.1 语法8.2 ⽰例 6. ⼦查询 ⼦查询是把⼀个SELECT语句的结果当做别⼀个SELECT语句的条件&…

10.22 MySQL

存储过程 存储函数 存储函数是有返回值的存储过程&#xff0c;存储函数的参数只能是in类型的。具体语法如下&#xff1a; characteristic 特性 练习&#xff1a; 从1到n的累加 ​​​​​​ create function fun1(n int) returns int deterministic begindeclare total i…

制氮机分子筛的作用

制氮机作为一种重要的工业设备&#xff0c;广泛应用于食品、饮料、化学、石油、电子和医疗保健等多个行业。其核心组件之一——分子筛。本文将详细探讨制氮机分子筛的作用及其重要性。 一、分子筛的基本概念 分子筛是一种具有均匀孔径的多孔材料&#xff0c;常用于气体分离和纯…

Elasticsearch 中的高效按位匹配

作者&#xff1a;来自 Elastic Alexander Marquardt 探索在 Elasticsearch 中编码和匹配二进制数据的六种方法&#xff0c;包括术语编码&#xff08;我喜欢的方法&#xff09;、布尔编码、稀疏位位置编码、具有精确匹配的整数编码、具有脚本按位匹配的整数编码以及使用 ESQL 进…

基于vue框架的的二手车交易系统的设计与实现thx7v(程序+源码+数据库+调试部署+开发环境)系统界面在最后面。

系统程序文件列表 项目功能&#xff1a;用户,卖家,车辆类型,二手车,在线留言,订单信息 开题报告内容 基于Vue框架的二手车交易系统的设计与实现开题报告 一、课题背景及意义 随着汽车消费市场的日益成熟与消费者换车频率的增加&#xff0c;二手车交易市场逐渐成为汽车市场的…

pycharm配置git版本控制

今天记录一下如何在pycharm工具中配置git版本控制&#xff0c;主要分以下步骤&#xff1a; 1、安装git 首先需要有git环境&#xff0c;去git官网下载git安装包&#xff0c;下一步下一步执行安装完成即可 2、在pycharm中配置git路径 下载git后&#xff0c;在pycharm的 setti…

「AIGC」n8n AI Agent开源的工作流自动化工具

n8n AI Agent 是一个利用大型语言模型(LLMs)来设计和构建智能体(agents)的工具,这些智能体能够执行一系列复杂的任务,如理解指令、模仿类人推理,以及从用户命令中理解隐含意图。n8n AI Agent 的核心在于构建一系列提示(prompts),使 LLM 能够模拟自主行为。 传送门→ …

GAMES104:17 游戏引擎的玩法系统:高级AI-学习笔记

文章目录 课前QA一&#xff0c;层次任务网络&#xff08;Hierarchical Tasks Network&#xff0c;HTN&#xff09;1.1 HTN Framework1.2 HTN Task Types1.2.1 Primitive Task基本任务1.2.2 Compound Task符合任务 1.3 Planning1.4 Replan1.5 总结 二&#xff0c;目标导向行为规…

在ECS实例上搭建WordPress博客平台

WordPress是使用PHP语言开发的博客平台&#xff0c;在支持PHP和MySQL数据库的服务器上&#xff0c;您可以用WordPress搭建自己的网站&#xff0c;也可以用作内容管理系统&#xff08;CMS&#xff09;。本教程介绍如何在不同操作系统的ECS实例上&#xff0c;手动搭建WordPress网…