zoukankan      html  css  js  c++  java
  • 转载Delphi 2010下使用sqlitesimpledelphi连接SQLite数据库及中文乱码问题的解决

    Delphi 2010下使用sqlitesimpledelphi连接SQLite数据库及中文乱码问题的解决

    应女朋友的要求,要写一款销售管理的软件。用于管理服装店每天的销售记录,已及管理服装店的客户,并对客户进行生日提醒

    因为之前使用C#写过一款家庭管理软件,主要是自己用,所以使用了服务器型数据库MySQL,积攒了一些数据库软件的开发经验。

    针对这次的软件需求决定采用SQLite,因为本人比较鄙视Access,主要是因为不能跨平台,而且在实际需求上SQLite更适合。

    因为女朋友要求尽快,那就必须得要快了。。。

    这次采用了Delphi+SQLite的结构来写的,之前一直喜欢Delphi,但是没有深入涉足,最重要的原因就是不支持Unicode,Delphi易主之后,开发很活跃,而且全面引进Unicode支持,所以决定重拾Delphi。

    首先简单说说SQLite在Delphi中的应用,SQLite是使用纯C写的,本身所有涉及字符串的地方都使用了UTF8(Unicode编码的一种存储传输格式),足可以看出作者的先见之明。如果使用VC(本人的主要开发工具),Unicode的问题几乎不存在,但是因为Delphi引入Unicode不久,很多控件暂时没有Unicode版的,就比如这里的sqlitesimpledelphi,原作者一直没有添加Unicode支持。

    刚开始搜集Delphi封装SQLite的控件的时候,因为sqlitesimpledelphi使用比较方便,封装也比较合理,用起来很舒服,但是直到软件Beta版都写出来了,乱码问题才暴露出来。。。所以只能自己改了。

    下面就进入正题说说sqlitesimpledelphi在Delphi 2010中的应用

    简单说sqlitesimpledelphi最重要的两个类是TSQLiteDatabase和TSQLiteTable

    TSQLiteDatabase负责连接数据库,和执行SQL语句。TSQLiteTable负责获取SQL的执行结果。

    (先引入SQLiteTable3单元)

    1.连接数据库

    要连接SQLite数据库,首先创建TSQLiteDatabase实例,在调用构造函数时传入数据库文件名,要使用UTF8Encode函数将文件名字符串编码为UTF8,尤其当数据库名中存在非ASCII码时

    例如:database:=TSQLiteDatabase.Create(datafile);

    2.执行SQL语句

    SQL语句可以简单分为两种:有结果和没结果

    对于没有结果的比如UPDATE,INSERT等,调用TSQLiteDatabase实例的ExeSQL函数,传入SQL。这里也要用UTF8Encode将SQL语句编码为UTF8编码

    会返回结果的,比如SELECT等,调用TSQLiteDataBase实例的GetTable函数,获得TSQLiteTable实例,通过该实例可以获取数据库的返回结果。一样的,SQL语句要用UTF8Encode进行编码

    例如:

        strsql:='insert into users(name,adddate) values('''+user.name+''','''+user.adddate+''')';
        database.ExecSQL(UTF8Encode(strsql));

        strsql:='select * from users where name='''+user.name+'''';
        table:=database.GetTable(UTF8Encode(strsql));

    3.获取SQLite数据库返回的结果

    获取结果由GetTable函数返回的TSQLiteTable实例得到

    主要使用如下子函数:

    Count函数可以获得数据行的行数,FieldIsNull函数可以判断对应的的字段是否为空。Next和Previous函数移动当前的数据行指针。

    获取数据字段:

    FieldAsString

    FieldAsInteger

    FieldAsBlob

    FieldAsDouble

    FieldAsBlobText

    上述函数的参数都是对应字段的序号,如果不能确定字段的序号,可以使用函数TSQLiteTable实例的FieldIndex属性获得,属性的参数为字段名字符串,一样的最好用UTF8Encode进行编码。

    如果对应的字段值为字符串类型,也可以使用FieldByName属性,传入字段名即可直接获得。

    在这里,如果对应的字段为字符串类型,则需要将其使用UTF8Decode进行解码,否则会出现乱码问题,因为SQLite返回的字符串为UTF8编码的,但是sqlitesimpledelphi在封装SQLite函数时将其声明为AnsiString,所以必须手动进行解码。

    最后一点重要提示,当使用完GetTable返回的TSQLiteTable实例之后,需要由调用方调用Free函数进行释放,否则会内存泄露,一定要注意。

    例如:

    table:=database.GetTable(UTF8Encode(strsql));
    try
    for i:= 1 to table.Count do
        begin
          // 对每一行检索信息
          New(pinfo);
          pinfo^.name:=UTF8Decode(table.FieldAsString(table.FieldIndex['name']));
          pinfo^.size:=UTF8Decode(table.FieldByName['size']);
          。。。。
          // 插入到链表中
          infolist.Add(pinfo);
          table.Next;
        end;
    finally
        table.Free;
    end;

    3.断开数据库连接,销毁TSQLiteDatabase实例即可

    例如:database.Free;

    #################################################################################################

    如果使用原版的sqlitesimpledelphi,当使用UTF8Decode函数之后,可能依然存在乱码问题,一个表现就是,最后一个汉字显示为框,后面跟一个问号,其他的汉字解码正常。

    后来跟踪了一下sqlitesimpledelpi的源代码,从SQLite获取的字符串数据是正确的,但是因为TSQLiteTable的构造函数在读取SQLite返回的UTF8字符串时使用了setstring函数,强行将数据字段进行了转换,引起字符串长度出现错误,所以在UTF8Decode解码时出现了漏解码的问题。

    我这里给出一种解决方案,基本修复了该Bug。

    1.将TSQLiteTable的构造函数Create中的setstring行注释掉。将thisStringValue的生命从pstring改为PRawByteString。将临时数据ptrValue直接赋值给thisStringValue。

    2.相关函数/属性声明修改如下:

    function GetFields(I: cardinal): string; 改为function GetFields(I: cardinal): RawByteString;

    function GetFieldByName(FieldName: string): string; 改为function GetFieldByName(FieldName: string): RawByteString;

    function FieldAsString(I: cardinal): string;改为function FieldAsString(I: cardinal): RawByteString;

    property Fields[I: cardinal]: string read GetFields;改为property Fields[I: cardinal]: RawByteString read GetFields;

    property FieldByName[FieldName: string]: string read GetFieldByName;改为property FieldByName[FieldName: string]: RawByteString read GetFieldByName;

    下面提供的是最新的sqlitesimpledelphi原版和修正版。

    原版网盘下载1(UUSHARE)    

    原版网盘下载2(115网盘)      

    修正版网盘下载1(UUSHARE)

    修正版玩盘下载2(115网盘)

    截几张图上来看看:

    add3a2ec9e5e370f27979157

    6d53dbf96289066b242df256

  • 相关阅读:
    jQuery选择器大全(48个代码片段+21幅图演示)
    抽象和模型
    什么叫做精通,我来给大家解释一下
    设置浏览器固定大小的固定位置的方法
    selenium对浏览器属性操作的方法
    selenium 最大化浏览器是解决浏览器和驱动不匹配的方法如下
    java selenium手动最大化chrome浏览器的方法
    java selenium启动火狐浏览器报错:Cannot find firefox binary in PATH. Make sure firefox is installed. OS appears to be: VISTA Build info: version: '3.8.1', revision: '6e95a6684b', time: '2017-12-01T19:05:14.666Z
    mygenerator().next() AttributeError: 'generator' object has no attribute 'next'
    马的遍历 搜索
  • 原文地址:https://www.cnblogs.com/MaxWoods/p/1849074.html
Copyright © 2011-2022 走看看