编写安全的SQL server扩展存储过程

2016-01-29 16:24 8 1 收藏

编写安全的SQL server扩展存储过程,编写安全的SQL server扩展存储过程

【 tulaoshi.com - SQLServer 】

 

SQL server 的扩展存储过程,其实就是一个普通的 windows DLL,只不过按照某种规则实现了某些函数而已。

近日在写一个扩展存储过程时,发现再写这类动态库时,还是有一些需要特别注意的地方。之所以会特别注意,是因为DLL运行于SQL server的地址空间,而SQL Server到底是怎么进行线程调度的,却不是我们能了解的,即便了解也无法控制。

我们写动态库一般是自己用,即便给别人用,也很少像SQL Server这样,一个动态库很有可能加载多次,并且都是加载到一个进程的地址空间中。我们知道,当一个动态库加载到进程的地址空间时,DLL所有全局与局部变量初始化且仅初始化一次,以后再次调用 LoadLibrary函数时,仅仅增加其引用计数而已,那么很显然,假如有一全局 int ,初始化为0,调用一个函数另其自加,此时其值为1,然后再调用LoadLibray,并利用返回的句柄调用输出函数输出该值,虽然调用者觉得自己加载后立即输出,然后该值确实1而不是0。windows是进程独立的,而在线程方面,假如不注意,上面的情况很可能会程序员带来麻烦。

介绍一下我的扩展存储过程,该动态库导出了三个函数: Init,work,Final,Init读文件,存储信息于内存,work简单的只是向该内存检索信息,Final回收内存。如上所说,假如不考虑同一进程空间多次加载问题,两次调用Init将造成无谓的浪费,因为我第一次已经读进了内存,要是通过堆分配内存,还会造成内存泄露。

我使用的引用计数解决的该问题,代码很短,直接贴上来:

#include "stdafx.h"
#include  <string

using namespace std;

extern "C" {
 RETCODE __declspec(dllexport) xp_part_init(SRV_PROC *srvproc);
 RETCODE __declspec(dllexport) xp_part_process(SRV_PROC *srvproc);
 RETCODE __declspec(dllexport) xp_part_finalize(SRV_PROC *srvproc);
}

#define XP_NOERROR      0
#define XP_ERROR        1

HINSTANCE hInst = NULL;
int nRef = 0;

void printError (SRV_PROC *pSrvProc, CHAR* szErrorMsg);

ULONG __GetXpVersion(){ return ODS_VERSION;}

SRVRETCODE xp_part_init(SRV_PROC* pSrvProc){
 typedef bool (*Func)();

 if(nRef == 0){
  hInst = ::LoadLibrary("part.dll");
  if(hInst == NULL){
   printError(pSrvProc,"不能加载part.dll");
   return XP_ERROR;
  }
  Func theFunc = (Func)::GetProcAddress(hInst,"Init");
  if(!theFunc()){
   ::FreeLibrary(hInst);
   printError(pSrvProc,"不能获得分类号与专辑的对应表");
   return XP_ERROR;
  }
 }

 ++ nRef;
 return (XP_NOERROR);
}

SRVRETCODE xp_part_process(SRV_PROC* pSrvProc){
 typedef bool (*Func)(char*);
 
 if(nRef == 0){
  printError(pSrvProc,"函数尚未初始化,请首先调用xp_part_init");
  return XP_ERROR;
 }
 Func theFunc = (Func)::GetProcAddress(hInst,"Get");

 BYTE bType;
 ULONG cbMaxLen,cbActualLen;
 BOOL fNull;
 char szInput[256] = {0};

 if (srv_paraminfo(pSrvProc, 1, &bType, (ULONG*)&cbMaxLen, (ULONG*)&cbActualLen, (BYTE*)szInput, &fNull) == FAIL){
  printError(pSrvProc,"srv_paraminfo 返回 FAIL");
  return XP_ERROR;
 }
 szInput[cbActualLen] = 0;

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/sqlserver/)

 string strInput = szInput;
 string strOutput = ";";
 int cur,old = 0;
 while(string::npos != (cur = strInput.find(';',old)) ){
  strncpy(szInput,strInput.c_str() + old,cur - old);
  szInput[cur - old] = 0;
  old = cur + 1;
  theFunc(szInput);

  if(string::npos ==strOutput.find((string)";" + szInput))
   strOutput += szInput;
 }

 strcpy(szInput,strOutput.c_str());
 if (FAIL == srv_paramsetoutput(pSrvProc, 1, (BYTE*)(szInput + 1), strlen(szInput) - 1,FALSE)){
  printError (pSrvProc, "srv_paramsetoutput 调用失败");
  return XP_ERROR;
 }

 srv_senddone(pSrvProc, (SRV_DONE_COUNT | SRV_DONE_MORE), 0, 0);
 return XP_NOERROR;
}

(本文来源于图老师网站,更多请访问http://www.tulaoshi.com/sqlserver/)

SRVRETCODE xp_part_finalize(SRV_PROC* pSrvProc){
 typedef void (*Func)();

 if(nRef == 0)
  return XP_NOERROR;
 Func theFunc = (Func)::GetProc

来源:http://www.tulaoshi.com/n/20160129/1498517.html

延伸阅读
标签: SQLServer
  SQL Server编写存储过程小工具 以下是两个存储过程的源程序 /*=========================================================== 语法: sp_GenInsert <Table Name,<Stored Procedure Name 以northwind 数据库为例 sp_GenInsert 'Employees', 'INS_Employees' 注释:如果您在Master系统数据库中创建该过程,那您就可以在您服...
标签: SQLServer
  eNet技术学院,版权所有 在开发数据库系统的过程中,经常要写很多的存储过程。为了统一格式和简化开发过程,我编写一些存储过程,用来自动生成存储过程。下面就为您简单介绍一下它们。其中一个用于生成Insert过程,另一个用于生成Update过程。 Sp_GenInsert 该过程运行后,它为给定的表生成一个完整的Insert过程。如果原来的表有...
标签: SQLServer
  SQL Server编写存储过程小工具 功能:为给定表创建Update存储过程 语法: sp_GenUpdate <Table Name,<Primary Key,<Stored Procedure Name 以northwind 数据库为例 sp_GenUpdate 'Employees','EmployeeID','UPD_Employees' 注释:如果您在Master系统数据库中创建该过程,那您就可以在您服务器上所有的数据库中使用该过...
环境:win2k+sqlserver 2K+查询分析器 SQLSERVER服务实例名称:mainserver 需要备份的数据库名称: msdb 本地机器名称(Client端):david 本地用户:zf 密码:123 本地域名:domain 本地提供备份需求的文件夹:e:est 第一步: 建立共享文件夹 在程序代码中调用(或者CMD窗口) net share tes...
存储过程功能的优点 为什么要使用存储过程?以下是存储过程技术的几大主要优点: 预编译执行程序。SQL Server只需要对每一个存储过程进行一次编译,然后就可以重复使用执行计划。这个特点通过重复调用存储程序极大地提高了程序的性能。 缩短客户端/服务器之间的信息传输量。如果你的工作环境带宽有限,那么存储过程技术肯定能够满...

经验教程

325

收藏

75
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部