DeepFM

2021-03-21  本文已影响0人  许志辉Albert

1.动机
对于CTR问题,被证明的最有效的提升任务表现的策略是特征组合,在CTR问题的探究历史上来看就是如何更好地学习特征组合,进而更加精准地描述数据的特点。可以说这是基础推荐模型到深度学习推荐模型遵循的一个主要思想。而特征组合大牛们通过组合二阶特征,三阶甚至更高阶,但是面临一个问题就是随着阶数的提升,复杂度就呈几何倍的升高。这样即使模型的表现更好了,但是推荐系统在实时性要求也不能满足了。所以很多模型的出现都是为了解决另一个更加深入的问题:如何更高效的学习特征组合?

为了解决上述问题,出现了FM与FFM来优化LR特征组合较差的一个问题。并且在这个时候科学家们已经发现了DNN在特征组合方面的优势,所以又出现了FNN和PNN等使用深度神经网络的模型。但是DNN也存在局限性。

此时通过增加全连接层就可以实现高阶的特征组合,如下图所示:


3

但是仍然缺少低阶组合,于是增加FM来表示低阶组合。

如上图所示,该模型仍然存在问题:在Output Units阶段直接将低阶和高阶特征进行组合,很容易让模型最终偏向学习到低阶或者高阶特征,而不能做到很好的结合。

综上所述,DeepFM模型横空出世

2.模型结构与原理


6

前面的Field和Embedding处理的是和前面的方法相同,如上图所示绿色部分;DeepFM将Wide部分替换为了FM layer如上图所示蓝色部分,这幅图其实有很多需要注意的地方

2.1 FM
下图是一个FM的结构图,从图中大致可以看出FM Layer是由一阶特征和二阶Concatenate到一起再经过一个Sigmoid得到logits,所以在实现的时候需要单独考虑Linear部分和FM交叉特征部分。
\hat{y}{FM}(x) = w_0+\sum{i=1}^N w_ix_i + \sum_{i=1}^N \sum_{j=i+1}^N v_i^T v_j x_ix_j

7

2.2 Deep


8

Deep Model是为了学习高阶的特征组合,在上图中使用全连接的方式将Dense Embedding输入到Hidden Layer,这里面Dense Embeddings就是为了解决DNN中参数爆炸的问题,这也是推荐系统中常用的处理方式。

Embedding层的输出是将所有id类特征对应的Embedding向量concat到一起输入到DNN中。其中v_i表示第i个field的embedding,m是field的数量。 z_1=[v_1, v_2, ..., v_m] 上一层的输出作为下一层的输入,我们得到: z_L=\sigma(W_{L-1} z_{L-1}+b_{L-1}) 其中\sigma表示激活函数,z, W, b分别表示该层的输入、权重和偏置。
最后进入DNN部分输出使用sigmod激活函数进行激活: y_{DNN}=\sigma(W^{L}a^L+b^L)

3.代码实现
DeepFM在模型的结构图中显示,模型大致由两部分组成,一部分是FM,还有一部分就是DNN, 而FM又由一阶特征部分与二阶特征交叉部分组成,所以可以将整个模型拆成三部分,分别是一阶特征处理linear部分,二阶特征交叉FM以及DNN的高阶特征交叉。在下面的代码中也能够清晰的看到这个结构。此外每一部分可能由是由不同的特征组成,所以在构建模型的时候需要分别对这三部分输入的特征进行选择。

def DeepFM(linear_feature_columns, dnn_feature_columns):
    # 构建输入层,即所有特征对应的Input()层,这里使用字典的形式返回,方便后续构建模型
    dense_input_dict, sparse_input_dict = build_input_layers(linear_feature_columns + dnn_feature_columns)

    # 将linear部分的特征中sparse特征筛选出来,后面用来做1维的embedding
    linear_sparse_feature_columns = list(filter(lambda x: isinstance(x, SparseFeat), linear_feature_columns))

    # 构建模型的输入层,模型的输入层不能是字典的形式,应该将字典的形式转换成列表的形式
    # 注意:这里实际的输入与Input()层的对应,是通过模型输入时候的字典数据的key与对应name的Input层
    input_layers = list(dense_input_dict.values()) + list(sparse_input_dict.values())

    # linear_logits由两部分组成,分别是dense特征的logits和sparse特征的logits
    linear_logits = get_linear_logits(dense_input_dict, sparse_input_dict, linear_sparse_feature_columns)

    # 构建维度为k的embedding层,这里使用字典的形式返回,方便后面搭建模型
    # embedding层用户构建FM交叉部分和DNN的输入部分
    embedding_layers = build_embedding_layers(dnn_feature_columns, sparse_input_dict, is_linear=False)

    # 将输入到dnn中的所有sparse特征筛选出来
    dnn_sparse_feature_columns = list(filter(lambda x: isinstance(x, SparseFeat), dnn_feature_columns))

    fm_logits = get_fm_logits(sparse_input_dict, dnn_sparse_feature_columns, embedding_layers) # 只考虑二阶项

    # 将所有的Embedding都拼起来,一起输入到dnn中
    dnn_logits = get_dnn_logits(sparse_input_dict, dnn_sparse_feature_columns, embedding_layers)
    
    # 将linear,FM,dnn的logits相加作为最终的logits
    output_logits = Add()([linear_logits, fm_logits, dnn_logits])

    # 这里的激活函数使用sigmoid
    output_layers = Activation("sigmoid")(output_logits)

    model = Model(input_layers, output_layers)
    return model
9
10
上一篇下一篇

猜你喜欢

热点阅读