- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
(这是在 GameDev 中错误发布的。感谢那里的版主在正确的位置提出问题!)
我正在将一个简单的结构转换为一个空指针,这样它就可以在网络上发送出去,但是由于我不知道的原因,它切断了我的第二个变量。
我发现了一些类似的主题,特别是 this one ,这是一个很大的帮助。我有理由相信这是一个“新手错误”,或者我忽略了一些非常简单的事情。
欢迎任何意见,提前致谢!
/* struct provided */
struct Login_Struct
{
/*000*/ char username[32];
/*032*/ char password[64];
/*096*/
};
void NetWrapper::sendLogin(std::string user, std::string pass)
{
// ensure size constraints are met
if( user.length() > sizeof(Login_Struct::username)
|| pass.length() > sizeof(Login_Struct::password)) { return; }
std::string hashPass = Security::strSHA256(pass); // returns hashed password
Login_Struct login; // empty struct
//strncpy(login.username, user.c_str(), user.length()); // i.e 'bob'
//strncpy(login.password, hashPass.c_str(), hashPass.length()); // hashed password (64 char)
memcpy(login.username, user.c_str(), user.length());
memcpy(login.password, hashPass.c_str(), hashPass.length());
char* pData = new char[sizeof(Login_Struct)];
//memcpy(static_cast<void*>(pData), static_cast<void*>(&login), sizeof(Login_Struct)); // serialize data for sending
memcpy(pData, &login, sizeof(Login_Struct)); // serialize data for sending
// if we have data, send it and cleanup
if(pData) { m_network->send(pData, OP_Login); delete pData; } // Network::send(char* d, short opCode)
pData = nullptr;
}
此外,调试输出看起来像这样(为了便于阅读从代码中省略,如果需要/请求可以提供):
NetWrapper::sendLogin: Login_Struct Dump [bob] (bob), [5e884898da28047151d0e56f8dc6292773603d0d6aabbdd62a11ef721d1542d8] (password)
DEBUG: username maxlength [32], user length [3]
DEBUG: password maxlength [64], pass length [64]
DEBUG: pData [x=�i=], [0x7ffc5e4fd0e0], sizeof(Login_Struct) [96], sizeof(login) [96]
NetWrapper::sendLogin: New pData! pData [x=�i=]
NetWrapper::sendLogin: Post memcpy! pData [bob]
Network::send : Sending [d0c8504000100bob] ## ignore characters preceding "bob"
@wondra 在我位置不正确的帖子中提到,这是空终止符为我杀死它的问题。我已进行调整以复制没有空终止符的字符串,但我要么误解了它的放置方式,要么这不是问题所在。
编辑:添加带有调试废话的代码以帮助理解这一点。感谢到目前为止所有回复的人! (是的,我对使用 cout spam 感到尴尬。愿意接受关于更好/更清洁方法的建议!)
void NetWrapper::sendLogin(std::string user, std::string pass)
{
if( user.length() > sizeof(Login_Struct::username)
|| pass.length() > sizeof(Login_Struct::password)) { return; }
std::string hashPass = Security::strSHA256(pass);
Login_Struct login;
memset(&login, 0, sizeof(Login_Struct)); // clear buffer
strncpy(login.username, user.c_str(), user.length());
strncpy(login.password, hashPass.c_str(), hashPass.length());
//memcpy(login.username, user.c_str(), user.length());
//memcpy(login.password, hashPass.c_str(), hashPass.length());
std::cout<<"NetWrapper::sendLogin: Login_Struct Dump ["<<login.username<<"] ("<<user<<"), ["<<login.password<<"] ("<<pass<<")\n";
std::cout<<"DEBUG: username maxlength ["<<sizeof(login.username)<<"], user length ["<<user.length()<<"], user size ["<<sizeof(user)<<"]\n";
std::cout<<"DEBUG: password maxlength ["<<sizeof(login.password)<<"], pass length ["<<hashPass.length()<<"], pass size ["<<sizeof(hashPass)<<"]\n";
char* pData = new char[sizeof(Login_Struct)];
std::cout<<"DEBUG: pData ["<<pData<<"], sizeof(pData) ["<<sizeof(pData)<<"], sizeof(&pData) ["<<sizeof(&pData)<<"], ["<<static_cast<void*>(&login)<<"], sizeof(Login_Struct) ["<<sizeof(Login_Struct)<<"], sizeof(login) ["<<sizeof(login)<<"]\n";
std::cout<<"NetWrapper::sendLogin: New pData! pData ["<<pData<<"], sizeof(pData) ["<<sizeof(pData)<<"], sizeof(&pData) ["<<sizeof(&pData)<<"]\n";
/// ------------------
//memcpy(static_cast<void*>(pData), static_cast<void*>(&login), sizeof(Login_Struct)); // serialize data for sending
memcpy(pData, &login, sizeof(Login_Struct)); // serialize data for sending
/// ------------------
std::cout<<"NetWrapper::sendLogin: Post memcpy! pData ["<<pData<<"], sizeof(pData) ["<<sizeof(pData)<<"], sizeof(&pData) ["<<sizeof(&pData)<<"], sizeof(*pData) ["<<sizeof(*pData)<<"]\n";
// if we have data, send it and cleanup
if(pData) { m_network->send(pData, OP_Login); delete pData; }
pData = nullptr;
}
最佳答案
从问题中不清楚您的问题到底是什么。不过,我发现有几个潜在的问题。
您查看代码是否正常的方法将不起作用。
std::cout<<"NetWrapper::sendSerialized: Post memcpy! pData ["<<pData<<"], sizeof(pData) ["<<sizeof(pData)<<"], sizeof(&pData) ["<<sizeof(&pData)<<"], sizeof(*pData) ["<<sizeof(*pData)<<"]\n";
char 数组 pData
将包含两个包含两个字符串的 char 缓冲区。如果您的测试名称是“bob”,那么后面会有一个空字符,这就是它将使用 cout
显示的全部内容。您需要一个函数来转储缓冲区的内容,或者可能执行类似 printf("password='%.*s'", 64, pData + 32)
的操作。那是你要弄明白的。
另一个问题是您正在使用 memcpy()
但之前没有清除缓冲区的内存。在您复制的字符串之后缓冲区中的内容是未定义的。
您通过网络发送它,这样您的字符串就不会以空值终止。那可能没问题,也可能不行。这取决于预期的结果。如果接收端的某些东西期望有效的空终止 C 字符串,那肯定会导致问题。或者也尝试对它们使用类似 printf()
的东西。
这将解决不首先清除内存的问题,假设它发送 not 作为空终止字符串:
memset(&login, 0, sizeof(login));
strncpy(login.username, user.c_str(), sizeof(login.username));
strncpy(login.password, hashPass.c_str(), sizeof(login.password));
第一个 memset()
将所有字节清除为 0,而 strncpy()
确保复制的字符数不超过目标缓冲区中可用的字符数。
如果问题是它确实需要一个空终止的 C 字符串,那么您需要将 +1
添加到 char 数组的大小以容纳空终止符,然后将它们复制为:
memset(&login, 0, sizeof(login));
strncpy(login.username, user.c_str(), sizeof(login.username)-1);
strncpy(login.password, hashPass.c_str(), sizeof(login.password)-1);
关于C++ 类到 Void 指针,数据丢失?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44166789/
我刚接触 C 语言几周,所以对它还很陌生。 我见过这样的事情 * (variable-name) = -* (variable-name) 在讲义中,但它到底会做什么?它会否定所指向的值吗? 最佳答案
我有一个指向内存地址的void 指针。然后,我做 int 指针 = void 指针 float 指针 = void 指针 然后,取消引用它们以获取值。 { int x = 25; vo
我正在与计算机控制的泵进行一些串行端口通信,我用来通信的 createfile 函数需要将 com 端口名称解析为 wchar_t 指针。 我也在使用 QT 创建一个表单并获取 com 端口名称作为
#include "stdio.h" #include "malloc.h" int main() { char*x=(char*)malloc(1024); *(x+2)=3; --
#include #include main() { int an_int; void *void_pointer = &an_int; double *double_ptr = void
对于每个时间步长,我都有一个二维矩阵 a[ix][iz],ix 从 0 到 nx-1 和 iz 从 0 到 nz-1。 为了组装所有时间步长的矩阵,我定义了一个长度为 nx*nz*nt 的 3D 指针
我有一个函数,它接受一个指向 char ** 的指针并用字符串填充它(我猜是一个字符串数组)。 *list_of_strings* 在函数内部分配内存。 char * *list_of_strings
我试图了解当涉及到字符和字符串时,内存分配是如何工作的。 我知道声明的数组的名称就像指向数组第一个元素的指针,但该数组将驻留在内存的堆栈中。 另一方面,当我们想要使用内存堆时,我们使用 malloc,
我有一个 C 语言的 .DLL 文件。该 DLL 中所有函数所需的主要结构具有以下形式。 typedef struct { char *snsAccessID; char *
指针, C语言的精髓 莫队先咕几天, 容我先讲完树剖 (因为后面树上的东西好多都要用树剖求 LCA). 什么是指针 保存变量地址的变量叫做指针. 这是大概的定义, 但是Defad认为
我得到了以下数组: let arr = [ { children: [ { children: [], current: tru
#include int main(void) { int i; int *ptr = (int *) malloc(5 * sizeof(int)); for (i=0;
我正在编写一个程序,它接受一个三位数整数并将其分成两个整数。 224 将变为 220 和 4。 114 将变为 110 和 4。 基本上,您可以使用模数来完成。我写了我认为应该工作的东西,编译器一直说
好吧,我对 C++ 很陌生,我确定这个问题已经在某个地方得到了回答,而且也很简单,但我似乎找不到答案.... 我有一个自定义数组类,我将其用作练习来尝试了解其工作原理,其定义如下: 标题: class
1) this 指针与其他指针有何不同?据我了解,指针指向堆中的内存。如果有指向它们的指针,这是否意味着对象总是在堆中构造? 2)我们可以在 move 构造函数或 move 赋值中窃取this指针吗?
这个问题在这里已经有了答案: 关闭 11 年前。 Possible Duplicate: C : pointer to struct in the struct definition 在我的初学者类
我有两个指向指针的结构指针 typedef struct Square { ... ... }Square; Square **s1; //Representing 2D array of say,
变量在内存中是如何定位的?我有这个代码 int w=1; int x=1; int y=1; int z=1; int main(int argc, char** argv) { printf
#include #include main() { char *q[]={"black","white","red"}; printf("%s",*q+3); getch()
我在“C”类中有以下函数 class C { template void Func1(int x); template void Func2(int x); }; template void
我是一名优秀的程序员,十分优秀!