gpt4 book ai didi

c++ - 将字符串或字符数组添加到字节 vector

转载 作者:行者123 更新时间:2023-11-30 01:00:23 32 4
gpt4 key购买 nike

我目前正在研究一个类来创建和读出通过网络发送的数据包,到目前为止我已经使用 16 位和 8 位整数(未签名但仍然如此)。

现在的问题是我已经尝试了多种方法来复制它,但不知何故 _buffer 被破坏了,它出现了段错误,或者结果是错误的。

如果有人可以向我展示一个工作示例,我将不胜感激。

我当前的代码可以在下面看到。

谢谢,施罗斯

主要内容

#include <iostream>
#include <stdio.h>
#include "Packet.h"

using namespace std;

int main(int argc, char** argv)
{
cout << "#################################" << endl;
cout << "# Internal Use Only #" << endl;
cout << "# Codename PACKETSTORM #" << endl;
cout << "#################################" << endl;
cout << endl;

Packet packet = Packet();
packet.SetOpcode(0x1f4d);

cout << "Current opcode is: " << packet.GetOpcode() << endl << endl;

packet.add(uint8_t(5))
.add(uint16_t(4000))
.add(uint8_t(5));

for(uint8_t i=0; i<10;i++)
printf("Byte %u = %x\n", i, packet._buffer[i]);

printf("\nReading them out: \n1 = %u\n2 = %u\n3 = %u\n4 = %s",
packet.readUint8(),
packet.readUint16(),
packet.readUint8());

return 0;
}

数据包.h

#ifndef _PACKET_H_
#define _PACKET_H_

#include <iostream>
#include <vector>

#include <stdio.h>
#include <stdint.h>
#include <string.h>

using namespace std;

class Packet
{
public:
Packet() : m_opcode(0), _buffer(0), _wpos(0), _rpos(0) {}
Packet(uint16_t opcode) : m_opcode(opcode), _buffer(0), _wpos(0), _rpos(0) {}

uint16_t GetOpcode() { return m_opcode; }
void SetOpcode(uint16_t opcode) { m_opcode = opcode; }

Packet& add(uint8_t value)
{
if(_buffer.size() < _wpos + 1)
_buffer.resize(_wpos + 1);

memcpy(&_buffer[_wpos], &value, 1);
_wpos += 1;

return *this;
}
Packet& add(uint16_t value)
{
if(_buffer.size() < _wpos + 2)
_buffer.resize(_wpos + 2);

memcpy(&_buffer[_wpos], &value, 2);
_wpos += 2;

return *this;
}

uint8_t readUint8()
{
uint8_t result = _buffer[_rpos];
_rpos += sizeof(uint8_t);
return result;
}
uint16_t readUint16()
{
uint16_t result;
memcpy(&result, &_buffer[_rpos], sizeof(uint16_t));

_rpos += sizeof(uint16_t);
return result;

}

uint16_t m_opcode;
std::vector<uint8_t> _buffer;
protected:

size_t _wpos; // Write position
size_t _rpos; // Read position
};

#endif // _PACKET_H_

最佳答案

由于您正在为缓冲区使用 std::vector,您不妨让它自己跟踪写入位置,避免必须手动调整它的大小。您还可以使用函数模板避免编写 add 函数的多个重载:

template <class T>
Packet& add(T value) {
std::copy((uint8_t*) &value, ((uint8_t*) &value) + sizeof(T), std::back_inserter(_buffer));
return *this;
}

现在您可以将任何 POD 类型写入您的缓冲区。

隐式地:

int i = 5;
o.write(i);

或明确地:

o.write<int>(5);

要从缓冲区读取,您需要跟踪读取位置:

template <class T>
T read() {
T result;
uint8_t *p = &_buffer[_rpos];
std::copy(p, p + sizeof(T), (uint8_t*) &result);
_rpos += sizeof(T);
return result;
}

您需要显式传递一个类型参数才能读取。即

int i = o.read<int>();

警告:我经常使用这种模式,但由于我是凭空输入,代码中可能存在一些错误。

编辑:我刚刚注意到您希望能够将字符串或其他非 POD 类型添加到您的缓冲区。您可以通过模板特化来做到这一点:

template <>
Packet& add(std::string s) {
add(string.length());
for (size_t i = 0; i < string.length(); ++i)
add(string[i]);
return *this;
}

这告诉编译器:如果使用字符串类型调用 add,请使用此函数而不是通用的 add() 函数。

读取一个字符串:

template <>
std::string read<>() {
size_t len = read<size_t>();
std::string s;
while (len--)
s += read<char>();
return s;
}

关于c++ - 将字符串或字符数组添加到字节 vector ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2729280/

32 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com