如果通过指针访问此功能,为什么会导致访问冲突?

时间:2016-06-29 14:07:19

标签: c++ c++11 crash

我有以下代码:

void Aurora64::Messaging::SendConsoleMessageToPlayer(int channelId , const char *msg) 
{
    CGameRules *pGameRules = new CGameRules;
    pGameRules->SendTextMessage(eTextMessageConsole, msg, eRMI_ToClientChannel, channelId);
    delete(pGameRules);
}

它通过指针pGameRules访问另一个类中的函数,但该函数在调用时会导致程序崩溃。

此代码位于另一个文件的Aurora64类中。

将以下代码放在指针的类CGameRules内(在其中执行的函数中)时,程序崩溃:

CGameRules *pGameRules = new CGameRules;
pGameRules->SendTextMessage(eTextMessageConsole, "$4test", eRMI_ToClientChannel, channelId);
delete(pGameRules);

这是在试验Aurora64类文件中的某些内容是否导致崩溃(它不是)。

然而,当放在同一个类中时,程序可以正常运行而不会崩溃:

SendTextMessage(eTextMessageConsole, "$4test", eRMI_ToClientChannel, channelId);

除了指针之外,代码在功能上是相同的(使用相同的输入值)。

评论delete来电无效。

调用堆栈是:

Aurora64.dll!IGameObject::GetEntityId() Line 301    C+Aurora64.dll!IGameObject::InvokeRMI_Primitive<CGameRules::MethodInfo_ClTextMessage,
          CGameRules::TextMessageParams>(const CGameRules::MethodInfo_ClTextMessage method, 
          const CGameRules::TextMessageParams
            & params, unsigned int where,
          IRMIListener * pListener, int userId, int channel, unsigned int dependentId)
          Line 281  C++
Aurora64.dll!IGameObject::InvokeRMI<CGameRules::MethodInfo_ClTextMessage,
          CGameRules::TextMessageParams>(const
          CGameRules::MethodInfo_ClTextMessage method, const CGameRules::TextMessageParams 
          & params, unsigned int where, int channel) 
          Line 275  C++
Aurora64.dll!CGameRules::SendTextMessage(ETextMessageType type, const char * msg, unsigned int
          to, int channelId, const char * p0, const char * p1, const char * p2, 
          const char * p3)
          Line 3075 C++
Aurora64.dll!CGameRules::OnClientConnect(int channelId, bool isReset) Line 596  C++

我不明白为什么它会崩溃......代码应该在技术上有效,因为调用完全相同的函数,输入完全相同。

我有source-code on GitHub should it be required(它没有Aurora64类,因为它是基于GitHub的另一个项目,但源代码尚未更改)。它也可能需要调试Crysis Wars游戏和专用服务器包,这使得这里的任何人都难以调试。

这个问题可能是我很遗憾的事情。

我做错了什么?

SendTextMessage方法:

int CScriptBind_GameRules::SendTextMessage(IFunctionHandler *pH, int type, const char *msg)
{
    CGameRules *pGameRules=GetGameRules(pH);

    if (!pGameRules)
        return pH->EndFunction();

    int to=eRMI_ToAllClients;
    int channelId=-1;

    if (pH->GetParamCount()>2)
        pH->GetParam(3, to);

    if (pH->GetParamCount()>3)
    {
        if (pH->GetParamType(4)==svtPointer)
        {
            ScriptHandle playerId;
            pH->GetParam(4, playerId);

            channelId=pGameRules->GetChannelId((EntityId)playerId.n);
        }
        else if (pH->GetParamType(4)==svtNumber)
            pH->GetParam(4, channelId);
    }

    if (pH->GetParamCount()>4)
    {
        string p[4];
        for (int i=0;i<pH->GetParamCount()-4;i++)
        {
            switch(pH->GetParamType(5+i))
            {
            case svtPointer:
                {
                    ScriptHandle sh;
                    pH->GetParam(5+i, sh);

                    if (IEntity *pEntity=gEnv->pEntitySystem->GetEntity((EntityId)sh.n))
                        p[i]=pEntity->GetName();
                }
                break;
            default:
                {
                    ScriptAnyValue value;
                    pH->GetParamAny(5+i, value);
                    switch(value.GetVarType())
                    {
                    case svtNumber:
                        p[i].Format("%g", value.number);
                        break;
                    case svtString:
                        p[i]=value.str;
                        break;
                    case svtBool:
                        p[i]=value.b?"true":"false";
                        break;
                    default:
                        break;
                    }           
                }
                break;
            }
        }
        pGameRules->SendTextMessage((ETextMessageType)type, msg, to, channelId, 
            p[0].empty()?0:p[0].c_str(),
            p[1].empty()?0:p[1].c_str(),
            p[2].empty()?0:p[2].c_str(),
            p[3].empty()?0:p[3].c_str()
            );
    }
    else
        pGameRules->SendTextMessage((ETextMessageType)type, msg, to, channelId);

    return pH->EndFunction();
}

1 个答案:

答案 0 :(得分:1)

我忽略了特定项目的对象系统,正如molbdnilo所解释的那样:

  

可能你不能只创建一个CGameRules实例并开始   使用它 - 它必须正确设置。最有可能只有一个   在程序中,在启动时创建,你应该使用它。

我看了一下这个对象的项目,我找到了它:

CGameRules *pGameRules=g_pGame->GetGameRules();