【问题标题】:Inserting NSData into SQLite on the iPhone在 iPhone 上将 NSData 插入 SQLite
【发布时间】:2010-07-24 23:24:39
【问题描述】:

到目前为止,我已经设法创建了这个方法,用于在 iPhone 上插入 SQLite 数据库:

- (void) insertToDB :(NSString *)Identifier :(NSString *)Name
{
    sqlite3 *database;

    if(sqlite3_open([databasePath UTF8String], &database) == SQLITE_OK)
    {
        char *sql1 = "INSERT INTO table VALUES ('";
        const char *sql2 = [Identifier cStringUsingEncoding:[NSString defaultCStringEncoding]];
        char *sql3 = "', '";
        const char *sql4 = [Name cStringUsingEncoding:[NSString defaultCStringEncoding]];
        char *sql5 = "')";

        char *sqlStatement[255];
        strcpy(sqlStatement, sql1);
        strcat(sqlStatement, sql2);
        strcat(sqlStatement, sql3);
        strcat(sqlStatement, sql4);
        strcat(sqlStatement, sql5);

        sqlite3_stmt *compiledStatement;

        if(sqlite3_prepare_v2(database, sqlStatement, -1, &compiledStatement, NULL) == SQLITE_OK)
        {
            sqlite3_last_insert_rowid(database);
            sqlite3_reset(compiledStatement);
        }

        sqlite3_finalize(compiledStatment);
    }
    sqlite3_close(database);
}

现在我正在考虑将图像存储在数据库中。到目前为止,我已经found this

UIImage *cachedImage = [UIImage imageNamed:@"Icon.png"];
NSData *dataForImage = UIImagePNGRepresentation(cachedImage);

但是我在尝试将这个 NSData 插入到创建 sqlStatement 的 char 数组中时遇到了麻烦。有人知道怎么做吗?

(我在数据库中有一个 blob 类型的字段)。

谢谢

【问题讨论】:

  • 不要将 BLOB 放入数据库。它在所有层面上都非常低效。在数据库中存储路径并将 BLOB 放入文件系统中。
  • 我会强烈推荐使用 Flying Meat 数据库包装器,它已经为您解决了所有这些问题,并且使用起来非常简单: github.com/ccgus/fmdb

标签: iphone objective-c binary image-manipulation nsdata


【解决方案1】:

我会使用sqlite3_stmt 而不是字符串。然后,您可以使用 sqlite3_bind_blob 将 blob 绑定到准备好的语句。

sqlite3_stmt *insert_statement;     
char *sql = "INSERT INTO table (blobcolumn, column2, column3) VALUES (? , ?, ?)" ;
if(sqlite3_prepare_v2(database, sql, -1, &insert_statement, NULL) != SQLITE_OK)
          {
              //handle error
          } 

sqlite3_bind_blob(insert_statement, 1, [dataForImage bytes], [dataForImage length], NULL);

但是,实际上,最好将图像存储在磁盘上并将路径存储在数据库中。

另一种方法是,我用来以 XML 格式发送图像数据的方法是对数据进行 base 64 编码。我在这里将它作为 NSString 上的一个类别:

static char base64EncodingTable[64] = {
'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P',
'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z', 'a', 'b', 'c', 'd', 'e', 'f',
'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v',
'w', 'x', 'y', 'z', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '+', '/'
};

@implementation NSString (NSStringCategories)

+ (NSString *) base64StringFromData: (NSData *)data length: (int)length {
    unsigned long ixtext, lentext;
    long ctremaining;
    unsigned char input[3], output[4];
    short i, charsonline = 0, ctcopy;
    const unsigned char *raw;
    NSMutableString *result;

    lentext = [data length]; 
    if (lentext < 1)
        return @"";
    result = [NSMutableString stringWithCapacity: lentext];
    raw = [data bytes];
    ixtext = 0; 

    while (true) {
        ctremaining = lentext - ixtext;
        if (ctremaining <= 0) 
            break;        
        for (i = 0; i < 3; i++) { 
            unsigned long ix = ixtext + i;
            if (ix < lentext)
                input[i] = raw[ix];
            else
                input[i] = 0;
        }
        output[0] = (input[0] & 0xFC) >> 2;
        output[1] = ((input[0] & 0x03) << 4) | ((input[1] & 0xF0) >> 4);
        output[2] = ((input[1] & 0x0F) << 2) | ((input[2] & 0xC0) >> 6);
        output[3] = input[2] & 0x3F;
        ctcopy = 4;
        switch (ctremaining) {
            case 1: 
                ctcopy = 2; 
                break;
            case 2: 
                ctcopy = 3; 
                break;
    }

    for (i = 0; i < ctcopy; i++)
        [result appendString: [NSString stringWithFormat: @"%c",             base64EncodingTable[output[i]]]];

    for (i = ctcopy; i < 4; i++)
        [result appendString: @"="];

    ixtext += 3;
    charsonline += 4;

    if ((length > 0) && (charsonline >= length))
        charsonline = 0;
    }
    return result;
}

【讨论】:

  • +1 用于 sqlite3_stmt 并将文件保存到磁盘而不是数据库中
  • 谢谢,我要到星期一才能尝试,但我会回来报告的。关于保存到磁盘,这是我过去一直在做的,但我现在正在流式传输和保存图像。我不认为你可以在代码中将图像保存到磁盘?
  • 哦,当然可以。您拥有 NSData:[dataForImage writeToFile: path atomically:YES]; 为您的图像构建一个唯一的文件名并将其保存在文档目录中。然后将该路径保存在数据库中。对于任何超过 100 kb 左右的内容,这比将 BLOB 存储在 sqlite 中要好得多。
  • 啊,好吧,那我得玩一玩了。你还要写在哪里?您可以为安装更新时不会删除的下载图像分配一个区域吗?
  • 是的,Documents 目录就是您所需要的(NSDocumentDirectory)。如果您想在那里创建子文件夹,请参阅此问题:stackoverflow.com/questions/2094376/…。对于临时文件,使用 NSCachesDirectory。
猜你喜欢
  • 2010-10-18
  • 1970-01-01
  • 2012-10-05
  • 2013-02-05
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2011-01-27
  • 1970-01-01
相关资源
最近更新 更多