分享

ADODB.pas的一些Bug,解决Number型转整型、ftWideString长度、ftBCD精度问题

 fshell 2013-07-03
D6同D7从oracle中数据类型为NUMBER(7)的负数时,数据到前端后负数变为正数?
http://www./delphibbs/dispq.asp?lid=2393545
来自:chnplzh, 时间:2004-1-8 14:55:00, ID:2400559
估计找到一些原因了,以下摘自adodb.pas单元,将以下语句注释掉就行了,这应该不是
adodb.pas的bug,而是delphi太聪明,认为类型为Numeric,且没有小数NumericScale,且
F.Precision < 10,也就是前面所说的精度10以下,就自动转换为ftInteger类型。解决方
法,我觉得还是不要修改adodb.pas单元,而是通过修改字段类型来处理,如果是Numeric
(7,0)这样的定义,那就干脆定义为整型数据类型,要不就定义为Numeric(7,1),也就是
float或decimal类型,即可避开以上问题。
谢谢楼主,您又让我发现了一个新问题,并被逼得研读vcl的源代码。
{
        if (F.Type_ = adNumeric) and (F.NumericScale = 0) and
           (F.Precision < 10) then
          FieldType := ftInteger;
}


ADO还值得信任么?
http://www./delphibbs/dispq.asp?lid=2148295
来自:三代坦克, 时间:2003-9-2 11:30:00, ID:2150213
早就发现了.ado express的处理出问题了. 这是我在borland新闻组发现的解决方法.
当时是发现  ado express + oracle oledb816. 出的问题.
现在看来处理 numeric(x,0) 都会有问题.
修改adodb.pas  
       // modify by SanDaiTanke  2002/10/02 为了解决ADO + oracle oledb816.
        // integer,number(x,0)整数不能正确返回类型的问题。
       (*OLD:
          if (F.Type_ = adNumeric) and (F.NumericScale = 0) and
           (F.Precision < 10) then
          FieldType := ftInteger;
        *)

        if (F.Type_ = adNumeric) and (F.NumericScale = 0) then
         begin
          if (F.Precision < 5) then
            FieldType := ftSmallInt
          else if (F.Precision < 10) then
            FieldType := ftInteger
          else
            FieldType := ftLargeInt;
        end;
        if (FieldType in [ftSmallInt, ftInteger, ftLargeInt])
        then
          FPrecision := F.Precision;
         // modify by SanDaiTanke END 2002/10/02


           // SanDaiTanke 2002/10/02  ado oledb816 oracle 解决不能返回负数的错误
        // OLD  if vt = VT_UI1 then
        //    SmallInt(Buffer^) := Byte(cVal) else
        //    SmallInt(Buffer^) := iVal;

           begin                                         (* SanDaiTanke add *)
            if vt = VT_UI1 then                         (* SanDaiTanke add*)
              SmallInt(Buffer^) := Byte(cVal) else      (* SanDaiTanke add*)
              SmallInt(Buffer^) := iVal;                 (* SanDaiTanke add*) 
            if (vt = VT_Decimal) and                    (* SanDaiTanke add*)
               (wReserved1 = $8000) and                 (* SanDaiTanke add *)
               (SmallInt(Buffer^) > 0) then             (* SanDaiTanke add *)
              SmallInt(Buffer^) := 0-SmallInt(Buffer^); (* SanDaiTanke add *)
          end;                                          (* SanDaiTanke add *)
      // SanDaiTanke END 2002/10/02
        ftWord:
          if vt = VT_UI1 then
            Word(Buffer^) := bVal else
            Word(Buffer^) := uiVal;

    // SanDaiTanke 2002/10/02  ado oledb816 oracle 解决不能返回负数的错误           
      //  ftAutoInc, ftInteger:
      //    Integer(Buffer^) := lVal;
           ftAutoInc, ftInteger:
          begin                                         (* SanDaiTanke add  *)
            Integer(Buffer^) := lVal;
            if (vt = VT_Decimal) and                    (* SanDaiTanke add  *)
               (wReserved1 = $8000) and                 (* SanDaiTanke add  *)
               (Integer(Buffer^) > 0) then              (* SanDaiTanke add  *)
              Integer(Buffer^) := 0-Integer(Buffer^);   (* SanDaiTanke add  *)
          end;
    // SanDaiTanke 2002/10/02 END   ado oledb816 oracle 解决不能返回负数的错误

       // ftLargeInt: LargeInt(Buffer^) := Decimal(Data).Lo64;
         ftLargeInt:
          begin                                         (* PtUpd *)
            LargeInt(Buffer^) := Decimal(Data).Lo64;
            if (vt = VT_Decimal) and                    (* PtUpd *)
               (wReserved1 = $8000) and                 (* PtUpd *)
               (LargeInt(Buffer^) > 0) then             (* PtUpd *)
              LargeInt(Buffer^) := 0-LargeInt(Buffer^); (* PtUpd *)
          end;


