前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
社区首页 >专栏 >OLEDB 调用存储过程

OLEDB 调用存储过程

作者头像
Masimaro
发布于 2018-08-31 08:08:34
发布于 2018-08-31 08:08:34
1.8K00
代码可运行
举报
运行总次数:0
代码可运行

除了常规调用sql语句和进行简单的插入删除操作外,OLEDB还提供了调用存储过程的功能,存储过程就好像是用SQL语句写成的一个函数,可以有参数,有返回值。 存储过程除了像普通函数那样返回一般的值以外,还可以返回结果集,对于返回的内容可以使用输出参数的方式获取,但是如果返回的是结果集,一般不推荐使用输出参数来获取,一般采用的是使用多结果集来接收。另外对于输入参数一般采用参数化查询的方式进行,因此它的使用与参数化查询类似,但是相比于参数化查询来说要复杂一些。

存储过程的使用

对于输出参数,在绑定DBBINDING 结构的时候,将结构的eParamIO指定为DBPARAMIO_OUTPUT,调用存储过程可以使用类似下面的格式

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
{? = call myProc(?, ?)}

这个样式中两个大括号是必须的,其中?代表的输入输出参数,call表示调用存储过程,也是必须的。 一般来说,存储过程的参数位置只接受输入,不作为输出参数,而存储过程的返回值位置只作为输出,不作为输入。另外最需要注意的一点是:当存储过程返回结果集的时候,返回的结果集指针如果没有被释放的话,输出参数的缓冲是不会被刷新的,也就是接收不到输出参数。这是由于数据提供者在返回这些数据的时候是按照流的方式。而结果集的流在输出参数和返回值的流之前,所以在结果集未被释放之前,应用程序是接收不到输出参数的。 针对他的这个特性,我们一般是先使用存储过程返回的结果集,然后释放结果集的相关指针,接着从输出参数的缓冲中取出数据,最后释放这些缓冲。

