SDK 是如何存储事件数据的?

更多信息,可关注公众号:神策技术社区

一、前言

为了最大限度地保证事件数据的准确性、完整性和及时性,数据采集 SDK 需要及时地将事件数据同步到服务端。但在某些情况下,比如手机处于断网环境,或者根据实际需求只能在 Wi-Fi 环境下才能同步数据等,可能会导致事件数据同步失败或者无法进行同步。因此,数据采集 SDK 需要先把事件数据缓存在本地,待符合一定的策略(条件)之后,再去同步数据[1]。

二、数据存储方式

在 iOS 应用程序中,从 “数据缓存在哪里” 这个维度看,缓存一般分为两种类型:

  • 内存缓存
  • 磁盘缓存

内存缓存是将数据缓存在内存中,供应用程序直接读取和使用。优点是读写速度极快。缺点是由于内存资源有限,应用程序在系统中申请的内存,会随着应用程序生命周期的结束而被释放。这就意味着,如果应用程序在运行的过程中被用户强杀或者出现崩溃的情况,都有可能导致内存中缓存的数据丢失。因此,将事件数据缓存在内存中不是最佳选择。

磁盘缓存是将数据缓存在磁盘空间中,其特点正好与内存缓存相反。磁盘缓存容量大,但是读写速度相对于内存缓存来说要慢一些。不过磁盘缓存是持久化存储,不受应用程序生命周期的影响。一般情况下,一旦数据成功保存在磁盘中,丢失的风险就非常低。因此,即使磁盘缓存数据读写速度较慢,但综合考虑下,磁盘缓存是缓存事件数据的最优选择。

由于磁盘缓存是一种可以持久化存储的方案,对于存储事件数据是一种最优的选择。在 iOS 中有多种持久化存储的方案,比如 KeyChain、NSUserDefaults、文件存储、数据库存储等都可以做持久化存储。那我们的事件数据使用哪种方案比较好呢?

我们知道 KeyChain、NSUserDefaults 是一种轻量级的存储方案,比如登录用户的用户名、登录状态等,使用 KeyChain 或者 NSUserDefaults 是一种不错的选择。但是对于大量的事件数据而言,这两种存储方案就无能为力了。

文件存储可以满足存储大量数据的需求,因此可以使用文件来存储采集的事件数据。其实,在 SDK 的一些前期版本,我们就是使用文件来存储事件数据的。文件存储相对来说还是比较简单的,主要操作就是写文件和读文件。我们每次都是将所有的数据写入同一个文件,写入的数据量越大,文件缓存性能越好。当然,文件存储还是不够灵活的,我们很难使用更细的粒度去操作数据,比如,很难对其中的某一条数据进行读和写的操作。

有没有其他的方式,可以满足对数据灵活操作的需求呢?答案是肯定的,数据库就满足这个需求。在 iOS 应用程序中,使用的数据库一般是 SQLite 数据库。SQLite 是一个轻量级的数据库,数据存储简单高效,使用也非常简单。相对于文件存储来说,数据库存储更加灵活,可以实现对单条数据的插入、查询和删除操作,同时调试也更容易[1]。

三、事件数据存储

3.1存储策略

实现 SDK 中的数据库时,为了保证数据的完整性和准确性,采用了较为完善的存储策略:

  1. 开发者在初始化 SDK 时,可以根据需要通过 - setMaxCacheSize: 方法设置本地缓存事件的最大条数。本地缓存事件的默认值是 10000 条。当开发者设置的最大缓存事件条数小于 10000 时,则使用默认值;

  2. 执行数据采集任务时,采集的数据首先缓存到本地数据库。数据写入时,会判断数据库里缓存的事件条数是否超过设定的最大值;如果超过设定的最大缓存事件条数,则删除最先入库的 100 条数据,然后执行入库操作;

  3. SDK 会定时检查是否满足上报策略,满足上报策略时,会把数据库里的数据打包上报到服务端,上报成功后会删除已上报的数据,上报失败则不删除。