Delphi 7中TCustomADODataSet组件的一些错误
http://www./delphi/showtl?ID=47491
小弟近日发现Delphi 7中TCustomADODataSet组件的一些错误,小结如下:     
     对应的单元文件是ADODB.pas。
     
     一、nvarchar字段长度错误     
     在SQL Server数据库表中定义了nvarchar类型的字段,但程序中得到的TField对象的Size属性只有数据库中的一半。同时这还会导致一个不明症状:在程序中动态添加的Lookup字段若结果值类型为TWideString,则汉字串会被截断一部分并在末尾出现问号。     
     1、找到procedure TCustomADODataSet.InternalInitFieldDefs;
     2、定位于这样一段:
     case FieldType of
     ftString, ftWideString, ftBytes, ftVarBytes, ftFixedChar:
     FSize := F.DefinedSize;
     3、 改成这样:
     case FieldType of
     ftString, ftBytes, ftVarBytes, ftFixedChar:
     FSize := F.DefinedSize;
     ftWideString:
     FSize := F.DefinedSize * 2; // 对于宽字符,其字节长度为定义值的2倍
     
     二、浮点数的小数部分丢失位数     
     TCustomADODataSet.EnableBCD默认为True,限制了小数位为4,这本是为了达到高精度,但是当数据库表中numeric字段定义的小数位超过4时,就不得不舍弃精度,得按实际小数位来做了:     
     1、找到procedure TCustomADODataSet.InternalInitFieldDefs;
     2、定位于其中的过程 procedure AddFieldDef(F: Field; FieldDefs: TFieldDefs);
     3、找到第一条语句:FieldType := ADOTypeToFieldType(F.Type_, EnableBCD);
     4、改成这样:
     // Added by iMe
     if F.NumericScale >4 then
     FieldType := ADOTypeToFieldType(F.Type_, False)
     else
     // -------------
     FieldType := ADOTypeToFieldType(F.Type_, EnableBCD); // 这行是原来的