存储过程使用的例子

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
BOOL ExecUsp(IOpenRowset *pIOpenRowset)
{
    IAccessor *pParamAccessor = NULL;

    LPOLESTR pSQL = OLESTR("{? = call Usp_InputOutput(?,?)}");
    DB_UPARAMS cParams = 0;
    DBPARAMINFO* rgParamInfo = NULL;
    DBBINDING* rgParamBind = NULL;
    LPOLESTR pParamNames = NULL;
    DWORD dwOffset = 0;
    HACCESSOR hAccessor = NULL;
    TCHAR szInputBuffer[11] = _T("");
    DBPARAMS dbParams = {0};
    BOOL bRet = FALSE;

    HRESULT hRes = pIOpenRowset->QueryInterface(IID_IDBCreateCommand, (void**)&pIDBCreateCommand);
    hRes = pIDBCreateCommand->CreateCommand(NULL, IID_ICommandText, (IUnknown**)&pICommandText);

    hRes = pICommandText->SetCommandText(DBGUID_DEFAULT, pSQL);
    hRes = pICommandText->QueryInterface(IID_ICommandPrepare, (void**)&pICommandPrepare);
    hRes = pICommandPrepare->Prepare(0);

    hRes = pICommandText->QueryInterface(IID_ICommandWithParameters, (void**)&pICommandWithParameters);
    hRes = pICommandWithParameters->GetParameterInfo(&cParams, &rgParamInfo, &pParamNames);

    //参数绑定
    rgParamBind = (DBBINDING*)MALLOC(sizeof(DBBINDING) * cParams);
    for (int i = 0; i < cParams; i++)
    {
        rgParamBind[i].bPrecision = 11;
        rgParamBind[i].bScale = rgParamInfo[i].bScale;
        rgParamBind[i].cbMaxLen = rgParamInfo[i].ulParamSize; //由于都是int型,所以这里就不需要指定多余的缓冲
        rgParamBind[i].iOrdinal = rgParamInfo[i].iOrdinal;
        rgParamBind[i].dwPart = DBPART_VALUE;
    //当发现参数是输入参数的时候,在eParamIO中加一个输入参数的性质,
        if (rgParamInfo[i].dwFlags & DBPARAMFLAGS_ISINPUT)
        {
            rgParamBind[i].eParamIO |= DBPARAMIO_INPUT;
        }

    //同上
        if (rgParamInfo[i].dwFlags & DBPARAMFLAGS_ISOUTPUT)
        {
            rgParamBind[i].eParamIO |= DBPARAMIO_OUTPUT;
        }

        rgParamBind[i].dwMemOwner = DBMEMOWNER_CLIENTOWNED;
        rgParamBind[i].obLength = 0;
        rgParamBind[i].obStatus = 0;
        rgParamBind[i].obValue = dwOffset;
        dwOffset = rgParamBind[i].obValue + rgParamBind[i].cbMaxLen;
        rgParamBind[i].wType = rgParamInfo[i].wType; //其实他们都是int型
    }

    //创建参数的访问器
    hRes = pICommandText->QueryInterface(IID_IAccessor, (void**)&pParamAccessor);
    COM_SUCCESS(hRes, _T("查询接口pParamAccessor失败,错误码为:%08x\n"), hRes);
    hRes = pParamAccessor->CreateAccessor(DBACCESSOR_PARAMETERDATA, cParams, rgParamBind, dwOffset, &hAccessor, NULL);
    COM_SUCCESS(hRes, _T("创建访问器失败,错误码为:%08x\n"), hRes);

    //设置参数值
    dbParams.pData = MALLOC(dwOffset);
    ZeroMemory(dbParams.pData , dwOffset);
    dbParams.cParamSets = cParams;
    dbParams.hAccessor = hAccessor;
    for (int i = 0; i < cParams; i++)
    {
        if (rgParamBind[i].eParamIO & DBPARAMIO_INPUT)
        {
            COM_PRINT(_T("请输入第%d个参数的值[%s]:"), i + 1, rgParamInfo[i].pwszName);
            while (S_OK != StringCchGets(szInputBuffer, 11));

            *(int*)((BYTE*)dbParams.pData + rgParamBind[i].obValue) = _ttoi(szInputBuffer);
            ZeroMemory(szInputBuffer, 11 * sizeof(TCHAR));
        }
    }

    hRes = pICommandText->Execute(NULL, IID_IMultipleResults, &dbParams, NULL, (IUnknown**)&pIMultipleResults);

    //读取结果集
    ReadRowset(pIMultipleResults);
    COM_PRINT(_T("返回值为:%d\n"), *(int*)dbParams.pData)
    bRet = TRUE;

__CLEAR_UP:
    CoTaskMemFree(rgParamInfo);
    FREE(rgParamBind);
    CoTaskMemFree(pParamNames);
    FREE(dbParams.pData);

    SAFE_RELEASE(pIDBCreateCommand);
    SAFE_RELEASE(pICommandText);
    SAFE_RELEASE(pICommandPrepare);
    SAFE_RELEASE(pICommandWithParameters);
    SAFE_RELEASE(pParamAccessor);
    SAFE_RELEASE(pIMultipleResults);

    return bRet;
}

上述代码中调用的存储过程如下:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
ALTER PROCEDURE [dbo].[Usp_InputOutput]
    @input int,
    @output int Output

AS
BEGIN
    select @input, @output
    set @output = 2
    return 7
END

该存储过程输入两个参数,并通过select返回有这两个参数组成的结果集。存储过程的输出参数为7. 在上述代码中,先定义了一个调用存储过程的sql语句,接着在ICommandText对象中设置该存储过程,然后获取参数的相关信息,然后绑定参数,提供输出、输出参数的缓冲,然后执行存储过程获取结果集。接着读取结果集。最后读取返回值。 上面我们说过如果不释放返回的结果集的指针的话,是接收不到返回值的,但是在这段代码中好像在读取返回值之前没有释放返回的IMultipleResults指针的操作,但是还是可以取到结果集的,这是为什么呢?其实之前说的是返回的结果集,但是并没有说是多行结果集,这里释放的主要是结果集。针对多行结果集,我们只要释放它里面包含的所有结果集对象就可以了。释放结果集的代码在函数ReadRowset中,这里并没有列举出来。

