当前所在位置:珠峰网资料 >> 计算机 >> 计算机等级考试 >> 正文
计算机二级辅导:VisualC++ADO数据库编程入门(4)
发布时间:2009/12/6 10:13:49 来源:城市学习网 编辑:admin

  1. 访问长数据
  在Microsoft SQL中的长数据包括text、image等这样长类型的数据,作为二进制字节来对待。
  可以用Field对象的GetChunk和AppendChunk方法来访问。每次可以读出或写入全部数据的一部分,它会记住上次访问的位置。但是如果中间访问了别的字段后,就又得从头来了。
  请看下面的例子:
  //写入一张照片到数据库:
  VARIANT varChunk;
  SAFEARRAY *psa;
  SAFEARRAYBOUND rgsabound[1];
  //VT_ARRAY│VT_UI1
  CFile f("h:\\aaa.jpg",Cfile::modeRead);
  BYTE bVal[ChunkSize+1];
  UINT uIsRead=0;
  //Create a safe array to store the array of BYTES
  while(1)
  {
  uIsRead=f.Read(bVal,ChunkSize);
  if(uIsRead==0)break;
  rgsabound[0].cElements =uIsRead; rgsabound[0].lLbound = 0; psa = SafeArrayCreate(VT_UI1,1,rgsabound);
  for(long index=0;index<uIsRead;index++)
  {
  if(FAILED(SafeArrayPutElement(psa,&index,&bVal[index]))) ::MessageBox(NULL,"啊,又出毛病了。","提示",MB_OK │ MB_ICONWARNING);
  }
  varChunk.vt = VT_ARRAY│VT_UI1; varChunk.parray = psa;
  try{
  m_pRecordset->Fields->GetItem("photo")->AppendChunk(varChunk);
  }
  catch (_com_error &e)
  {
  CString str=(char*)e.Description(); ::MessageBox(NULL,str+"\n又出毛病了。","提示",MB_OK │ MB_ICONWARNING);
  }
  ::VariantClear(&varChunk);
  ::SafeArrayDestroyData( psa);
  if(uIsRead<ChunkSize)break;
  }//while(1)
  f.Close();
  //从数据库读一张照片:

  CFile f;
  f.Open("h:\\bbb.jpg",Cfile::modeWrite│Cfile::modeCreate);
  long lPhotoSize = m_pRecordset->Fields->Item["photo"]->ActualSize;
  long lIsRead=0;
  _variant_t varChunk;
  BYTE buf[ChunkSize];
  while(lPhotoSize>0)
  {
  lIsRead=lPhotoSize>=ChunkSize? ChunkSize:lPhotoSize;
  varChunk = m_pRecordset->Fields->
  Item["photo"]->GetChunk(lIsRead);
  for(long index=0;index<lIsRead;index++)
  {
  ::SafeArrayGetElement(varChunk.parray,&index,buf+index);
  }
  f.Write(buf,lIsRead);
  lPhotoSize-=lIsRead;
  }//while()
  f.Close();
  2. 使用SafeArray问题
  学会使用SafeArray也是很重要的,因为在ADO编程中经常要用。它的主要目的是用于automation中的数组型参数的传递。因为在网络环境中,数组是不能直接传递的,而必须将其包装成SafeArray。实质上SafeArray就是将通常的数组增加一个描述符,说明其维数、长度、边界、元素类型等信息。SafeArray也并不单独使用,而是将其再包装到VARIANT类型的变量中,然后才作为参数传送出去。在VARIANT的vt成员的值如果包含VT_ARRAY│...,那么它所封装的就是一个SafeArray,它的parray成员即是指向SafeArray的指针。 SafeArray中元素的类型可以是VARIANT能封装的任何类型,包括VARIANT类型本身。
  使用SafeArray的具体步骤:
  方法一:
  包装一个SafeArray:
  (1). 定义变量,如:
  VARIANT varChunk;
  SAFEARRAY *psa;
  SAFEARRAYBOUND rgsabound[1];
  (2). 创建SafeArray描述符:
  uIsRead=f.Read(bVal,ChunkSize);//read array from a file.
  if(uIsRead==0)break;
  rgsabound[0].cElements =uIsRead; rgsabound[0].lLbound = 0; psa = SafeArrayCreate(VT_UI1,1,rgsabound);
  (3). 放置数据元素到SafeArray:
  for(long index=0;index<uIsRead;index++)

  {
  if(FAILED(SafeArrayPutElement(psa,&index,&bVal[index]))) ::MessageBox(NULL,"出毛病了。","提示",MB_OK │ MB_ICONWARNING);
  }
  一个一个地放,挺麻烦的。
  (4). 封装到VARIANT内:
  varChunk.vt = VT_ARRAY│VT_UI1; varChunk.parray = psa;
  这样就可以将varChunk作为参数传送出去了。
  读取SafeArray中的数据的步骤:
  (1). 用SafeArrayGetElement一个一个地读
  BYTE buf[lIsRead];
  for(long index=0;index<lIsRead;index++)
  {
  ::SafeArrayGetElement(varChunk.parray,&index,buf+index);
  }
  就读到缓冲区buf里了。
  方法二:
  使用SafeArrayAccessData直接读写SafeArray的缓冲区:
  (1). 读缓冲区:
  BYTE *buf;
  SafeArrayAccessData(varChunk.parray, (void **)&buf); f.Write(buf,lIsRead); SafeArrayUnaccessData(varChunk.parray);
  (2). 写缓冲区:
  BYTE *buf;
  ::SafeArrayAccessData(psa, (void **)&buf);
  for(long index=0;index<uIsRead;index++)
  {
  buf[index]=bVal[index];
  }
  ::SafeArrayUnaccessData(psa);
  varChunk.vt = VT_ARRAY│VT_UI1; varChunk.parray = psa;
  这种方法读写SafeArray都可以,它直接操纵SafeArray的数据缓冲区,比用SafeArrayGetElement和 SafeArrayPutElement速度快。特别适合于读取数据。但用完之后不要忘了调用::SafeArrayUnaccessData(psa),否则会出错的。

广告合作:400-664-0084 全国热线:400-664-0084
Copyright 2010 - 2017 www.my8848.com 珠峰网 粤ICP备15066211号
珠峰网 版权所有 All Rights Reserved