Delphi中两个BUG的分析与修复(选择自 lifanxi 的 Blog)
http://dev.csdn.net/article/18/18720.shtm
http://www./netbook/tech/delphi/html/delphi.ohchina.155.htm
  在使用Delphi 7进行三层数据库开发时,遇到了两个小问题,通过反复试验,终于找出了Delphi 7中的两个小BUG并进行了修复(好像Delphi 6中也有相同的BUG),撰写此文与大家一起分享成功的喜悦。我也是初学Delphi,文中一定存在不少说的不对的地方,还请各位朋友多多指正。

  BUG1.传参时中文被截断的问题:

  BUG再现的方法:

  后台用SQL Server 2000,里面有一个XsHeTong表用于试验,您可以根据您的实际情况进行调整。

  先创建一个数据服务器:新建项目,创建一个远程数据模块,上面放置ADOConnection、ADODataSet、DataSetProvider各一,并做好相应设置,其中ADODataSet的ComamndText留空,并把它的Option中的poAllowCommandText设置为True。编译运行。

  再创建客户端程序:新建项目,在窗体上放置DCOMConnection,连上前面上创建的数据服务器,再放置一个ClientDataSet,把它的连接设成这里的DCOMConnection,并设置它的ProviderName为上面的服务器上的DataSetProvider的名字。最后放置DataSource和DBGrid各一并作相应设置用于查看结果,再放置一Button用于测试。

  在Button的OnClick中写下类似于下面的代码(这里我用了XsHeTong的表和它的两个字段HTH(char 15)、GCMC(varchar 100),您可以根据你的实际测试情况进行调整):

  with ClientDataSet1 do
  begin
   Close;
   CommandText := 'Insert Into XsHeTong(HTH, GCMC) values(:HTH,:GCMC)';
   Params[0].AsString := '12345';
   Params[1].AsString := '会截断的中文字';
   Execute;
   Close;
   CommandText := 'Select * from XsHeTong';
   Open;
  end;

  运行程序,点击按钮,看到记录被插入了,可惜结果并不正确,“会截断的中文字”变成了“会截断”,但没有中文的“12345”倒是正确的插入了。

  BUG分析与修复:

  为了对照起见,我试着直接用一个ADOConnection和ADOCommand、ADOTable进行C/S构架测试,结果是正确的,中文字不会被切断。这说明了此BUG只在三层构架上出现。

   用SQL Server事件探查器探查提交到SQL Server上运行的语句,发现两层构架与三层构架的情况有以下不同:

  两层构架:
  exec sp_executesql N'Insert into XsHeTong(HTH, GCMC) values(@P1,@P2)', N'@P1 varchar(15),@P2 varchar(100)', '12345', '会截断的中文字'

  三层构架:
  exec sp_executesql N'Insert into XsHeTong(HTH, GCMC) values(@P1,@P2)', N'@P1 varchar(5),@P2 varchar(7)', '12345', '会截断

  显然,两层构架时,参数的长度是按实际库结构传的,三层构架时,参数长度是按实际参数的字符串长度传的,而实际字符串长度又似乎是算错了,没有把一个中文当两个字符长度处理。

  没有办法只好进行跟踪调试,为了调试Delphi的VCL库,需要在工程选项的“Compiler Options”中选上“Use Debug DCUs”。

  先跟踪客户端程序,ClientDataSet1.Execute后,先后经历了TCustomClientDataSet.Exectue、TCustomeClientDataSet.PackageParams、TCustomClientDataSet.DoExecute等一系列函数,一直到AppServer.AS_Execute(ProviderName, CommandText, Params, OwnerData); 把请求提交到服务器均没有什么异常情况,看来问题出在服务器端。

  对服务器进行跟踪,反复试验后,我把重点落在了TCustomADODataSet.PSSetCommandText函数身上,经过反复细致的跟踪,目标越来越精确:TCustomADODataSet.PSSetParams、TParameter.Assign、TParameter.SetValue、VarDataSize。终于找到了BUG的源头:VarDataSize函数,下面是它的代码:

  function VarDataSize(const Value: OleVariant): Integer;
  begin
   if VarIsNull(Value) then
    Result := -1
   else if VarIsArray(Value) then
    Result := VarArrayHighBound(Value, 1) + 1
   else if TVarData(Value).VType = varOleStr then
    begin
     Result := Length(PWideString(@TVarData(Value).VOleStr)^); //出问题的行
     if Result = 0 then
      Result := -1;
    end
   else
    Result := SizeOf(OleVariant);
  end;

  就是在这个函数中计算实参的长度的,它把Value中的值取出地址,并把它作为一个WideString的指针去求字符串长度,结果就导致了“会截断的中文字”这个字符串的长度变成了7,而不是14。

  问题找到了,解决起来也就不困难了,只要简单的把
  Result := Length(PWideString(@TVarData(Value).VOleStr)^); //出问题的行
  改成
  Result := Length(PAnsiString(@TVarData(Value).VOleStr)^); //没问题了
  就可以了。

  但是这样就会导致求英文字符串的长度时长度被加倍了,所以也可以把这一行改成:
  Result := Length(Value); 

  这样,不管是中文还是英文还是中英混合的字符串就都可求得正确的长度了。这就我至今仍百思不解的问题,为什么Borland要绕个圈子通过指针去求参数值的长度呢?哪位朋友知道的话还请给我解释一下,非常感谢!

  有些朋友可能会有疑问,为什么在不通过三层构架来做的时候不产生这个字符串被截断的问题呢?答案并不复杂,在直接通过ADOCommand来向SQL Server发送命令时,它是按表结构来决定参数长度的。它会先向SQL Server发一条

  SET FMTONLY ON select HTH,GCMC from XsHeTong SET FMTONLY OFF

  来获取表结构。而在三层构架下,TCustomADODataSet内部虽然也是用TADOCommand对象来发命令,但它却在取得表结构的后,并不用这个值来作为传参长度,而是重新去按实际参数来计算长度,结果就导致了错误。
 

  BUG2.ClientDataSet的Lookup字段的问题:

  BUG再现的方法:

  新建工程,在上面放置两个ClientDataSet,分别为cds1和cds2,它的数据来源任意,其中cds1为主数据集,在里面增加一个新的Lookup字段,这个Lookup字段根据cds1中的一个字符型的字段值到cds2中找出对应值来。

  运行程序,一般来说是正常的,但是一旦cds1的被Lookup字段中的值出现了一个单引号"'"(您可以修改或新增一条记录,输入单引号试试),立即会导致出错: Unterminated string constant(未结束的字符串常量)。 

  BUG分析与修复:

  这个BUG的产生原因要比上一个明显得多,一定是没有正确处理单引号带来的副作用引起的。

  同样的,我们来跟踪VCL的源码:

  运行程序,出错时打开Call Stack窗口(在View->Debug Windows)菜单中,查看函数调用情况,前面的一些调用是显而易见的,没有问题,我们从跟Lookup有关的地方开始查原因,第一个与Lookup有关的函数调用是TField.CalcLookupValue,我们在这个函数中设置断点,重新运行程序,中断下来后,进行单步调试。

  TCustomClientDataSet.Lookup->TCustomClientDataSet.LocateRecord

  经过上面的几次函数调用,很快的,我们就把目标定在了LocateRecord过程中,在这个过程中,它根据Lookup字段的设置情况,生成相应的过滤条件,然后到目标数据集中把对应的值找到,错就错在过滤条件的生成上了。比如,我们要按cds1中Cust字段(假设是001)的值到cds2中按CustID字段值找到对应的CustName字段值。那生成的条件就应该是[CustID] = '001',但如果Cust的值是aa'bb,按生成的条件就会变成[CustID] = 'aa'bb',显然导致了一个未结束的字符串常量。

  通常我们解决单引号中又出现单引号的情况,只需把引号中的引号写两就行了,这里也是一样,只要让生成的条件变成[CustID] = 'aa''bb'就不会出错了。所以可以这样修改源代码:

  在LocateRecord过程中找到下面的代码:

  ftString, ftFixedChar, ftWideString, ftGUID
   if (i = Fields.Count - 1) and (loPartialKey in Options) then
    ValStr := Format('''%s*''',[VarToStr(Value)]) else
    ValStr := Format('''%s''',[VarToStr(Value)]);

  改成:

  ftString, ftFixedChar, ftWideString, ftGUID:
   if (i = Fields.Count - 1) and (loPartialKey in Options) then
    ValStr := Format('''%s*''',[ StringReplace(VarToStr(Value),'''','''''',[rfReplaceAll])])
   else
    ValStr := Format('''%s''',[ StringReplace(VarToStr(Value),'''','''''',[rfReplaceAll])]);

  也就是在生成过滤条件字符串时把条件的过滤值中的单引号全部一个变两。

  为了确保这样修改的正确性,我查看了TCustomADODataSet中的对应的LocateRecord过程(在用TADODataSet中的Lookup字段时不会因单引号出错,只在用TCustomClientDataSet时有这样的情况),它的处理方法与TCustomClientDataSet稍有不同,它是通过GetFilterStr函数来构造过滤条件的,但在GetFilterStr中,它正确处理了单引号的问题。所以这样来看,没有在TCustomClientDataSet的LocateRecord中正确处理单引号的问题,确实是Borland一个不大不小的疏漏。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多