最后:完整的代码:请点击这里查看


本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2018-03-31 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
OLEDB 枚举数据源
在之前的程序中,可以看到有这样一个功能,弹出一个对话框让用户选择需要连接的数据源,并输入用户名和密码,最后连接;而且在一些数据库管理软件中也提供这种功能——能够自己枚举出系统中存在的数据源,同时还可以枚举出能够连接的SQL Server数据库的实例。其实这个功能是OLEDB提供的高级功能之一。 枚举对象用于搜寻可用的数据源和其它的枚举对象(层次式),枚举出来的对象是一个树形结构。在程序中提供一个枚举对象就可以枚举里面的所有数据源,如果没有指定所使用的的上层枚举对象,则可以使用顶层枚举对象来枚举可用的OLEDB提供程序,其实我们使用枚举对象枚举数据源时它也是在注册表的对应位置进行搜索,所以我们可以直接利用操作注册表的方式来获取数据源对象,但是注册表中的信息过于复杂,而且系统对注册表的依赖比较严重,所以并不推荐使用这种方式。 枚举对象的原型如下:
Masimaro
2018/08/31
6950
OLEDB数据源
OLEDB虽然是基于COM的一组接口,但是它与标准的COM接口有点不同,它的一大特色在于它自身的属性设置,有的接口虽然对象中存在但是调用QueryInterface是查询不出来的,只有设置相应的接口才会打开,有的接口可以根据属性值表现不同的行为。比如设置了对应的只读属性则不允许使用更新接口。 每个属性都有值、类型、说明和读写属性,对于行集对象,还有一个用于指示是否可以逐列应用它的指示器。 属性由一个GUID和一个整数ID进行唯一标识。 属性集是所有具有相同 组GUID 的一组属性。在逻辑上它们都用于同一种功能,比如有的属性集用于设置数据源连接属性,有的用于设置行集属性等等。它们是应用在同一个特定对象上的一组属性。在每个这样的属性组中都有属性每个属性属于一个或者多个属性组。 属性定义如下:
Masimaro
2018/08/31
8010
存储过程详解
存储过程简介 什么是存储过程:存储过程可以说是一个记录集吧,它是由一些T-SQL语句组成的代码块,这些T-SQL语句代码像一个方法一样实现一些功能(对单表或多表的增删改查),然后再给这个代码块取一个名字,在用到这个功能的时候调用他就行了。 存储过程的好处: 1.由于数据库执行动作时,是先编译后执行的。然而存储过程是一个编译过的代码块,所以执行效率要比T-SQL语句高。 2.一个存储过程在程序在网络中交互时可以替代大堆的T-SQL语句,所以也能降低网络的通信量,提高通信速率。 3.通过存储过程能够使没
郑小超.
2018/01/24
2.2K0
多结果集IMultipleResult接口
在某些任务中,需要执行多条sql语句,这样一次会返回多个结果集,在应用程序就需要处理多个结果集,在OLEDB中支持多结果集的接口是IMultipleResult。
Masimaro
2018/08/31
1.1K0
MSSQL之九 存储过程与函数
批处理是一组一起提交给SQL Sever执行的SQL语句。当执行批处理时SQL Sever将批处理的语句编译到一个称为执行计划的可执行单元。这样可以节省执行时间。
用户9184480
2024/12/13
1130
MSSQL之九  存储过程与函数
DirectShow Filter的开发实践
一、介绍      摄像头图像采集处理在业界有着多种成熟的方案。从老的DirectShow、Grabber技术,到新的Windows Media Foundation框架,网络上都有着丰富的参考资料。OpenCV库里面甚至提供了非常简洁的接口,用户只要一两行代码即可实现数据采集、编解码等功能,使用起来甚是方便。但是,如果把数据采集的任务放到我们自己的程序中来实现的话,CPU的占用率会比较高。这在某些情况下不太可取。虽然可以实现,但是在客户端使用时效率非常低下。而公司恰好有一项开发任务:要求采集到摄像头数据后
24K纯开源
2018/01/18
1.9K0
DirectShow Filter的开发实践
如何定制一款12306抢票浏览器——处理预订页面和验证码自动识别功能
        我们先看一下预订页面的结构(转载请指明出于breaksoftware的csdn博客)
