是MSVC 2010中的错误还是我做错了什么

Is it a bug in MSVC 2010 or I am doing something wrong?

本文关键字:错了 什么 错误 MSVC 2010      更新时间:2023-10-16

请考虑以下代码段:

在database_sqlite.h中:

class __declspec(dllexport) SQLiteDatabase : public Database
{
    virtual void GetTableProperties(DatabaseTable *table, std::vector<std::wstring> &errorMsg);
protected:
    struct SQLiteImpl;
};
struct SQLiteDatabase::SQLiteImpl
{
    std::wstring m_catalog;
    std::wstring_convert<std::codecvt_utf8<wchar_t> > m_myconv;
};

在database_sqlite.cpp中:

void SQLiteDatabase::GetTableProperties(DatabaseTable *table, std::vector<std::wstring> &errorMsg)
{
    sqlite3_stmt *stmt = NULL;
    std::wstring errorMessage;
    int result;
    std::wstring query = L"SELECT * FROM "sys.abcattbl" WHERE "abt_tnam" = ? AND "abt_ownr" = ?;";
    const unsigned char *dataFontName, *headingFontName, *labelFontName;
    int res = sqlite3_prepare_v2( m_db, sqlite_pimpl->m_myconv.to_bytes( query.c_str() ).c_str(), (int) query.length(), &stmt, 0 );
    if( res == SQLITE_OK )
    {
const char *name = sqlite_pimpl->m_myconv.to_bytes( table->GetTableName().c_str() ).c_str(); // I used this line for debugging purposes
    res = sqlite3_bind_text( stmt, 1, sqlite_pimpl->m_myconv.to_bytes( table->GetTableName().c_str() ).c_str(), -1, SQLITE_STATIC );
    if( res == SQLITE_OK )
    {
        res = sqlite3_bind_text( stmt, 2, sqlite_pimpl->m_myconv.to_bytes( table->GetSchemaName().c_str() ).c_str(), -1, SQLITE_STATIC );
        if( res == SQLITE_OK )
        {
            while( true )
            {

char *result_query = sqlite3_expanded_sql(stmt); res = sqlite3_step(stmt);

在这里 * result_query contains =" select * select * sys.abcattbl"其中" abt_tnam" ='''''''and abt_ownr" ='''';";我正在使用MSVC2010,令我惊讶的是,"*名称"包含空字符串。第一个呼吁" to_bytes()"成功,因为我可以检查将要使用的查询。

在第一次打电话给" to_bytes()"之后,我必须做些什么吗?或者我只需要升级编译器?

nope,你做错了什么。

我现在没有to_bytes返回什么,但是我假设它是std::basic_string上的类型基础,因此std::string和family。

to_bytes将临时对象返回到该字符串实例,因此,当您调用c_str()时,您将指示指向字符串的第一个字符,该字符串在语句完成后不再存在。由于name是一个悬空的指针,因此您基本上具有不确定的行为,并且您试图输出它。

您必须分配内存并将字符串复制到其中,或者仅使用std::string,在哪里不会发生问题类型。

您呼叫sqlite3_prepare_v2成功的原因是to_bytes返回的字符串仍然存在。语句结束时,该字符串将被销毁,因此当res分配sqlite3_prepare_v2的返回值时。不是以前,因此呼叫成功,因为传递了一个有效的指针。

,尽管如此,请升级您的编译器。