3.2 数据库表的设计

SDK 采集的事件数据中,会有很多字段,比如事件名称、预置公共属性和用户自定义属性等。虽然事件数据中包含的属性比较多,但是存储数据无需关心具体的细节,可以将一个事件数据当做整体存储到数据表的一个字段中,从而提高数据的操作效率。

具体的结构如表 3-1 所示:


3-1表.png

表 3-1 事件数据的存储结构

3.3具体实现

SDK 采集数据过程中,会频繁的执行缓存数据、上报数据和删除数据等耗时操作。为了保证 SDK 的数据采集不影响用户的 App 性能,这些耗时的操作全部在子线程中完成。SDK 在执行数据存储和数据上报会涉及到 SAEventStore 、SAEventFlush、SAHTTPSession、SAEventTracker 等几个关键类:

  • SAEventStore: 负责事件数据的存储操作;

  • SAEventFlush: 负责数据的上报;

  • SAHTTPSession: 负责将上报数据的任务添加到队列,等待执行;

  • SAEventTracker: 负责 track 事件和检查是否达到上报条件。

3.3.1. 初始化工具类

  1. 在初始化 SDK 时,会对 SAEventTracker 工具类进行初始化:
_eventTracker = [[SAEventTracker alloc] initWithQueue:_serialQueue];
  1. 在 SAEventTracker 的初始化方法里对 SAEventStore 和 SAEventFlush 两个工具类进行初始化:
- (instancetype)initWithQueue:(dispatch_queue_t)queue {
    self = [super init];
    if (self) {
        _queue = queue;
 
        dispatch_async(self.queue, ^{
            self.eventStore = [[SAEventStore alloc] initWithFilePath:[SAFileStore filePath:@"message-v2"]];
            self.eventFlush = [[SAEventFlush alloc] init];
        });
    }
    return self;
}
  1. 初始化 SAEventStore 时,传入的 filePath 参数是用于创建数据库的路径。SAEventStore 的初始化如下:
- (instancetype)initWithFilePath:(NSString *)filePath {
    self = [super init];
    if (self) {
        NSString *label = [NSString stringWithFormat:@"cn.sensorsdata.SAEventStore.%p", self];
        _serialQueue = dispatch_queue_create(label.UTF8String, DISPATCH_QUEUE_SERIAL);
        // 直接初始化,防止数据库文件,意外删除等问题
        _recordCaches = [NSMutableArray array];
 
        [self setupDatabase:filePath];
    }
    return self;
}
  1. 在方法 - setupDatabase: 里对封装了数据库的工具类 SADatabase 初始化,在 SADatabase 创建了数据库文件和表:
- (instancetype)initWithFilePath:(NSString *)filePath {
    self = [super init];
    if (self) {
        _filePath = filePath;
        _serialQueue = dispatch_queue_create("cn.sensorsdata.SADatabaseSerialQueue", DISPATCH_QUEUE_SERIAL);
        [self createStmtCache];
        [self open];
        [self createTable];
    }
    return self;
}

3.3.2. 数据入库

  1. 对于校验成功的数据,会尝试把数据存入到数据库,如果数据库打开失败,会把数据先保存在内存中的一个数组中:
- (BOOL)insertRecord:(SAEventRecord *)record {
    BOOL success = [self.database insertRecord:record];
    if (!success) {
        [self.recordCaches addObject:record];
    }
    return success;
}
  1. 在监听到数据库创建成功时,会尝试把缓存在内存中的数据插入数据库,如果插入失败,会重试 3 次:
#pragma mark - observe
 
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
    if (context != SAEventStoreContext) {
        return [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
    }
    if (![keyPath isEqualToString:SAEventStoreObserverKeyPath]) {
        return;
    }
    if (![change[NSKeyValueChangeNewKey] boolValue] || self.recordCaches.count == 0) {
        return;
    }
    // 对于内存中的数据,重试 3 次插入数据库中。
    for (NSInteger i = 0; i < 3; i++) {
        if ([self.database insertRecords:self.recordCaches]) {
            [self.recordCaches removeAllObjects];
            return;
        }
    }
}
  1. 插入事件数据是比较频繁的操作,如果每次都做 “预解析 SQL 语句” 的操作,将会造成资源的大量浪费。对于插入数据来说,每次操作的 SQL 语句都是相同的,因此 “预解析 SQL 语句” 只需执行一次即可。由于每次需要绑定不同的数据,我们只需要重置一下之前的 sqlite3_stmt,然后重新绑定新的数据即可[1]。插入数据的逻辑如下:
- (BOOL)insertRecord:(SAEventRecord *)record {
    if (![record isValid]) {
        SALogError(@"%@ input parameter is invalid for addObjectToDatabase", self);
        return NO;
    }
    if (![self databaseCheck]) {
        return NO;
    }
 
    if (![self preCheckForInsertRecords:1]) {
        return NO;
    }
 
    NSString *query = @"INSERT INTO dataCache(type, content) values(?, ?)";
    sqlite3_stmt *insertStatement = [self dbCacheStmt:query];
    int rc;
    if (insertStatement) {
        sqlite3_bind_text(insertStatement, 1, [record.type UTF8String], -1, SQLITE_TRANSIENT);
        sqlite3_bind_text(insertStatement, 2, [record.content UTF8String], -1, SQLITE_TRANSIENT);
        rc = sqlite3_step(insertStatement);
        if (rc != SQLITE_DONE) {
            SALogError(@"insert into dataCache table of sqlite fail, rc is %d", rc);
            return NO;
        }
        self.count++;
        SALogDebug(@"insert into dataCache table of sqlite success, current count is %lu", self.count);
        return YES;
    } else {
        SALogError(@"insert into dataCache table of sqlite error");
        return NO;
    }
}

3.3.3. 数据删除

  1. 在达到上报条件时,会触发数据上报。默认情况下是每 15 秒上报一次,或者缓存的数据达到 100 条时进行一次上报。在非 Debug 模式下,每次上报 50 条数据:
- (void)flushAllEventRecords {
    if (![self canFlush]) {
        return;
    }
    BOOL isFlushed = [self flushRecordsWithSize:self.isDebugMode ? 1 : 50];
    if (isFlushed) {
        SALogInfo(@"Events flushed!");
    }
}
  1. 对于已经上报成功的数据,SDK 会将其从数据库中移除,防止数据的重复上报:
......
// flush
__weak typeof(self) weakSelf = self;
[self.eventFlush flushEventRecords:encryptRecords completion:^(BOOL success) {
__strong typeof(weakSelf) strongSelf = weakSelf;
void(^block)(void) = ^ {
if (!success) {
[strongSelf.eventStore updateRecords:recordIDs status:SAEventRecordStatusNone];
return;
}
// 5. 删除数据
if ([strongSelf.eventStore deleteRecords:recordIDs]) {
[strongSelf flushRecordsWithSize:size];
}
};
if (sensorsdata_is_same_queue(strongSelf.queue)) {
block();
} else {
dispatch_sync(strongSelf.queue, block);
}
}];
......

3.4数据流程

当 SDK 调用 track 相关方法时,首先是 SDK 会对事件数据的各项属性进行合法性校验,校验通过后将事件数据存储到数据库。在 SDK 初始化时启动的定时器会定时检查是否满足上报条件,当符合上报时,再将数据上报到服务端,最后再把上报成功的数据从数据库中删除。工作流程如图 3-1 所示:


未命名文件.jpg.png

图 3-1 数据采集流程

四、总结

本文介绍了神策 iOS SDK[2] 中使用到的存储方式和具体使用流程。希望通过这篇文章的介绍,大家能够对神策 iOS SDK 存储模块有一个较为全面的了解。

参考文献:

[1]王灼洲.iOS全埋点解决方案[M].北京:机械工业出版社,2020:162-197.

[2]https://github.com/sensorsdata/sa-sdk-ios

更多信息,可关注公众号:神策技术社区

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容