R&S | 手把手搞推荐[3]:数据集存取思路

上一期讲到我们进行一整套特征工程,然而可怕的是,事实上这样存储的训练集数据体积无敌大,这样的数据在计算图中将遇到内存不足的关键问题,给大家看看我在读取数据的时候的壮观景象:

内存爆炸

因此,在进行LR之前,我还是冒着被大家打的风险,写一篇挽尊之作,教大家设计更合适的方法存取数据集。

简单说思路

口述思路

现在的当务之急是找到一个合适的方式压缩存储空间,从而保证空间复杂度较低,然而在更多现实场景,其实还会涉及很多问题,因此在这里,我统一讲讲如何设计一个好的方式进行数据集的存取,说白了就是要思考这几个问题:

  • 用什么存?纯文本?CSV?SQL?excel?甚至是其他更新更多样化的操作
  • 用什么结构存?

那么,在进行选择的时候,实际上要考虑的是这几个问题:

  • 安全性。存取安全,一定级别下防止泄露。
  • 完整性。不会存在数据出错。
  • 高效性。存取的复杂度要在可控范围内。

实际问题

在我们的这个问题下,安全性不要求;完整性上述方案基本能保证,所以也没问题;问题就在于高效性,现在这个是我们目前面临的重大问题,所以我们要好好处理,保证数据不失真的情况下去处理。

压缩的核心在于略去不必要的信息,或者用更简单的方式来描述更多的信息,例如条件允许的情况下二元数组可以用一元数组+函数的方式存储,那么在此处,我们先看看数据。

  • 我们的数据是一套one-hot数据
  • one-hot数据本身是一种稀疏矩阵的结构
  • 稀疏矩阵中含有大量的0,仅有少部分非0

因此,我们可以用稀疏矩阵的方式进行存取,只记住非零位置下的值,其他位置为0即可。

稀疏矩阵存取

稀疏矩阵的有关理论在此处不赘述,可以自行查阅,此处给出一种我最终选择的方案,用CSR格式,技术方案是用scipy.sparse。

简单说说CSR,CSR格式实际上就是用一个三元组数组来表示这个稀疏矩阵,三元组分别表示(col,row,data),即行,列,数值,非零的位置的数值得以保留,然后其他位置都是0。

存储

首先来看,用旧方案和新方案的数据结构:

旧数据:纯onehot,每行接近1w个数据
新数据:分为两块存储,X部分用稀疏矩阵,然后用npz存储,Y部分用one-hot存储。
0.0 0.0 1.0 0.0 0.0 23,1.0 3737,1.0 3750,1.0 3757,1.0 6249,1.0 9801,1.0 9806,1.0 9819,1.0

存储完,体积只有59M,这样读取到内存的体积也会小很多,甚至整块都读进去都没问题。

这里会用到稀疏矩阵的2个函数,此处导入:

from scipy.sparse import csr_matrix, save_npz

下面来看看怎么实现的,下面是根据之前上游合并好的数据,利用加载得到的oh_encoder,分别进行转化,分为两块输出,一方面是x特征数据矩阵,另一方面是y标签one-hot矩阵。

def gen_res(source_data, oh_encoder):
    col_all = []
    row_all = []
    data_all = []
    idx = 0
    y_res = []
    with open(source_data, encoding="utf8") as f:
        for line in f:
            if idx == 0:
                idx = 1
                continue
            ll = line.strip().split("::")
            ll = line.strip().split("::")
            data_item = []
            scores_item = []
            scores_item = scores_item + oh_encoder["scores"].transform([[ll[0]]])[0].tolist()
            data_item = data_item + oh_encoder["movie_id"].transform([[ll[1]]])[0].tolist()
            data_item = data_item + oh_encoder["movie_year"].transform([[get_year(ll[2])]])[0].tolist()
            data_item = data_item + get_movie_type_oh(ll[3], oh_encoder["movie_type"]).tolist()
            data_item = data_item + oh_encoder["user_id"].transform([[ll[4]]])[0].tolist()
            data_item = data_item + oh_encoder["user_gentle"].transform([[ll[5]]])[0].tolist()
            data_item = data_item + oh_encoder["user_age"].transform([[ll[6]]])[0].tolist()
            data_item = data_item + oh_encoder["user_occupation"].transform([[ll[7]]])[0].tolist()
            # Y处理
            y_res.append(scores_item)
            # X处理
            col, data = sparse_list(data_item)
            col_all = col_all + col
            row_all = row_all + [idx - 1 for item in range(len(col))]
            data_all = data_all + data
            idx = idx + 1
            if idx % 10000 == 0:
                print("generating %s data items" % (idx))
                # break
    x_res = csr_matrix((data_all, (row_all, col_all)), shape=(max(row_all) + 1, 9831))
    return x_res, y_res

