类型强制转换异常跟踪

2018-07-03

最近发现文件服务客户端 sdk 异常:上传文件,文件数据经常只传一部分就进入完成状态,经过跟踪调试,原来类型强制转换错误,程序没有抛异常。


1. 问题

进度逻辑里,pTask 指针对象原来是 CUploadTask 类型的,被强制转换成 CDownloadTask,程序正常运行,没有抛异常。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
// 下载任务
class CDownloadTask : public CTask {
public:
    CDownloadTask() : m_ullDownloadedSize(0) {}
    virtual ~CDownloadTask() {}

public:
    unsigned __int64 m_ullDownloadedSize; //已下载文件大小
    ...
};

// 上传任务
class CUploadTask : public CTask {
public:
    CUploadTask() : m_bPicResample(false), m_ullUploadedSize(0) {}
    virtual ~CUploadTask() {}

public:
    bool m_bPicResample;                     //是否有压缩图片
    std::string m_strFileType;               //文件类型
    unsigned __int64 m_ullUploadedSize;      //已上传文件大小
    ...
};

// 进度更新逻辑
bool CTaskMgr::UpdateTaskProgress(int iTaskID, 
               unsigned __int64 ullFileSize, unsigned __int64 ullNowSize) {
    CTask* pTask = NULL;
    if (m_mapTask.Lookup(iTaskID, pTask) && pTask != NULL) {
        CDownloadTask* pDLoadTask = (CDownloadTask*)pTask; // 出现问题语句。
        pDLoadTask->m_oFileInfo.m_ullFilesize = ullFileSize;
        pDLoadTask->m_ullDownloadedSize = ullNowSize;
        return true;
    }

    return false;
}

2. 解决方案

用安全的强制转换 dynamic_cast,它会在运行期对可疑的转型操作进行安全检查,类型不匹配会返回 NULL。

1
CDownloadTask* pDLoadTask = dynamic_cast<CDownloadTask*>(pTask);

3. 调试手段

如果不清楚变量是在哪里发生改变的,VS 系列,可以用“数据断点”功能,监控某个变量的改变。类似于 gdb 的 watch 功能。

调试手段 watch


4. 参考