# ODBC

# 1.ODBC驱动

# 1.1.概述

Ignite包括一个ODBC驱动,可以通过标准SQL查询和原生ODBC API查询和修改存储于分布式缓存中的数据。

要了解ODBC的细节,可以参照ODBC开发者参考

Ignite的ODBC驱动实现了ODBC API的3.0版。

# 1.2.集群配置

Ignite的ODBC驱动在Windows中被视为一个动态库,在Linux中被视为一个共享对象,应用不会直接加载它。作为替代,必要时它会使用一个驱动加载器API来加载和卸载ODBC驱动。

Ignite的ODBC驱动在内部使用TCP来接入Ignite集群,这个连接在Ignite中是通过一个叫做ClientListenerProcessor的组件来处理的。除了ODBC连接,它还处理JDBC连接以及瘦客户端连接。当节点启动时,ClientListenerProcessor默认是开启的,通过下面的代码可以对参数进行调整:

    配置了ClientListenerProcessor之后,就会以默认的配置启动,部分列举如下:

    属性 描述 默认值
    host 绑定的主机名或者IP地址,如果为null,会绑定localhost null
    port 绑定的TCP端口,如果指定的端口被占用,Ignite会使用portRange属性寻找其它的可用端口。 10800
    portRange 定义尝试绑定的端口范围。比如port配置为10800并且portRange100,那么服务端会按照顺序去尝试绑定[10800, 10900]范围内的端口,直到找到可用的端口。 100
    maxOpenCursorsPerConnection 单个连接可以同时打开的最大游标数。 128
    threadPoolSize 线程池中负责请求处理的线程数。 MAX(8, CPU核数)
    socketSendBufferSize TCP套接字发送缓冲区大小,如果配置为0,会使用系统默认值 0
    socketReceiveBufferSize TCP套接字接收缓冲区大小,如果配置为0,会使用系统默认值。 0
    tcpNoDelay 是否使用TCP_NODELAY选项。 true
    idleTimeout 客户端连接的空闲超时时间。如果空闲时间超过配置的超时时间,客户端会自动断开与服务端的连接。如果该参数配置为0或者为负值,空闲超时会被禁用。 0
    isOdbcEnabled 是否允许通过ODBC访问。 true
    isThinClientEnabled 是否允许通过瘦客户端访问。 true

    可以通过如下方式修改参数:

      通过ClientListenerProcessor从ODBC驱动端建立的连接也是可以配置的,关于如何从驱动端修改连接的配置,可以看这里

      # 1.3.线程安全

      Ignite ODBC驱动的当前实现仅在连接层提供了线程安全,这意味着如果没有额外的同步处理,多线程无法访问同一个连接。不过可以为每个线程创建独立的连接,然后同时使用。

      # 1.4.要求

      Ignite的ODBC驱动官方在如下环境中进行了测试:

      OS Windows(XP及以上,32位和64位版本)
      Windows Server(2008及以上,32位和64位版本)
      Ubuntu(14.x和15.x,64位)
      C++编译器 MS Visual C++ (10.0及以上), g++ (4.4.0及以上)
      Visual Studio 2010及以上

      # 1.5.构建ODBC驱动

      在Windows中,Ignite提供了预构建的32位和64位驱动的安装器,因此如果只是想在Windows中安装驱动,那么直接看下面的安装驱动章节就可以了。

      对于Linux环境,安装之前还是需要进行构建,因此如果使用的是Linux或者使用Windows但是仍然想自己构建驱动,那么往下看。

      Ignite的ODBC驱动的源代码随着Ignite版本一起发布,在使用之前可以自行进行构建。关于如何获取和设置Ignite本身,可以参照基本概念章节。

      因为ODBC驱动是用C++编写的,因此它是作为Ignite C++的一部分提供的,并且依赖于一些C++库,具体点说依赖于utilsbinaryIgnite库,这就意味着,在构建ODBC驱动本身之前,需要先构建它们。

      这里假定使用的是二进制版本,如果使用的是源代码版本,那么需要将所有使用的%IGNITE_HOME%\platforms\cpp替换为%IGNITE_HOME%\modules\platforms\cpp

      # 1.5.1.在Windows上构建

      如果要在Windows上构建ODBC驱动,需要MS Visual Studio 2010及以后的版本。一旦打开了Ignite方案%IGNITE_HOME%\platforms\cpp\project\vs\ignite.sln(或者ignite_86.sln,32位平台),在方案浏览器中点击odbc项目,然后选择“Build”,Visual Studio会自动地检测并且构建所有必要的依赖。

      .sln文件的路径可能会有所不同,具体取决于是从源文件还是从二进制文件进行构建。如果在%IGNITE_HOME%\platforms\cpp\project\vs\中找不到.sln文件,可以尝试在%IGNITE_HOME%\modules\platforms\cpp\project\vs\中查找。

      注意

      如果使用VS 2015及以后的版本(MSVC14.0及以后),需要将legacy_stdio_definitions.lib作为额外的库加入odbc项目的链接器配置以构建项目。要在IDE中将库文件加入链接器,可以打开项目节点的上下文菜单,选择Properties,然后在Project Properties对话框中,选择Linker,然后编辑Linker Input,这时就可以将legacy_stdio_definitions.lib加入分号分割的列表中。

      构建过程完成之后,会生成ignite.odbc.dll文件,对于64位版本,位于%IGNITE_HOME%\platforms\cpp\project\vs\x64\Release中,对于32位版本,位于%IGNITE_HOME%\platforms\cpp\project\vs\Win32\Release中。

      注意

      确认为系统使用相应的驱动(32位或64位)。

      在Windows中构建安装器

      为了简化安装,构建完驱动之后可能想构建安装器,Ignite使用WiX工具包来生成ODBC的安装器,因此需要下载并安装WiX,记得一定要把Wix工具包的bin目录加入PATH变量中。

      一切就绪之后,打开终端然后定位到%IGNITE_HOME%\platforms\cpp\odbc\install目录,按顺序执行如下的命令来构建安装器:

        完成之后,目录中会出现ignite-odbc-amd64.msiignite-odbc-x86.msi文件,然后就可以使用它们进行安装了。

        # 1.5.2.在Linux上构建

        在一个基于Linux的操作系统中,如果要构建及使用Ignite ODBC驱动,需要安装选择的ODBC驱动管理器,Ignite ODBC驱动已经使用UnixODBC进行了测试。

        要构建驱动及其依赖,还需要额外的GCC,G++以及Make

        如果所有必需的都安装好了,可以通过如下方式构建Ignite ODBC驱动:

        cd $IGNITE_HOME/platforms/cpp
        libtoolize && aclocal && autoheader && automake --add-missing && autoreconf
        ./configure --enable-odbc --disable-node --disable-core
        make
        
        #The following step will most probably require root privileges:
        make install
        

        构建过程完成后,可以通过如下命令找到ODBC驱动位于何处:

        whereis libignite-odbc
        

        路径很可能是:/usr/local/lib/libignite-odbc.so

        # 1.6.安装ODBC驱动

        要使用ODBC驱动,首先要在系统中进行注册,因此ODBC驱动管理器必须能找到它。

        # 1.6.1.在Windows上安装

        在32位的Windows上需要使用32位版本的驱动,而在64位的Windows上可以使用64位和32位版本的驱动,也可以在64位的Windows上同时安装32位和64位版本的驱动,这样32位和64位的应用都可以使用驱动。

        使用安装器进行安装

        注意

        首先要安装微软的Microsoft Visual C++ 2010 Redistributable 32位或者64位包。

        这是最简单的方式,也是建议的方式,只需要启动指定版本的安装器即可:

        • 32位:%IGNITE_HOME%\platforms\cpp\bin\odbc\ignite-odbc-x86.msi
        • 64位:%IGNITE_HOME%\platforms\cpp\bin\odbc\ignite-odbc-amd64.msi

        手动安装

        要在Windows上手动安装ODBC驱动,首先要为驱动在文件系统中选择一个目录,选择一个位置后就可以把驱动放在哪并且确保所有的驱动依赖可以被解析,也就是说,它们要么位于%PATH%,要么和驱动位于同一个目录。

        之后,就需要使用%IGNITE_HOME%/platforms/cpp/odbc/install目录下的安装脚本之一,注意,要执行这些脚本,很可能需要管理员权限。

          # 1.6.2.在Linux上安装

          要在Linux上构建和安装ODBC驱动,首先需要安装ODBC驱动管理器,Ignite ODBC驱动已经和UnixODBC进行了测试。

          如果已经构建完成并且执行了make install命令,libignite-odbc.so很可能会位于/usr/local/lib,要在ODBC驱动管理器中安装ODBC驱动并且可以使用,需要按照如下的步骤进行操作:

          • 确保链接器可以定位ODBC驱动的所有依赖。可以使用ldd命令像如下这样进行检查(假定ODBC驱动位于/usr/local/lib):ldd /usr/local/lib/libignite-odbc.so,如果存在到其它库的无法解析的链接,需要将这些库文件所在的目录添加到LD_LIBRARY_PATH
          • 编辑$IGNITE_HOME/platforms/cpp/odbc/install/ignite-odbc-install.ini文件,并且确保Apache Ignite段的Driver参数指向libignite-odbc.so所在的位置;
          • 要安装Ignite的ODBC驱动,可以使用如下的命令:odbcinst -i -d -f $IGNITE_HOME/platforms/cpp/odbc/install/ignite-odbc-install.ini,要执行这条命令,很可能需要root权限。

          到现在为止,Ignite的ODBC驱动已经安装好了并且可以用了,可以像其它ODBC驱动一样,连接、使用。

          # 2.连接串和DSN

          # 2.1.连接串格式

          Ignite的ODBC驱动支持标准的连接串格式,下面是正常的语法:

          connection-string ::= empty-string[;] | attribute[;] | attribute; connection-string
          empty-string ::=
          attribute ::= attribute-keyword=attribute-value | DRIVER=[{]attribute-value[}]
          attribute-keyword ::= identifier
          attribute-value ::= character-string
          

          简单来说,连接串就是一个字符串,其中包含了用分号分割的参数。

          # 2.2.支持的参数

          Ignite的ODBC驱动可以使用一些连接串/DSN参数,所有的参数都是大小写不敏感的,因此ADDRESSAddressaddress都是有效的参数名,并且指向的是同一个参数。如果参数未指定,会使用默认值,其中的一个例外是ADDRESS属性,如果未指定,会使用SERVERPORT属性代替:

          属性关键字 描述 默认值
          ADDRESS 要连接的远程节点的地址,格式为:<host>[:<port>]。比如:localhost, example.com:12345, 127.0.0.1, 192.168.3.80:5893,如果指定了这个属性,SERVERPORT将会被忽略。
          SERVER 要连接的节点地址,如果指定了ADDRESS属性,本属性会被忽略。
          PORT 节点的OdbcProcessor监听的端口,如果指定了ADDRESS属性,本属性会被忽略。 10800
          USER SQL连接的用户名。如果服务端开启了认证,该参数为必需。 “”
          PASSWORD SQL连接的密码。如果服务端开启了认证,该参数为必需。 “”
          SCHEMA 模式名。 PUBLIC
          DSN 要连接的DSN名
          PAGE_SIZE 数据源的响应中返回的行数,默认值会适用于大多数场景,小些的值会导致获取数据变慢,大些的值会导致驱动的额外内存占用,以及获取下一页时的额外延迟。 1024
          DISTRIBUTED_JOINS 为在ODBC连接上执行的所有查询开启非并置的分布式关联特性。 false
          ENFORCE_JOIN_ORDER 强制SQL查询中表关联顺序,如果设置为true,查询优化器在关联时就不会对表进行再排序。 false
          PROTOCOL_VERSION 使用的ODBC协议版本,目前支持如下的版本:2.1.0、2.1.5、2.3.0、2.3.2和2.5.0,因为向后兼容,也可以使用协议的早期版本。 2.3.0
          REPLICATED_ONLY 配置查询只在全复制的表上执行,这是个提示,用于更高效地执行。 false
          COLLOCATED 如果SQL语句包含按主键或关联键对结果集进行分组的GROUP BY子句,可以将此参数设置为true。当Ignite执行分布式查询时,会向单个集群节点发送子查询,如果事先知道待查询的数据是在同一个节点上并置在一起的,并且是按主键或关联键分组的,那么Ignite通过在参与查询的每个节点本地分组数据来实现显著的性能和网络优化。 false
          LAZY 查询延迟执行。Ignite默认会将所有的结果集放入内存然后将其返回给客户端,对于不太大的结果集,这样会提供较好的性能,并且使内部的数据库锁时间最小化,因此提高了并发能力。但是,如果相对于可用内存来说结果集过大,那么会导致频繁的GC暂停,甚至OutOfMemoryError,如果使用这个标志,可以提示Ignite延迟加载结果集,这样可以在不大幅降低性能的前提下,最大限度地减少内存的消耗。 false
          SKIP_REDUCER_ON_UPDATE 开启服务端的更新特性。当Ignite执行DML操作时,首先,它会获取所有受影响的中间行给查询发起方进行分析(通常被称为汇总),然后会准备一个更新值的批量发给远程节点。这个方式可能影响性能,如果一个DML操作会移动大量数据条目时,还可能会造成网络堵塞。使用这个标志可以提示Ignite在对应的远程节点上进行中间行的分析和更新。默认值为false,这意味着会首先获取中间行然后发给查询发起方。 false
          SSL_MODE 确定服务端是否需要SSL连接。可以根据需要使用require或者disable
          SSL_KEY_FILE 指定包含服务端SSL私钥的文件名。
          SSL_CERT_FILE 指定包含SSL服务器证书的文件名。
          SSL_CA_FILE 指定包含SSL服务器证书颁发机构(CA)的文件名。

          # 2.3.连接串示例

          下面的串,可以用于SQLDriverConnectODBC调用,来建立与Ignite节点的连接。

            # 2.4.配置DSN

            如果要使用DSN(数据源名)来进行连接,可以使用同样的参数。

            要在Windows上配置DSN,需要使用一个叫做odbcad32(32位x86系统)/odbcad64(64位)的系统工具,这是一个ODBC数据源管理器。

            安装DSN工具时,如果使用的是预构建的msi文件,一定要先安装Microsoft Visual C++ 2010(32位x86,或者64位x64)。

            要启动这个工具,打开Control panel->Administrative Tools->数据源(ODBC),当ODBC数据源管理器启动后,选择Add...->Apache Ignite,然后以正确的方式配置DSN。

            在Linux上配置DSN,需要找到odbc.ini文件,这个文件的位置各个发行版有所不同,依赖于发行版使用的特定驱动管理器,比如,如果使用unixODBC,那么可以执行如下的命令来输出系统级的ODBC相关信息:

            odbcinst -j
            

            文件的路径会显示在SYSTEM DATA SOURCESUSER DATA SOURCES属性之间。 找到odbc.ini文件之后,可以用喜欢的任意编辑器打开它,然后像下面这样添加DSN片段:

            [DSN Name]
            description=<Insert your description here>
            driver=Apache Ignite
            <Other arguments here...>
            

            # 3.查询和修改数据

            像数据库一样访问Ignite。

            # 3.1.概述

            本章会详细描述如何接入Ignite集群,如何使用ODBC驱动执行各种SQL查询。

            在实现层,Ignite的ODBC驱动使用SQL字段查询来获取Ignite缓存中的数据,这意味着通过ODBC只可以访问这些集群配置中定义的字段。

            另外,从Ignite的1.8.0版本开始,ODBC驱动支持DML,这意味着通过ODBC连接不仅仅可以访问数据,还可以修改网格中的数据。

            提示

            这里是完整的ODBC示例

            # 3.2.配置Ignite集群

            第一步,需要对集群节点进行配置,这个配置需要包含缓存的配置以及定义了QueryEntities的属性。如果应用(当前场景是ODBC驱动)要通过SQL语句进行数据的查询和修改,QueryEntities是必须的,或者,也可以使用DDL创建表。

              从上述配置中可以看出,定义了两个缓存,包含了PersonOrganization类型的数据,它们都列出了使用SQL可以读写的特定字段和索引。

              OdbcConfiguration

              确保在配置中显式地配置了OdbcConfiguration,具体参见集群配置

              # 3.3.接入集群

              配置好然后启动集群,就可以从ODBC驱动端接入了。如何做呢?准备一个有效的连接串然后连接时将其作为一个参数传递给ODBC驱动就可以了。

              另外,也可以像下面这样使用一个预定义的DSN来接入。

              SQLHENV env;
              
              // Allocate an environment handle
              SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &env);
              
              // Use ODBC ver 3
              SQLSetEnvAttr(env, SQL_ATTR_ODBC_VERSION, reinterpret_cast<void*>(SQL_OV_ODBC3), 0);
              
              SQLHDBC dbc;
              
              // Allocate a connection handle
              SQLAllocHandle(SQL_HANDLE_DBC, env, &dbc);
              
              // Prepare the connection string
              SQLCHAR connectStr[] = "DSN=My Ignite DSN";
              
              // Connecting to Ignite Cluster.
              SQLRETURN ret = SQLDriverConnect(dbc, NULL, connectStr, SQL_NTS, NULL, 0, NULL, SQL_DRIVER_COMPLETE);
              
              if (!SQL_SUCCEEDED(ret))
              {
                SQLCHAR sqlstate[7] = { 0 };
                SQLINTEGER nativeCode;
              
                SQLCHAR errMsg[BUFFER_SIZE] = { 0 };
                SQLSMALLINT errMsgLen = static_cast<SQLSMALLINT>(sizeof(errMsg));
              
                SQLGetDiagRec(SQL_HANDLE_DBC, dbc, 1, sqlstate, &nativeCode, errMsg, errMsgLen, &errMsgLen);
              
                std::cerr << "Failed to connect to Apache Ignite: "
                          << reinterpret_cast<char*>(sqlstate) << ": "
                          << reinterpret_cast<char*>(errMsg) << ", "
                          << "Native error code: " << nativeCode
                          << std::endl;
              
                // Releasing allocated handles.
                SQLFreeHandle(SQL_HANDLE_DBC, dbc);
                SQLFreeHandle(SQL_HANDLE_ENV, env);
              
                return;
              }
              

              # 3.4.查询数据

              都准备好后,就可以使用ODBC API执行SQL查询了。

              SQLHSTMT stmt;
              
              // Allocate a statement handle
              SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
              SQLCHAR query[] = "SELECT firstName, lastName, salary, Organization.name FROM Person "
                "INNER JOIN \"Organization\".Organization ON Person.orgId = Organization.id";
              SQLSMALLINT queryLen = static_cast<SQLSMALLINT>(sizeof(queryLen));
              
              SQLRETURN ret = SQLExecDirect(stmt, query, queryLen);
              
              if (!SQL_SUCCEEDED(ret))
              {
                SQLCHAR sqlstate[7] = { 0 };
                SQLINTEGER nativeCode;
              
                SQLCHAR errMsg[BUFFER_SIZE] = { 0 };
                SQLSMALLINT errMsgLen = static_cast<SQLSMALLINT>(sizeof(errMsg));
              
                SQLGetDiagRec(SQL_HANDLE_DBC, dbc, 1, sqlstate, &nativeCode, errMsg, errMsgLen, &errMsgLen);
              
                std::cerr << "Failed to perfrom SQL query upon Apache Ignite: "
                          << reinterpret_cast<char*>(sqlstate) << ": "
                          << reinterpret_cast<char*>(errMsg) << ", "
                          << "Native error code: " << nativeCode
                          << std::endl;
              }
              else
              {
                // Printing the result set.
                struct OdbcStringBuffer
                {
                  SQLCHAR buffer[BUFFER_SIZE];
                  SQLLEN resLen;
                };
              
                // Getting a number of columns in the result set.
                SQLSMALLINT columnsCnt = 0;
                SQLNumResultCols(stmt, &columnsCnt);
              
                // Allocating buffers for columns.
                std::vector<OdbcStringBuffer> columns(columnsCnt);
              
                // Binding colums. For simplicity we are going to use only
                // string buffers here.
                for (SQLSMALLINT i = 0; i < columnsCnt; ++i)
                  SQLBindCol(stmt, i + 1, SQL_C_CHAR, columns[i].buffer, BUFFER_SIZE, &columns[i].resLen);
              
                // Fetching and printing data in a loop.
                ret = SQLFetch(stmt);
                while (SQL_SUCCEEDED(ret))
                {
                  for (size_t i = 0; i < columns.size(); ++i)
                    std::cout << std::setw(16) << std::left << columns[i].buffer << " ";
              
                  std::cout << std::endl;
              
                  ret = SQLFetch(stmt);
                }
              }
              
              // Releasing statement handle.
              SQLFreeHandle(SQL_HANDLE_STMT, stmt);
              

              列绑定

              在上例中,所有的列都绑定到SQL_C_CHAR,这意味着获取时所有的值都会被转换成字符串,这样做是为了简化,获取时进行值转换是非常慢的,因此默认的做法应该是与存储采用同样的方式进行获取。

              # 3.5.插入数据

              要将新的数据插入集群,ODBC端可以使用INSERT语句。

              SQLHSTMT stmt;
              
              // Allocate a statement handle
              SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
              SQLCHAR query[] =
              	"INSERT INTO Person (id, orgId, firstName, lastName, resume, salary) "
              	"VALUES (?, ?, ?, ?, ?, ?)";
              
              SQLPrepare(stmt, query, static_cast<SQLSMALLINT>(sizeof(query)));
              
              // Binding columns.
              int64_t key = 0;
              int64_t orgId = 0;
              char name[1024] = { 0 };
              SQLLEN nameLen = SQL_NTS;
              double salary = 0.0;
              
              SQLBindParameter(stmt, 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_BIGINT, 0, 0, &key, 0, 0);
              SQLBindParameter(stmt, 2, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_BIGINT, 0, 0, &orgId, 0, 0);
              SQLBindParameter(stmt, 3, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_VARCHAR,	sizeof(name), sizeof(name), name, 0, &nameLen);
              SQLBindParameter(stmt, 4, SQL_PARAM_INPUT, SQL_C_DOUBLE, SQL_DOUBLE, 0, 0, &salary, 0, 0);
              
              // Filling cache.
              key = 1;
              orgId = 1;
              strncpy(name, "John", sizeof(name));
              salary = 2200.0;
              
              SQLExecute(stmt);
              SQLMoreResults(stmt);
              
              ++key;
              orgId = 1;
              strncpy(name, "Jane", sizeof(name));
              salary = 1300.0;
              
              SQLExecute(stmt);
              SQLMoreResults(stmt);
              
              ++key;
              orgId = 2;
              strncpy(name, "Richard", sizeof(name));
              salary = 900.0;
              
              SQLExecute(stmt);
              SQLMoreResults(stmt);
              
              ++key;
              orgId = 2;
              strncpy(name, "Mary", sizeof(name));
              salary = 2400.0;
              
              SQLExecute(stmt);
              
              // Releasing statement handle.
              SQLFreeHandle(SQL_HANDLE_STMT, stmt);
              

              下面,是不使用预编译语句插入Organization数据:

              SQLHSTMT stmt;
              
              // Allocate a statement handle
              SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
              SQLCHAR query1[] = "INSERT INTO \"Organization\".Organization (id, name)
                  VALUES (1L, 'Some company')";
              
              SQLExecDirect(stmt, query1, static_cast<SQLSMALLINT>(sizeof(query1)));
              
              SQLFreeStmt(stmt, SQL_CLOSE);
              
              SQLCHAR query2[] = "INSERT INTO \"Organization\".Organization (id, name)
                  VALUES (2L, 'Some other company')";
              
                SQLExecDirect(stmt, query2, static_cast<SQLSMALLINT>(sizeof(query2)));
              
              // Releasing statement handle.
              SQLFreeHandle(SQL_HANDLE_STMT, stmt);
              

              错误检查

              为了简化,上面的代码没有进行错误检查,但是在生产环境中不要这样做。

              # 3.6.更新数据

              下面使用UPDATE语句更新存储在集群中的部分人员的工资信息:

              void AdjustSalary(SQLHDBC dbc, int64_t key, double salary)
              {
                SQLHSTMT stmt;
              
                // Allocate a statement handle
                SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
                SQLCHAR query[] = "UPDATE Person SET salary=? WHERE id=?";
              
                SQLBindParameter(stmt, 1, SQL_PARAM_INPUT,
                    SQL_C_DOUBLE, SQL_DOUBLE, 0, 0, &salary, 0, 0);
              
                SQLBindParameter(stmt, 2, SQL_PARAM_INPUT, SQL_C_SLONG,
                    SQL_BIGINT, 0, 0, &key, 0, 0);
              
                SQLExecDirect(stmt, query, static_cast<SQLSMALLINT>(sizeof(query)));
              
                // Releasing statement handle.
                SQLFreeHandle(SQL_HANDLE_STMT, stmt);
              }
              
              ...
              AdjustSalary(dbc, 3, 1200.0);
              AdjustSalary(dbc, 1, 2500.0);
              

              # 3.7.删除数据

              最后,使用DELETE语句删除部分记录:

              void DeletePerson(SQLHDBC dbc, int64_t key)
              {
                SQLHSTMT stmt;
              
                // Allocate a statement handle
                SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
                SQLCHAR query[] = "DELETE FROM Person WHERE id=?";
              
                SQLBindParameter(stmt, 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_BIGINT,
                    0, 0, &key, 0, 0);
              
                SQLExecDirect(stmt, query, static_cast<SQLSMALLINT>(sizeof(query)));
              
                // Releasing statement handle.
                SQLFreeHandle(SQL_HANDLE_STMT, stmt);
              }
              
              ...
              DeletePerson(dbc, 1);
              DeletePerson(dbc, 4);
              

              # 3.8.通过参数数组进行批处理

              Ignite的ODBC驱动支持在DML语句中通过参数数组进行批处理。

              还是使用上述插入数据的示例,但是只调用一次SQLExecute:

              SQLHSTMT stmt;
              
              // Allocating a statement handle.
              SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
              SQLCHAR query[] =
              	"INSERT INTO Person (id, orgId, firstName, lastName, resume, salary) "
              	"VALUES (?, ?, ?, ?, ?, ?)";
              
              SQLPrepare(stmt, query, static_cast<SQLSMALLINT>(sizeof(query)));
              
              // Binding columns.
              int64_t key[4] = {0};
              int64_t orgId[4] = {0};
              char name[1024 * 4] = {0};
              SQLLEN nameLen[4] = {0};
              double salary[4] = {0};
              
              SQLBindParameter(stmt, 1, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_BIGINT, 0, 0, key, 0, 0);
              SQLBindParameter(stmt, 2, SQL_PARAM_INPUT, SQL_C_SLONG, SQL_BIGINT, 0, 0, orgId, 0, 0);
              SQLBindParameter(stmt, 3, SQL_PARAM_INPUT, SQL_C_CHAR, SQL_VARCHAR,	1024, 1024, name, 0, &nameLen);
              SQLBindParameter(stmt, 4, SQL_PARAM_INPUT, SQL_C_DOUBLE, SQL_DOUBLE, 0, 0, salary, 0, 0);
              
              // Filling cache.
              key[0] = 1;
              orgId[0] = 1;
              strncpy(name, "John", 1023);
              salary[0] = 2200.0;
              nameLen[0] = SQL_NTS;
              
              key[1] = 2;
              orgId[1] = 1;
              strncpy(name + 1024, "Jane", 1023);
              salary[1] = 1300.0;
              nameLen[1] = SQL_NTS;
              
              key[2] = 3;
              orgId[2] = 2;
              strncpy(name + 1024 * 2, "Richard", 1023);
              salary[2] = 900.0;
              nameLen[2] = SQL_NTS;
              
              key[3] = 4;
              orgId[3] = 2;
              strncpy(name + 1024 * 3, "Mary", 1023);
              salary[3] = 2400.0;
              nameLen[3] = SQL_NTS;
              
              // Asking the driver to store the total number of processed argument sets
              // in the following variable.
              SQLULEN setsProcessed = 0;
              SQLSetStmtAttr(stmt, SQL_ATTR_PARAMS_PROCESSED_PTR, &setsProcessed, SQL_IS_POINTER);
              
              // Setting the size of the arguments array. This is 4 in our case.
              SQLSetStmtAttr(stmt, SQL_ATTR_PARAMSET_SIZE, reinterpret_cast<SQLPOINTER>(4), 0);
              
              // Executing the statement.
              SQLExecute(stmt);
              
              // Releasing the statement handle.
              SQLFreeHandle(SQL_HANDLE_STMT, stmt);
              

              注意

              注意这种类型的批处理目前只支持INSERT、UPDATE、 DELETE、和MERGE语句,还不支持SELECT,data-at-execution功能也不支持通过参数数组进行批处理。

              # 3.9.流处理

              Ignite的ODBC驱动可以通过SET STREAMING命令对流化数据进行批量处理,具体可以看SET STREAMING的相关内容。

              注意

              流处理模式中,参数数组和data-at-execution参数是不支持的。

              # 4.规范

              # 4.1.概述

              ODBC定义了若干接口一致性级别,在本章中可以知道Ignite的ODBC驱动支持了哪些特性。

              # 4.2.核心接口一致性

              特性 支持程度 备注
              通过调用SQLAllocHandleSQLFreeHandle来分配和释放所有处理器类型
              使用SQLFreeStmt函数的所有形式
              通过调用SQLBindCol,绑定列结果集
              通过调用SQLBindParameterSQLNumParams,处理动态参数,包括参数数组,只针对输入方向,
              指定绑定偏移量
              使用数据执行对话框,涉及SQLParamDataSQLPutData的调用
              管理游标和游标名 部分 实现了SQLCloseCursor,Ignite不支持命名游标
              通过调用SQLColAttributeSQLDescribeColSQLNumResultColsSQLRowCount,访问结果集的描述(元数据)
              通过调用目录函数SQLColumnsSQLGetTypeInfoSQLStatisticsSQLStatistics查询数据字典 部分 不支持SQLStatistics
              通过调用SQLConnectSQLDataSourcesSQLDisconnectSQLDriverConnect管理数据源和连接,通过SQLDrivers获取驱动的信息,不管支持ODBC那个级别。
              通过调用SQLExecDirectSQLExecuteSQLPrepare预编译和执行SQL语句。
              通过调用SQLFetch,或者将FetchOrientation参数设置为SQL_FETCH_NEXT之后调用SQLFetchScroll,获取一个结果集或者多行数据中的一行,只能向前
              通过调用SQLGetData,获得一个未绑定的列
              通过调用SQLGetConnectAttrSQLGetEnvAttrSQLGetStmtAttr,获取所有属性的当前值,或者通过调用SQLSetConnectAttrSQLSetEnvAttrSQLSetStmtAttr,将所有属性赋为默认值,以及为特定属性赋为非默认值。 部分 并不支持所有属性
              通过调用SQLCopyDescSQLGetDescFieldSQLGetDescRecSQLSetDescFieldSQLSetDescRec,操作描述符的特定字段。
              通过调用SQLGetDiagFieldSQLGetDiagRec,获得诊断信息。
              通过调用SQLGetFunctionsSQLGetInfo,检测驱动兼容性,以及通过调用SQLNativeSql,在发送到数据源之前检测SQL语句中的任何文本代换的结果
              使用SQLEndTran的语法提交一个事务,驱动的核心级别不需要支持真事务,因此,应用无法指定SQL_ROLLBACK或者为SQL_ATTR_AUTOCOMMIT连接属性指定SQL_AUTOCOMMIT_OFF
              调用SQLCancel取消数据执行对话框,以及多线程环境中,在另一个线程中取消ODBC函数的执行,核心级别的接口一致性不需要支持函数的异步执行,也不需要使用SQLCancel取消一个ODBC函数的异步执行。平台和ODBC驱动都不需要多线程地同时自主活动,不过在多线程环境中,ODBC驱动必须是线程安全的,从应用来的请求的序列化是实现这个规范的一致的方式,即使它导致了一系列的性能问题。 当前的ODBC驱动实现不支持异步执行
              通过调用SQLSpecialColumns获得表的行标识符SQL_BEST_ROWID 部分 当前的实现总是返回空

              # 4.3.Level1接口一致性

              特性 支持程度 备注
              指定数据库表和视图的模式(使用两部分命名)。
              ODBC函数调用的真正异步执行,在给定的连接上,适用的函数要么是全同步的,要么是全异步的。
              使用可滚动的游标,调用SQLFetchScroll时使用FetchOrientation参数而不是SQL_FETCH_NEXT,可以在方法内访问结果集而不是只能向前。
              通过调用SQLPrimaryKeys获得表的主键。 部分 目前返回空结果集。
              使用存储过程,通过调用SQLProcedureColumnsSQLProcedures,使用ODBC的转义序列进行存储过程数据字典的查询以及存储过程的调用。
              通过调用SQLBrowseConnect,通过交互式浏览可用的服务器接入一个数据源。
              使用ODBC函数而不是SQL语句来执行特定的数据库操作:带有SQL_POSITIONSQL_REFRESHSQLSetPos
              通过调用SQLMoreResults,访问由批处理和存储过程生成的多结果集的内容。
              划定跨越多个ODBC函数的事务边界,获得真正的原子性以及在SQLEndTran中指定SQL_ROLLBACK的能力。 Ignite SQL不支持事务

              # 4.4.Level2接口一致性

              特性 支持程度 备注
              使用三部分命名的数据库表和视图。 Ignite SQL不支持catalog。
              通过调用SQLDescribeParam描述动态参数。
              不仅仅使用输入参数,还使用输出参数以及输入/输出参数,还有存储过程的结果。 Ignite SQL不支持输出参数。
              使用书签,通过在第0列上调用SQLDescribeColSQLColAttribute获得书签;通过调用SQLFetchScroll时将参数FetchOrientation配置为SQL_FETCH_BOOKMARK,在书签上进行获取;通过调用SQLBulkOperations时将参数配置为SQL_UPDATE_BY_BOOKMARKSQL_DELETE_BY_BOOKMARKSQL_FETCH_BY_BOOKMARK可以进行书签的更新、删除和获取操作。 Ignite SQL不支持书签。
              通过调用SQLColumnPrivilegesSQLForeignKeysSQLTablePrivileges获取数据字典的高级信息。 部分 SQLForeignKeys已经实现,但是返回空的结果集。
              通过在SQLBulkOperations中使用SQL_ADD或者在SQLSetPos中使用SQL_DELETESQL_UPDATE,使用ODBC函数而不是SQL语句执行额外的数据库操作。
              为特定的个别语句开启ODBC函数的异步执行。
              通过调用SQLSpecialColumns获得表的SQL_ROWVER列标识符。 部分 已实现,但是返回空结果集。
              SQL_ATTR_CONCURRENCY语句参数配置除了SQL_CONCUR_READ_ONLY以外的至少一个值。
              登录请求以及SQL查询的超时功能(SQL_ATTR_LOGIN_TIMEOUTSQL_ATTR_QUERY_TIMEOUT)。 部分 SQL_ATTR_QUERY_TIMEOUT支持已实现,SQL_ATTR_LOGIN_TIMEOUT还未实现。
              修改默认隔离级别的功能,在隔离级别为序列化时支持事务的功能。 Ignite SQL不支持事务。

              # 4.5.函数支持

              函数名 支持程度 一致性级别
              SQLAllocHandle Core
              SQLBindCol Core
              SQLBindParameter Core
              SQLBrowseConnect Level1
              SQLBulkOperations Level1
              SQLCancel Core
              SQLCloseCursor Core
              SQLColAttribute Core
              SQLColumnPrivileges Level2
              SQLColumns Core
              SQLConnect Core
              SQLCopyDesc Core
              SQLDataSources N/A Core
              SQLDescribeCol Core
              SQLDescribeParam Level2
              SQLDisconnect Core
              SQLDriverConnect Core
              SQLDrivers N/A Core
              SQLEndTran 部分 Core
              SQLExecDirect Core
              SQLExecute Core
              SQLFetch Core
              SQLFetchScroll Core
              SQLForeignKeys 部分 Level2
              SQLFreeHandle Core
              SQLFreeStmt Core
              SQLGetConnectAttr 部分 Core
              SQLGetCursorName Core
              SQLGetData Core
              SQLGetDescField Core
              SQLGetDescRec Core
              SQLGetDiagField Core
              SQLGetDiagRec Core
              SQLGetEnvAttr 部分 Core
              SQLGetFunctions Core
              SQLGetInfo Core
              SQLGetStmtAttr 部分 Core
              SQLGetTypeInfo Core
              SQLMoreResults Level1
              SQLNativeSql Core
              SQLNumParams Core
              SQLNumResultCols Core
              SQLParamData Core
              SQLPrepare Core
              SQLPrimaryKeys 部分 Level1
              SQLProcedureColumns Level1
              SQLProcedures Level1
              SQLPutData Core
              SQLRowCount Core
              SQLSetConnectAttr 部分 Core
              SQLSetCursorName Core
              SQLSetDescField Core
              SQLSetDescRec Core
              SQLSetEnvAttr 部分 Core
              SQLSetPos Level1
              SQLSetStmtAttr 部分 Core
              SQLSpecialColumns 部分 Core
              SQLStatistics Core
              SQLTablePrivileges Level2
              SQLTables Core

              # 4.6.环境属性一致性

              特性 支持程度 一致性级别
              SQL_ATTR_CONNECTION_POOLING 可选
              SQL_ATTR_CP_MATCH 可选
              SQL_ATTR_ODBC_VER Core
              SQL_ATTR_OUTPUT_NTS 可选

              # 4.7.连接属性一致性

              特性 支持程度 一致性级别
              SQL_ATTR_ACCESS_MODE Core
              SQL_ATTR_ASYNC_ENABLE Level1/Level2
              SQL_ATTR_AUTO_IPD Level2
              SQL_ATTR_AUTOCOMMIT Level1
              SQL_ATTR_CONNECTION_DEAD Level1
              SQL_ATTR_CONNECTION_TIMEOUT Level2
              SQL_ATTR_CURRENT_CATALOG Level2
              SQL_ATTR_LOGIN_TIMEOUT Level2
              SQL_ATTR_ODBC_CURSORS Core
              SQL_ATTR_PACKET_SIZE Level2
              SQL_ATTR_QUIET_MODE Core
              SQL否_ATTR_TRACE Core
              SQL_AT否TR_TRACEFILE Core
              SQL_AT否TR_TRANSLATE_LIB Core
              SQL_ATTR_TRANSLATE_OPTION Core
              SQL_ATTR_TXN_ISOLATION Level1/Level2

              # 4.8.语句属性一致性

              特性 支持程度 一致性级别
              SQL_ATTR_APP_PARAM_DESC 部分 Core
              SQL_ATTR_APP_ROW_DESC 部分 Core
              SQL_ATTR_ASYNC_ENABLE Level1/Level2
              SQL_ATTR_CONCURRENCY Level1/Level2
              SQL_ATTR_CURSOR_SCROLLABLE Level1
              SQL_ATTR_CURSOR_SENSITIVITY Level2
              SQL_ATTR_CURSOR_TYPE Level1/Level2
              SQL_ATTR_ENABLE_AUTO_IPD Level2
              SQL_ATTR_FETCH_BOOKMARK_PTR Level2
              SQL_ATTR_IMP_PARAM_DESC 部分 Core
              SQL_ATTR_IMP_ROW_DESC 部分 Core
              SQL_ATTR_KEYSET_SIZE Level2
              SQL_ATTR_MAX_LENGTH Level1
              SQL_ATTR_MAX_ROWS Level1
              SQL_ATTR_METADATA_ID Core
              SQL_ATTR_NOSCAN Core
              SQL_ATTR_PARAM_BIND_OFFSET_PTR Core
              SQL_ATTR_PARAM_BIND_TYPE Core
              SQL_ATTR_PARAM_OPERATION_PTR Core
              SQL_ATTR_PARAM_STATUS_PTR Core
              SQL_ATTR_PARAMS_PROCESSED_PTR Core
              SQL_ATTR_PARAMSET_SIZE Core
              SQL_ATTR_QUERY_TIMEOUT Level2
              SQL_ATTR_RETRIEVE_DATA Level1
              SQL_ATTR_ROW_ARRAY_SIZE Core
              SQL_ATTR_ROW_BIND_OFFSET_PTR Core
              SQL_ATTR_ROW_BIND_TYPE Core
              SQL_ATTR_ROW_NUMBER Level1
              SQL_ATTR_ROW_OPERATION_PTR Level1
              SQL_ATTR_ROW_STATUS_PTR Core
              SQL_ATTR_ROWS_FETCHED_PTR Core
              SQL_ATTR_SIMULATE_CURSOR Level2
              SQL_ATTR_USE_BOOKMARKS Level2

              # 4.9.描述符头字段一致性

              特性 支持程度 一致性级别
              SQL_DESC_ALLOC_TYPE Core
              SQL_DESC_ARRAY_SIZE Core
              SQL_DESC_ARRAY_STATUS_PTR Core/Level1
              SQL_DESC_BIND_OFFSET_PTR Core
              SQL_DESC_BIND_TYPE Core
              SQL_DESC_COUNT Core
              SQL_DESC_ROWS_PROCESSED_PTR Core

              # 4.10.描述符记录字段一致性

              特性 支持程度 一致性级别
              SQL_DESC_AUTO_UNIQUE_VALUE Level2
              SQL_DESC_BASE_COLUMN_NAME Core
              SQL_DESC_BASE_TABLE_NAME Level1
              SQL_DESC_CASE_SENSITIVE Core
              SQL_DESC_CATALOG_NAME Level2
              SQL_DESC_CONCISE_TYPE Core
              SQL_DESC_DATA_PTR Core
              SQL_DESC_DATETIME_INTERVAL_CODE Core
              SQL_DESC_DATETIME_INTERVAL_PRECISION Core
              SQL_DESC_DISPLAY_SIZE Core
              SQL_DESC_FIXED_PREC_SCALE Core
              SQL_DESC_INDICATOR_PTR Core
              SQL_DESC_LABEL Level2
              SQL_DESC_LENGTH Core
              SQL_DESC_LITERAL_PREFIX Core
              SQL_DESC_LITERAL_SUFFIX Core
              SQL_DESC_LOCAL_TYPE_NAME Core
              SQL_DESC_NAME Core
              SQL_DESC_NULLABLE Core
              SQL_DESC_OCTET_LENGTH Core
              SQL_DESC_OCTET_LENGTH_PTR Core
              SQL_DESC_PARAMETER_TYPE Core/Level2
              SQL_DESC_PRECISION Core
              SQL_DESC_ROWVER Level1
              SQL_DESC_SCALE Core
              SQL_DESC_SCHEMA_NAME Level1
              SQL_DESC_SEARCHABLE Core
              SQL_DESC_TABLE_NAME Level1
              SQL_DESC_TYPE Core
              SQL_DESC_TYPE_NAME Core
              SQL_DESC_UNNAMED Core
              SQL_DESC_UNSIGNED Core
              SQL_DESC_UPDATABLE Core

              # 4.11.SQL数据类型

              下面是支持的SQL数据类型:

              数据类型 是否支持
              SQL_CHAR
              SQL_VARCHAR
              SQL_LONGVARCHAR
              SQL_WCHAR
              SQL_WVARCHAR
              SQL_WLONGVARCHAR
              SQL_DECIMAL
              SQL_NUMERIC
              SQL_SMALLINT
              SQL_INTEGER
              SQL_REAL
              SQL_FLOAT
              SQL_DOUBLE
              SQL_BIT
              SQL_TINYINT
              SQL_BIGINT
              SQL_BINARY
              SQL_VARBINARY
              SQL_LONGVARBINARY
              SQL_TYPE_DATE
              SQL_TYPE_TIME
              SQL_TYPE_TIMESTAMP
              SQL_TYPE_UTCDATETIME
              SQL_TYPE_UTCTIME
              SQL_INTERVAL_MONTH
              SQL_INTERVAL_YEAR
              SQL_INTERVAL_YEAR_TO_MONTH
              SQL_INTERVAL_DAY
              SQL_INTERVAL_HOUR
              SQL_INTERVAL_MINUTE
              SQL_INTERVAL_SECOND
              SQL_INTERVAL_DAY_TO_HOUR
              SQL_INTERVAL_DAY_TO_MINUTE
              SQL_INTERVAL_DAY_TO_SECOND
              SQL_INTERVAL_HOUR_TO_MINUTE
              SQL_INTERVAL_HOUR_TO_SECOND
              SQL_INTERVAL_MINUTE_TO_SECOND
              SQL_GUID

              # 4.12.C数据类型

              下面是支持的C数据类型:

              数据类型 是否支持
              SQL_C_CHAR
              SQL_C_WCHAR
              SQL_C_SHORT
              SQL_C_SSHORT
              SQL_C_USHORT
              SQL_C_LONG
              SQL_C_SLONG
              SQL_C_ULONG
              SQL_C_FLOAT
              SQL_C_DOUBLE
              SQL_C_BIT
              SQL_C_TINYINT
              SQL_C_STINYINT
              SQL_C_UTINYINT
              SQL_C_BIGINT
              SQL_C_SBIGINT
              SQL_C_UBIGINT
              SQL_C_BINARY
              SQL_C_BOOKMARK
              SQL_C_VARBOOKMARK
              SQL_C_INTERVAL* (all interval types)
              SQL_C_TYPE_DATE
              SQL_C_TYPE_TIME
              SQL_C_TYPE_TIMESTAMP
              SQL_C_NUMERIC
              SQL_C_GUID

              # 5.数据类型

              支持如下的SQL数据类型(规范中列出):

              • SQL_CHAR
              • SQL_VARCHAR
              • SQL_LONGVARCHAR
              • SQL_SMALLINT
              • SQL_INTEGER
              • SQL_FLOAT
              • SQL_DOUBLE
              • SQL_BIT
              • SQL_TINYINT
              • SQL_BIGINT
              • SQL_BINARY
              • SQL_VARBINARY
              • SQL_LONGVARBINARY
              • SQL_GUID
              • SQL_DECIMAL
              • SQL_TYPE_DATE
              • SQL_TYPE_TIMESTAMP
              • SQL_TYPE_TIME

              # 6.错误码

              要获取错误码, 可以使用SQLGetDiagRec()函数,它会返回一个ANSI SQL标准定义的错误码字符串,比如:

              SQLHENV env;
              SQLAllocHandle(SQL_HANDLE_ENV, SQL_NULL_HANDLE, &env);
              
              SQLSetEnvAttr(env, SQL_ATTR_ODBC_VERSION, reinterpret_cast<void*>(SQL_OV_ODBC3), 0);
              
              SQLHDBC dbc;
              SQLAllocHandle(SQL_HANDLE_DBC, env, &dbc);
              
              SQLCHAR connectStr[] = "DRIVER={Apache Ignite};SERVER=localhost;PORT=10800;SCHEMA=Person;";
              SQLDriverConnect(dbc, NULL, connectStr, SQL_NTS, 0, 0, 0, SQL_DRIVER_COMPLETE);
              
              SQLAllocHandle(SQL_HANDLE_STMT, dbc, &stmt);
              
              SQLCHAR query[] = "SELECT firstName, lastName, resume, salary FROM Person";
              SQLRETURN ret = SQLExecDirect(stmt, query, SQL_NTS);
              
              if (ret != SQL_SUCCESS)
              {
              	SQLCHAR sqlstate[7] = "";
              	SQLINTEGER nativeCode;
              
              	SQLCHAR message[1024];
              	SQLSMALLINT reallen = 0;
              
              	int i = 1;
              	ret = SQLGetDiagRec(SQL_HANDLE_STMT, stmt, i, sqlstate,
                                    &nativeCode, message, sizeof(message), &reallen);
              
              	while (ret != SQL_NO_DATA)
              	{
              		std::cout << sqlstate << ": " << message;
              
              		++i;
              		ret = SQLGetDiagRec(SQL_HANDLE_STMT, stmt, i, sqlstate,
                                      &nativeCode, message, sizeof(message), &reallen);
              	}
              }
              

              下表中列出了所有Ignite目前支持的错误码,该列表未来可能会扩展:

              错误码 描述
              01S00 无效连接串属性
              01S02 驱动程序不支持指定的值,并替换了一个类似的值
              08001 驱动接入集群失败
              08002 连接已经建立
              08003 未知原因导致的连接处于关闭状态
              08004 连接被集群踢出
              08S01 连接失败
              22026 字符串长度与数据执行对话框不匹配
              23000 违反完整性约束(比如主键重复、主键为空等等)
              24000 无效的游标状态
              42000 请求的语法错误
              42S01 表已经存在
              42S02 表不存在
              42S11 索引已经存在
              42S12 索引不存在
              42S21 列已经存在
              42S22 列不存在
              HY000 一般性错误,具体看错误消息
              HY001 内存分配错误
              HY003 无效的应用缓冲区类型
              HY004 无效的SQL数据类型
              HY009 无效的空指针使用
              HY010 函数调用顺序错误
              HY090 无效的字符串和缓冲区长度(比如长度为负或者为0)
              HY092 可选类型超范围
              HY097 列类型超范围
              HY105 无效的参数类型
              HY106 获取类型超范围
              HYC00 特性未实现
              IM001 函数不支持

              18624049226

              最后更新时间:: 10/21/2020, 4:44:25 PM