方亮
2019/01/16
9690
数据更新接口与延迟更新
title: 数据更新接口与延迟更新 tags: [OLEDB, 数据库编程, VC++, 数据库] date: 2018-02-12 14:29:35 categories: windows 数据库编程 keywords: OLEDB, 数据库编程, VC++, 数据库,数据库数据更新, 延迟提交 --- 在日常使用中,更新数据库数据经常使用delete 、update等SQL语句进行,但是OLEDB接口提供了额外的接口,来直接修改和更新数据库数据。
Masimaro
2018/08/31
1.6K0
OLEDB 参数化查询
一般情况下,SQL查询是相对固定的,一条语句变化的可能只是条件值,比如之前要求查询二年级学生信息,而后面需要查询三年级的信息,这样的查询一般查询的列不变,后面的条件只有值在变化,针对这种查询可以使用参数化查询的方式来提高效率,也可以时SQL操作更加安全,从根本上杜绝SQL注入的问题。
Masimaro
2018/08/31
1.4K0
OLEDB 简单数据查找定位和错误处理
在数据库查询中,我们主要使用的SQL语句,但是之前也说过,SQL语句需要经历解释执行的步骤,这样就会拖慢程序的运行速度,针对一些具体的简单查询,比如根据用户ID从用户表中查询用户具体信息,像这样的简单查询OLEDB提供了专门的查询接口。使用该接口可以很大程度上提升程序性能。 另外在之前的代码中,只是简单的通过HRESULT这个返回值来判断是否成功,针对错误没有具体的处理,但是OLEDB提供了自己的处理机制,这篇博文主要来介绍这两种情况下的处理方式
Masimaro
2018/08/31
6990
SQL语句执行与结果集的获取
title: SQL语句执行与结果集的获取 tags: [OLEDB, 数据库编程, VC++, 数据库] date: 2018-01-28 09:22:10 categories: windows 数据库编程 keywords: OLEDB, 数据库编程, VC++, 数据库,执行SQL, 获取结果集 --- 上次说到命令对象是用来执行SQL语句的。数据源在执行完SQL语句后会返回一个结果集对象,将SQL执行的结果返回到结果集对象中,应用程序在执行完SQL语句后,解析结果集对象中的结果,得到具体的结果,这次的主要内容是如何解析结果集对象并获取其中的值。
Masimaro
2018/08/31
3.9K0
事务对象和命令对象
title: 事务对象和命令对象 tags: [OLEDB, 数据库编程, VC++, 数据库] date: 2018-01-21 09:22:10 categories: windows 数据库编程 keywords: OLEDB, 数据库编程, VC++, 数据库 --- 上次说到数据源对象,这次接着说事务对象和命令对象。 事务是一种对数据源的一系列更新进行分组或批处理以便当所有更新都成功时同时提交这些更新,或者如果任何一个更新失败则不提交任何更新并且回滚整个事务的方法. 命令对象一般是用来执行sql语句并生成结果集的对象
Masimaro
2018/08/31
8620
OLEDB 数据变更通知
除了之前介绍的接口,OLEDB还定义了其他一些支持回调的接口,可以异步操作OLEDB对象或者得到一些重要的事件通知,从而使应用程序有机会进行一些必要的处理。其中较有用的就是结果集对象的变更通知接口。通过这个接口可以及时得到结果集被增删改数据变化的情况,并有机会进行必要的数据合法性审核。 数据变更通知的接口是IRowsetNotify,数据源对象要求的异步通知事件接口是IDBAsynchNotify。
Masimaro
2018/08/31
1.5K0
OLEDB 数据变更通知
OLEDB不使用SQL语句直接打开数据表
这样的sql语句。SQL语句必然伴随着数据库的解释执行,一般来说效率比较低下,而且使用SQL语句时需要数据库支持ICommandText对象,但是在OLEDB中它是一个可选接口,也就是有的数据库可能不支持,这个时候OLEDB给我们提供了一种方法让我们能够在不使用SQL的情况下操作数据库表对象。 直接打开表对象需要使用IOpenRowset接口。该接口属于Session对象。
Masimaro
2018/08/31
5990
OLEDB存取BLOB型数据
现代数据库系统除了支持一些标准的通用数据类型以外,大多数还支持一种称之为BLOB型的数据。 BLOB全称为big large object bytes, 大二进制对象类型,这种类型的数据通常用于存储文档、图片、音频等文件,这些文件一般体积较大,保存这些文件可以很方便的管理和检索这类信息。在MS SQLSERVER中常见的BLOB数据类型有text、ntext(n表示unicode)、image、nvarchar、varchar、varbinary等。其中image基本可以用来保存一切二进制文件,比如word、Excel、音频、视频等等类型。 针对BLOB型数据,OLEDB也提供了对它的支持
Masimaro
2018/08/31
2.2K0
OLEDB存取BLOB型数据
OLEDB事务
学过数据的人一般都知道事务的重要性,事务是一种对数据源的一系列更新进行分组或者批处理以便当所有更新都成功时同时提交更新,或者任意一个更新失败时进行回滚将数据库中的数据回滚到执行批处理中的所有操作之前的一种方法。使用事务保证了数据的完整性。这里不展开详细的说事务,只是谈谈OLEDB在事务上的支持
Masimaro
2018/08/31
6430
Oracle应用实战八(完结)——存储过程、函数+对象曹组
游标 在写java程序中有结果集的概念,那么在pl/sql中也会用到多条记录,这时候我们就要用到游标,游标可以存储查询返回的多条数据。 游标可以理解为是PL/SQL中的结果集,我们通过游标可以提取结果
Java帮帮
2018/03/19
1.9K0
Oracle应用实战八(完结)——存储过程、函数+对象曹组
PL/SQL 编程(二)游标、存储过程、函数
游标--数据的缓存区 游标:类似集合,可以让用户像操作数组一样操作查询出来的数据集,实质上,它提供了一种从集合性质的结果中提取单条记录的手段。 可以将游标形象的看成一个变动的光标,他实质上是一个指针,在一段Oracle存放数据查询结果集或者数据操作结果集的内存中,这个指针可以指向结果集任何一条记录。 游标分静态游标和REF游标两类,静态游标包含显式游标和隐式游标。 显式游标: 在使用之前必须有明确的游标声明和定义,这样的游标定义会关联数据查询语句,通常会返回一行或多行。打开游标后,用户可以利用游标的位置对结
二十三年蝉
2018/02/28
3.8K0
Sql Server 调用存储过程
创建存储过程: 1、在企业管理器中新建存储过程 2、在查询分析器中编辑存储过程(带有参数和返回值的) SET  QUOTED_IDENTIFIER  ON     GO   SET  ANSI_NULLS  ON     GO     ALTER      procedure  insertUserinfo  @name   varchar ( 50 ), @sex   varchar ( 50 ), @birthday   varchar ( 50 ), @returnValue   int   =   
西门呀在吹雪
2020/11/09
1.8K0
SkeyeLive中DirectShow采集音视频流程及几种采集方式介绍
前段时间SkeyeLive开放了DirectShow采集库,这个库底层采用DirectShow SDK的接口实现音视频的预览(播放)和采集;很多人可能还不太了解这个封装库的回调方式和之前的DShow线程采集方式有什么不同,或者说对DirectShow的采集流程还不太熟悉,下面我将就Windows平台下用使用DirectShow的过滤器(滤波器)进行流媒体开发的前端采集部分进行简要介绍,如果大家想深入的学习和探索,推荐大家去看看《Visual C++音频/视频处理技术及工程实践》这本书,第9章有详细的流程讲解。
Openskeye
2023/04/18
9350
相关推荐
OLEDB 枚举数据源
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
查看详情【社区公告】 技术创作特训营有奖征文