我在C#中有以下代码:
结构体:
[StructLayout(LayoutKind.Sequential, Pack = 1, CharSet = CharSet.Ansi, Size = 116)]
public struct pLogin
{
public pHeader _header;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 12)]
public string senha;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 12)]
public string login;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 8)]
public byte[] unk1;
public int algo1;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 42)]
public byte[] unk2;
public short algo2;
//versao do cliente
public ushort cliver;
public ushort unk3;
public int umBool;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 16)]
public byte[] mac;
}
[StructLayout(LayoutKind.Sequential, Pack = 1, CharSet = CharSet.Ansi, Size = 12)]
public struct pHeader
{
public ushort size;
public byte key;
public byte checksum;
public ushort packetId;
public ushort clientId;
public uint timestamp;
}
登录功能:
pLogin pLogin;
public void iniciarLogin(string login, string senha, int cliver, string p_mac = "")
{
pLogin = new pLogin();
pLogin._header = buildHeader(0x20D, 116);
pLogin.senha = senha;
pLogin.login = login;
pLogin.cliver = (ushort)cliver;
pLogin.umBool = 1;
pLogin.algo1 = 132;
pLogin.algo2 = 152;
if (p_mac.Length == 0)
{
pLogin.mac = Encoding.ASCII.GetBytes(Functions.RandomString(16));
}
else
{
pLogin.mac = Functions.StringToByteArray(p_mac);
}
byte[] buffer = BufferConverter.StructureToBuffer<pLogin>(pLogin);
EncDec.Encrypt(ref buffer);
socket.Send(BufferConverter.StringToByteArray("11F3111F"));
socket.Send(buffer);
logger.Text += "[Cliente] Solicitando login...\n";
}
pHeader packetHeader;
private pHeader buildHeader(int _packetID, int size)
{
packetHeader = new pHeader();
packetHeader.size = (ushort)size;
packetHeader.key = EncDec.GetHashByte();
packetHeader.checksum = 0;
packetHeader.packetId = (ushort)_packetID;
packetHeader.clientId = (ushort)serverData.playerMob.Mob.ClientId;
packetHeader.timestamp = getCurrentTime();
return packetHeader;
}
现在,缓冲区转换器类:
public static Byte[] StructureToBuffer<T>(T structure)
{
Byte[] buffer = new Byte[Marshal.SizeOf(typeof(T))];
unsafe
{
fixed (byte* pBuffer = buffer)
{
Marshal.StructureToPtr(structure, new IntPtr((void*)pBuffer), true);
}
}
return buffer;
}
public static T BufferToStructure<T>(Byte[] buffer, Int32 offset)
{
unsafe
{
fixed (Byte* pBuffer = buffer)
{
return (T)Marshal.PtrToStructure(new IntPtr((void*)&pBuffer[offset]), typeof(T));
}
}
}
上面的代码从一个结构中创建一个带有登录数据的字节数组。
有没有办法在python中对缓冲区数组进行序列化/反序列化? -
我不知道如何在python中执行此操作,因为我看不到很多有关字节数组内容的文章。
参考方案
当然有两种方法。
有built-in struct
module,需要一些手动工作才能找出结构的格式字符串。
您还可以使用更高级别的第三方库,例如construct
(我推荐)。
使用Construct,您的结构可能看起来像
Header = Struct(
'size' / Int16ub,
'key' / Int8ub,
'checksum' / Int8ub,
'packetId' / Int16ub,
'clientId' / Int16ub,
'timestamp' / Int32ub,
)
Login = Struct(
"header" / Header,
# ...
)
–对C#原始文件进行相当直接的翻译,并给定数据缓冲区,然后您可以执行以下操作
login_data = Login.parse(buffer)
Python sqlite3数据库已锁定 - python我在Windows上使用Python 3和sqlite3。我正在开发一个使用数据库存储联系人的小型应用程序。我注意到,如果应用程序被强制关闭(通过错误或通过任务管理器结束),则会收到sqlite3错误(sqlite3.OperationalError:数据库已锁定)。我想这是因为在应用程序关闭之前,我没有正确关闭数据库连接。我已经试过了: connectio…
python-docx应该在空单元格已满时返回空单元格 - python我试图遍历文档中的所有表并从中提取文本。作为中间步骤,我只是尝试将文本打印到控制台。我在类似的帖子中已经看过scanny提供的其他代码,但是由于某种原因,它并没有提供我正在解析的文档的预期输出可以在https://www.ontario.ca/laws/regulation/140300中找到该文档from docx import Document from…
Python:集群作业管理 - python我在具有两个阶段的计算群集(Slurm)上运行python脚本,它们是顺序的。我编写了两个python脚本,一个用于阶段1,另一个用于阶段2。每天早上,我检查所有第1阶段的工作是否都以视觉方式完成。只有这样,我才开始第二阶段。通过在单个python脚本中组合所有阶段和作业管理,是否有一种更优雅/自动化的方法?我如何知道工作是否完成?工作流程类似于以下内容:w…
Python-Excel导出 - python我有以下代码:import pandas as pd import requests from bs4 import BeautifulSoup res = requests.get("https://www.bankier.pl/gielda/notowania/akcje") soup = BeautifulSoup(res.cont…
Python:传递记录器是个好主意吗? - python我的Web服务器的API日志如下:started started succeeded failed 那是同时收到的两个请求。很难说哪一个成功或失败。为了彼此分离请求,我为每个请求创建了一个随机数,并将其用作记录器的名称logger = logging.getLogger(random_number) 日志变成[111] started [222] start…