总结这么几个点:

  • scores数据我还是保留着one-hot,存储差别不是很大,后面的代码也不用改太多,条件允许的话偷个懒吧
  • 其他的onehot数据,我是先用transform计算出来,进行组合,然后再用sparse_list转化,此处,只需要抽取col和data,存储即可(row因为没必要存,行数后面取可以算出来)
  • csr_matrix是一个转化为稀疏矩阵的函数
  • 9831是除了特征维度,此处规范化,避免特征出现个数不足的现象
  • 有关稀疏函数在这块的应用,非常建议大家多看看API文档,传送门在这里:https://docs.scipy.org/doc/scipy/reference/generated/scipy.sparse.csr_matrix.html
  • 这块的时间其实挺长的,毕竟涉及大量的处理,有条件的各位可以考虑用mapreduce分布式尝试

下面给出sparse_list的定义,这块的任务是对原来的向量,转为用[col,data]存储的形式:

def sparse_list(array_get):
    col = []
    data = []
    for idx in range(len(array_get)):
        if array_get[idx] != 0:
            col.append(idx)
            data.append(array_get[idx])
    return col, data

在gen_res得到结果后,就可以进行进一步存储,下面给出一个训练数据方面的例子。

# 训练数据生成
print("generating training data")
x_train, y_train = gen_res(TRAIN_DATA_PATH, oh_encoder)
with open(GEN_DATA_TRAIN_Y_PATH, "w", encoding='utf8') as f:
    for item in y_train:
        f.write("%s\n" % (",".join([str(i) for i in item])))
save_npz(GEN_DATA_TRAIN_X_PATH, x_train)
print("training data generation done")

大写基本是写死的路径和参数,oh_encoder是使用的转化器,save_npz是存储稀疏矩阵的函数。

读取

这块本来是在后续机器学习模型那集才会说的,此处为了内容完善写出来提早放出来~下一集这个函数就不单独放出来,直接调用啦。

首先是看这里需要的重要包。

from scipy.sparse import load_npz

load_npz是对应稀疏函数的加载包。

def load_dataset(x_path, y_path):
    y_ = []
    y_oh = []
    idx = 0
    with open(y_path, encoding="utf8") as f:
        for line in f:
            ll = line.strip().split(",")
            y_item = max([idx * int(float(ll[idx])) for idx in range(5)]) + 1
            y_oh_item = [int(float(item)) for item in ll[:5]]
            y_.append(y_item)
            y_oh.append(y_oh_item)
            idx = idx + 1
            if idx % 10000 == 0:
                print("loading %s" % idx)
    x_ = load_npz(x_path)
    return x_, y_, y_oh

同样简单粗暴地给关键点吧

  • 此处要用scipy给的数据结构来存
    • 原因是scipy.sparse这个数据类型能够直接放在sklearn下的机器学习模型训练输入中
    • 存取比较直接,不用关心内部逻辑,复杂度一般不会太高
  • x和y分别取出分别放好,毕竟后续训练的时候也是要分开放入的
  • 此处的输出我分了三个,作为稀疏矩阵的特征,数值型的y,和one-hot的y,日后结果评估会有用。

小结

算法从来不止是机器学习,而是对整个项目流程的把握,根据特定目标做好规划,一整个项目才得以完成。本文主要以movielens为例谈论数据集存取策略的问题,是特征工程后确定数据存储格式、存取方式的重要一步,合理高效的存取策略会令整个系统在进行模型计算时更加高效,省时省空间。

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

推荐阅读更多精彩内容