gpt4 book ai didi

C++,一个 "impossible"行为

转载 作者:塔克拉玛干 更新时间:2023-11-02 23:57:18 27 4
gpt4 key购买 nike

如果您已经编程了一段时间,那么您可能会注意到时不时会发生一些完全不可能发生的事情,您确信对此没有可能的解释(“这是一个编译器错误!!”)。在你找出它是由什么引起的之后,虽然你就像“哦哦”。

嗯,这只是发生在我身上:(

这里 AuthDb 是NOT NULL 而是一个有效的指针:

SingleResult sr(AuthDb, format("SELECT Id, Access, Flags, SessionKey, RealmSplitPreference FROM accounts WHERE Name = '%s'") % Escaped(account_name));

在这里它神秘地变为 NULL:

struct SingleResult : public BaseResult
{
SingleResult(Database *db, const boost::format& query) { _ExecuteQuery(db, query.str()); }
}

请注意,这是紧接着的下一个电话。可以用两个屏幕截图更好地解释它: http://img187.imageshack.us/img187/5757/ss1zm.png
http://img513.imageshack.us/img513/5610/ss2b.png

编辑:AuthDb 是一个全局变量。它本身一直指向正确的东西;但是 ptr 数据库 *db 的拷贝指向 NULL。


ASM 代码(不幸的是我什至不知道如何阅读它:/)

第一张截图的

01214E06  mov         eax,dword ptr [ebp-328h] 
01214E0C push eax
01214E0D push offset string "SELECT Id, Access, Flags, Sessio"... (13C6278h)
01214E12 lea ecx,[ebp-150h]
01214E18 call boost::basic_format<char,std::char_traits<char>,std::allocator<char> >::basic_format<char,std::char_traits<char>,std::allocator<char> > (11A3260h)
01214E1D mov dword ptr [ebp-32Ch],eax
01214E23 mov ecx,dword ptr [ebp-32Ch]
01214E29 mov dword ptr [ebp-330h],ecx
01214E2F mov byte ptr [ebp-4],2
01214E33 mov ecx,dword ptr [ebp-330h]
01214E39 call boost::basic_format<char,std::char_traits<char>,std::allocator<char> >::operator%<Snow::Escaped> (11A3E18h)
01214E3E push eax
01214E3F mov edx,dword ptr [__tls_index (144EC40h)]
01214E45 mov eax,dword ptr fs:[0000002Ch]
01214E4B mov ecx,dword ptr [eax+edx*4]
01214E4E mov edx,dword ptr [ecx+12A3Ch]
01214E54 push edx
01214E55 lea ecx,[sr]
01214E58 call Snow::SingleResult::SingleResult (11A27D4h)
01214E5D mov byte ptr [ebp-4],4 // VS GREEN ARROW IS HERE
01214E61 lea ecx,[ebp-150h]
01214E67 call boost::basic_format<char,std::char_traits<char>,std::allocator<char> >::~basic_format<char,std::char_traits<char>,std::allocator<char> > (11A1DBBh)
01214E6C mov byte ptr [ebp-4],5
01214E70 lea ecx,[ebp-170h]
01214E76 call Snow::Escaped::~Escaped (11A42D2h)
const bool account_found = !sr.Error();
01214E7B lea ecx,[sr]
01214E7E call Snow::BaseResult::Error (11A2964h)
01214E83 movzx eax,al
01214E86 test eax,eax
01214E88 sete cl
01214E8B mov byte ptr [account_found],cl

if (!account_found) {
01214E8E movzx edx,byte ptr [account_found]
01214E92 test edx,edx
01214E94 jne AuthSession+1C0h (1214F10h)
client.Kill(format("%s: Attempted to login with non existant account `%s'") % client % account_name, true);

第二张截图

011A8E7D  mov         dword ptr [ebp-10h],ecx 
011A8E80 mov ecx,dword ptr [this]
011A8E83 call Snow::BaseResult::BaseResult (11A31D9h)
011A8E88 mov dword ptr [ebp-4],0
011A8E8F lea eax,[ebp-30h]
011A8E92 push eax
011A8E93 mov ecx,dword ptr [query]
011A8E96 call boost::basic_format<char,std::char_traits<char>,std::allocator<char> >::str (11A1E01h)
011A8E9B mov dword ptr [ebp-34h],eax
011A8E9E mov ecx,dword ptr [ebp-34h]
011A8EA1 mov dword ptr [ebp-38h],ecx
011A8EA4 mov byte ptr [ebp-4],1
011A8EA8 mov edx,dword ptr [ebp-38h]
011A8EAB push edx
011A8EAC mov eax,dword ptr [db]
011A8EAF push eax
011A8EB0 mov ecx,dword ptr [this]
011A8EB3 call Snow::SingleResult::_ExecuteQuery (124F380h)
011A8EB8 mov byte ptr [ebp-4],0 // VS GREEN ARROW HERE
011A8EBC lea ecx,[ebp-30h]
011A8EBF call std::basic_string<char,std::char_traits<char>,std::allocator<char> >::~basic_string<char,std::char_traits<char>,std::allocator<char> > (11A2C02h)
011A8EC4 mov dword ptr [ebp-4],0FFFFFFFFh
011A8ECB mov eax,dword ptr [this]
011A8ECE mov ecx,dword ptr [ebp-0Ch]
011A8ED1 mov dword ptr fs:[0],ecx
011A8ED8 pop edi
011A8ED9 add esp,38h
011A8EDC cmp ebp,esp
011A8EDE call _RTC_CheckEsp (12B4450h)
011A8EE3 mov esp,ebp
011A8EE5 pop ebp
011A8EE6 ret 8

更新根据 peterchen 的建议,我添加了 ASSERT(AuthDb);这里:

ASSERT(AuthDb);
SingleResult sr(AuthDb, format("SELECT Id, Access, Flags, SessionKey, RealmSplitPreference FROM accounts WHERE Name = '%s'") % Escaped(account_name));

它失败了 O.o 然而调试器一直坚持它不是 NULL..它没有被本地隐藏

更新 2* cout << AuthDb;那里是 0,即使调试器说它不是


找到问题

Database *AuthDb = NULL, *GameDb = NULL;在 .cpp 中

extern thread Database *AuthDb, *GameDb;在.h

变量在 header 中标记为线程(TLS - 线程本地存储),但在定义中未标记为 TLS...

无数小时浪费在这个 super 愚蠢的错误上,没有任何警告或提示或来自编译器的任何我现在想杀掉的东西。 :( 哦,好吧,就像我说的,对于每一个不可能的行为,都有一个解决方案,一旦已知似乎显而易见:)

感谢所有帮助过我的人,我真的很绝望!

最佳答案

AuthDB 是线程局部变量吗?

也许调试器没有正确处理它。如果在调用构造函数之前使用 ASSERT(AuthDB) 会怎样?

更新:如果它是线程本地的,它只是尚未在此线程中初始化。

关于C++,一个 "impossible"行为,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1793678/

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