重新创建/修复WinAPI ODBC连接

时间:2012-07-12 12:25:01

标签: c++ winapi visual-c++ odbc

我们创建并缓存使用SQLDriverConnect创建的ODBC连接;我们发现连接丢失的情况......准备好的语句停止工作等等。

我没有看到一个明显的功能来测试连接/语句是否有效,或者重置/重新创建连接,这似乎是一个相当常见的模式。谁能建议如何最好地实施这个?

This answer也是如此,这也是正确的解决方案吗?如果是这样,有什么方法可以“重新启动”现有语句以使用新连接?如果发现连接已经死了,它仍然需要被释放吗?

1 个答案:

答案 0 :(得分:0)

在我看来,对于SQL_ATTR_CONNECTION_DEAD是否能够在您进行要求它处于活动状态的呼叫之前测试连接是否仍然存在时,一直存在一些混淆。这是基于许多ODBC驱动程序的使用,包括我写的那些,我怀疑它们可能都以不同的方式实现它。

在帖子中你提到Nick的答案(SQL_ATTR_CONNECTION_DEAD)不适用于我有权访问的许多驱动程序:

#include <stdio.h>
#include <sql.h>
#include <sqlext.h>
#include <stdlib.h>

static void extract_error(
    char *fn,
    SQLHANDLE handle,
    SQLSMALLINT type);

main() {
    SQLHENV henv;
    SQLHDBC hdbc;
    SQLHSTMT hstmt;
    SQLCHAR outstr[1024];
    SQLSMALLINT         outstr_len;
    SQLRETURN ret;
    SQLINTEGER dead;

    SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &henv);
    SQLSetEnvAttr(henv, SQL_ATTR_ODBC_VERSION, (void *) SQL_OV_ODBC3, 0);
    SQLAllocHandle(SQL_HANDLE_DBC, henv, &hdbc);
    ret = SQLDriverConnect(hdbc, (void *)0, "DSN=xx;UID=xx;PWD=xx", SQL_NTS,
                           outstr, sizeof(outstr), &outstr_len,
                           SQL_DRIVER_COMPLETE);
    if (!SQL_SUCCEEDED(ret)) {
        extract_error("SQLDriverConnect", hdbc, SQL_HANDLE_DBC);
        exit(1);
    }

    SQLAllocHandle(SQL_HANDLE_STMT, hdbc, &hstmt);
    ret = SQLPrepare(hstmt, "select 123", SQL_NTS);
    if (!SQL_SUCCEEDED(ret)) {
        extract_error("SQLPrepare", hstmt, SQL_HANDLE_STMT);
        exit(1);
    }

    ret = SQLExecute(hstmt);
    if (!SQL_SUCCEEDED(ret)) {
        extract_error("SQLExecute", hstmt, SQL_HANDLE_STMT);
        exit(1);
    }

    sleep(120);

    SQLGetConnectAttr(hdbc, SQL_ATTR_CONNECTION_DEAD, &dead, 0, NULL);
    printf ("dead=%ld\n", dead);

    ret = SQLExecute(hstmt);
    if (!SQL_SUCCEEDED(ret)) {
        extract_error("SQLExecute", hstmt, SQL_HANDLE_STMT);
    }
    SQLGetConnectAttr(hdbc, SQL_ATTR_CONNECTION_DEAD, &dead, 0, NULL);
    printf ("dead=%ld\n", dead);


}


static void extract_error(
    char *fn,
    SQLHANDLE handle,
    SQLSMALLINT type)
{
    SQLINTEGER i = 0, native;
    SQLCHAR state[ 7 ];
    SQLCHAR text[256];
    SQLSMALLINT len;
    int ret;

    fprintf(stderr,
            "\n"
            "The driver reported the following diagnostics whilst running "
            "%s\n\n",
            fn);

    do
    {
        ret = SQLGetDiagRec(type, handle, ++i, state, &native, text,
                            sizeof(text), &len );
        if (SQL_SUCCEEDED(ret))
            printf( "%s:%ld:%ld:%s\n", state, i, native, text );
    }
    while( ret == SQL_SUCCESS );
}

与多个驱动程序一起输出:

dead=0
# At this point I took the server down
The driver reported the following diagnostics whilst running SQLExecute
08S01:1:0:[SQL Server Driver 10.0][SQL Server]Communication link failure: short write
dead=0

您引用的帖子中指定的另一个选项是SQL_COPT_SS_CONNECTION_DEAD,但这是MS SQL Server特有的。

由于我假设您不打算任何语句句柄失败,通常不能只运行stmt,如果失败则假设连接已经死并重新连接并重新映射?

如果连接确实死亡,您仍然需要为各种句柄调用SQLFreeHandle。