python 人岗推荐论文,SHPJF模型代码,人岗推荐思路和实践

2024-03-28 21:44

本文主要是介绍python 人岗推荐论文,SHPJF模型代码,人岗推荐思路和实践,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.《Leveraging Search History for Improving Person-Job Fit》论文精讲:

摘要不多说,自己看。

文章的核心就是思路

(1)首先把简历(resume) 和工作简介(job desc)用bert进行训练,请看公式一(竟然用bert的下一句预测思路去训练,而不是sbert,文中说单个模型比双塔模型要好一些,个人持怀疑态度),输出结果Ot,后面备用;

(2)对历史的qurey记录和工作、候选者的id进行id embedding,分别得到hj(job的id embedding结果)、hu(candidate的输出)、Hj(job history的输出),最后输出为Q、K、V,最后计算attention,输出ej,公式2-8;

(3)对history query进行word embedding,然后平均池化,输出为HQ,对job desc进行word embedding输出为htj,作为Q,然后HQ和Hj作用输出CQ作为K,CJ-hat作为V,然后计算attention输出eQ;公式9-14

(4)然后ej和eQ,通过e˜ = λeJ + (1 − λ)eQ组合;再把前面的结果进行拼接(oI = MLP [e˜; hj ; e˜ − hj ; e˜ ◦ hj ],由原向量和求差值和乘积),得到oI,公式15-16;

(5)hj 和hu乘积,得到s-match;

(6)最后三部分向量拼接,经过全连接MLP,最后由sigmoid输出为概率值;yˆu,j = σ(MLP [oT ; oI ; smatch] ),公式17

(7)损失函数为:简单的交叉熵损失函数。

 

 

 

 2. 模型代码

代码乱糟糟的,无力吐槽

SHPJF_model

import torch
import torch.nn as nnfrom model.abstract import PJFModel
from model.layer import MLPLayersclass SHPJF(PJFModel):def __init__(self, config, pool):super(SHPJF, self).__init__(config, pool)self.wd_embedding_size = config['wd_embedding_size']self.user_embedding_size = config['user_embedding_size']self.bert_embedding_size = config['bert_embedding_size']self.hd_size = config['hidden_size']self.dropout = config['dropout']self.num_heads = config['num_heads']self.query_his_len = config['query_his_len']self.max_job_longsent_len = config['job_longsent_len']self.pretrained_mf_path = config['pretrained_mf_path']self.beta = config['beta']self.k = config['k']self.emb = nn.Embedding(pool.wd_num, self.wd_embedding_size, padding_idx=0)self.geek_emb = nn.Embedding(self.geek_num, self.user_embedding_size, padding_idx=0)nn.init.xavier_normal_(self.geek_emb.weight.data)self.job_emb = nn.Embedding(self.job_num, self.user_embedding_size, padding_idx=0)nn.init.xavier_normal_(self.job_emb.weight.data)self.text_matching_fc = nn.Linear(self.bert_embedding_size, self.hd_size)self.pos_enc = nn.parameter.Parameter(torch.rand(1, self.query_his_len, self.user_embedding_size), requires_grad=True)self.q_pos_enc = nn.parameter.Parameter(torch.rand(1, self.query_his_len, self.user_embedding_size), requires_grad=True)self.job_desc_attn_layer = nn.Linear(self.wd_embedding_size, 1)self.wq = nn.Linear(self.wd_embedding_size, self.user_embedding_size, bias=False)self.text_based_lfc = nn.Linear(self.query_his_len, self.k, bias=False)self.job_emb_lfc = nn.Linear(self.query_his_len, self.k, bias=False)self.text_based_attn_layer = nn.MultiheadAttention(embed_dim=self.user_embedding_size,num_heads=self.num_heads,dropout=self.dropout,bias=False)self.text_based_im_fc = nn.Linear(self.user_embedding_size, self.user_embedding_size)self.job_emb_attn_layer = nn.MultiheadAttention(embed_dim=self.user_embedding_size,num_heads=self.num_heads,dropout=self.dropout,bias=False)self.job_emb_im_fc = nn.Linear(self.user_embedding_size, self.user_embedding_size)self.intent_fusion = MLPLayers(layers=[self.user_embedding_size * 4, self.hd_size, 1],dropout=self.dropout,activation='tanh')self.pre_mlp = MLPLayers(layers=[self.hd_size \+ 1 \+ 1 \, self.hd_size, 1],dropout=self.dropout,activation='tanh')self.sigmoid = nn.Sigmoid()self.loss = nn.BCEWithLogitsLoss(pos_weight=torch.FloatTensor([config['pos_weight']]))def _text_matching_layer(self, interaction):x = bert_vec = interaction['bert_vec']                      # (B, bertD)x = self.text_matching_fc(bert_vec)                         # (B, wordD)return xdef _intent_modeling_layer(self, interaction):job_longsent = interaction['job_longsent']job_longsent_len = interaction['job_longsent_len']job_desc_vec = self.emb(job_longsent)                   # (B, L, wordD)job_desc_mask = torch.arange(self.max_job_longsent_len, device=job_desc_vec.device) \.expand(len(job_longsent_len), self.max_job_longsent_len) \>= job_longsent_len.unsqueeze(1)job_desc_attn_weight = self.job_desc_attn_layer(job_desc_vec)job_desc_attn_weight = torch.masked_fill(job_desc_attn_weight, job_desc_mask.unsqueeze(-1), -10000)job_desc_attn_weight = torch.softmax(job_desc_attn_weight, dim=1)job_desc_vec = torch.sum(job_desc_attn_weight * job_desc_vec, dim=1)job_desc_vec = self.wq(job_desc_vec)                    # (B, idD)job_id = interaction['job_id']                          # (B)job_id_vec = self.job_emb(job_id)                       # (B, idD)job_his = interaction['job_his']                        # (B, Q)job_his_vec = self.job_emb(job_his)                     # (B, Q, idD)job_his_vec = job_his_vec + self.pos_encqwd_his = interaction['qwd_his']                        # (B, Q, W)qlen_his = interaction['qlen_his']                      # (B, Q)qwd_his_vec = self.emb(qwd_his)                         # (B, Q, W, wordD)qwd_his_vec = torch.sum(qwd_his_vec, dim=2) / \qlen_his.unsqueeze(-1)                    # (B, Q, wordD)qwd_his_vec = self.wq(qwd_his_vec)                      # (B, Q, idD)qwd_his_vec = self.q_pos_enc + qwd_his_vecproj_qwd_his_vec = self.text_based_lfc(qwd_his_vec.transpose(2, 1)).transpose(2, 1) * self.k / self.query_his_len# (B, K, idD)proj_job_his_vec = self.job_emb_lfc(job_his_vec.transpose(2, 1)).transpose(2, 1) * self.k / self.query_his_len# (B, K, idD)text_based_intent_vec, _ = self.text_based_attn_layer(query=job_desc_vec.unsqueeze(0),key=proj_qwd_his_vec.transpose(1, 0),value=proj_job_his_vec.transpose(1, 0))text_based_intent_vec = text_based_intent_vec.squeeze(0)# (B, idD)text_based_intent_vec = self.text_based_im_fc(text_based_intent_vec)job_emb_intent_vec, _ = self.job_emb_attn_layer(query=job_id_vec.unsqueeze(0),key=proj_job_his_vec.transpose(1, 0),value=proj_job_his_vec.transpose(1, 0),)job_emb_intent_vec = job_emb_intent_vec.squeeze(0)      # (B, idD)job_emb_intent_vec = self.job_emb_im_fc(job_emb_intent_vec)intent_vec = (1 - self.beta) * text_based_intent_vec + self.beta * job_emb_intent_vecintent_modeling_vec = self.intent_fusion(torch.cat([job_id_vec, intent_vec, job_id_vec - intent_vec, job_id_vec * intent_vec], dim=1))return intent_modeling_vecdef _mf_layer(self, interaction):geek_id = interaction['geek_id']job_id = interaction['job_id']geek_vec = self.geek_emb(geek_id)job_vec = self.job_emb(job_id)x = torch.sum(torch.mul(geek_vec, job_vec), dim=1, keepdim=True)return xdef predict_layer(self, vecs):x = torch.cat(vecs, dim=-1)score = self.pre_mlp(x).squeeze(-1)return scoredef forward(self, interaction):text_matching_vec = self._text_matching_layer(interaction)intent_modeling_vec = self._intent_modeling_layer(interaction)mf_vec = self._mf_layer(interaction)score = self.predict_layer([text_matching_vec, intent_modeling_vec, mf_vec])return scoredef calculate_loss(self, interaction):label = interaction['label']output = self.forward(interaction)return self.loss(output, label)def predict(self, interaction):return self.sigmoid(self.forward(interaction))

abstract.py

rom logging import getLogger
import numpy as np
import torch.nn as nnclass PJFModel(nn.Module):r"""Base class for all Person-Job Fit models"""def __init__(self, config, pool):super(PJFModel, self).__init__()self.logger = getLogger()self.device = config['device']self.geek_num = pool.geek_numself.job_num = pool.job_numdef calculate_loss(self, interaction):"""Calculate the training loss for a batch data.Args:interaction (dict): Interaction class of the batch.Returns:torch.Tensor: Training loss, shape: []"""raise NotImplementedErrordef predict(self, interaction):"""Predict the scores between users and items.Args:interaction (dict): Interaction class of the batch.Returns:torch.Tensor: Predicted scores for given users and items, shape: [batch_size]"""raise NotImplementedErrordef __str__(self):"""Model prints with number of trainable parameters"""model_parameters = filter(lambda p: p.requires_grad, self.parameters())params = sum([np.prod(p.size()) for p in model_parameters])return super(PJFModel, self).__str__() + '\n\tTrainable parameters: {}'.format(params)

layer.py

import torch
import torch.nn as nn
from torch.nn.init import normal_class MLPLayers(nn.Module):""" MLPLayersArgs:- layers(list): a list contains the size of each layer in mlp layers- dropout(float): probability of an element to be zeroed. Default: 0- activation(str): activation function after each layer in mlp layers. Default: 'relu'.candidates: 'sigmoid', 'tanh', 'relu', 'leekyrelu', 'none'Shape:- Input: (:math:`N`, \*, :math:`H_{in}`) where \* means any number of additional dimensions:math:`H_{in}` must equal to the first value in `layers`- Output: (:math:`N`, \*, :math:`H_{out}`) where :math:`H_{out}` equals to the last value in `layers`Examples:>>> m = MLPLayers([64, 32, 16], 0.2, 'relu')>>> input = torch.randn(128, 64)>>> output = m(input)>>> print(output.size())>>> torch.Size([128, 16])"""def __init__(self, layers, dropout=0., activation='relu', bn=False, init_method=None):super(MLPLayers, self).__init__()self.layers = layersself.dropout = dropoutself.activation = activationself.use_bn = bnself.init_method = init_methodmlp_modules = []for idx, (input_size, output_size) in enumerate(zip(self.layers[:-1], self.layers[1:])):mlp_modules.append(nn.Dropout(p=self.dropout))mlp_modules.append(nn.Linear(input_size, output_size))if self.use_bn:mlp_modules.append(nn.BatchNorm1d(num_features=output_size))activation_func = activation_layer(self.activation, output_size)if activation_func is not None:mlp_modules.append(activation_func)self.mlp_layers = nn.Sequential(*mlp_modules)if self.init_method is not None:self.apply(self.init_weights)def init_weights(self, module):# We just initialize the module with normal distribution as the paper saidif isinstance(module, nn.Linear):if self.init_method == 'norm':normal_(module.weight.data, 0, 0.01)if module.bias is not None:module.bias.data.fill_(0.0)def forward(self, input_feature):return self.mlp_layers(input_feature)def activation_layer(activation_name='relu', emb_dim=None):"""Construct activation layersArgs:activation_name: str, name of activation functionemb_dim: int, used for Dice activationReturn:activation: activation layer"""if activation_name is None:activation = Noneelif isinstance(activation_name, str):if activation_name.lower() == 'sigmoid':activation = nn.Sigmoid()elif activation_name.lower() == 'tanh':activation = nn.Tanh()elif activation_name.lower() == 'relu':activation = nn.ReLU()elif activation_name.lower() == 'leakyrelu':activation = nn.LeakyReLU()elif activation_name.lower() == 'none':activation = Noneelif issubclass(activation_name, nn.Module):activation = activation_name()else:raise NotImplementedError("activation function {} is not implemented".format(activation_name))return activationclass SimpleFusionLayer(nn.Module):def __init__(self, hd_size):super(SimpleFusionLayer, self).__init__()self.fc = nn.Linear(hd_size * 4, hd_size)def forward(self, a, b):assert a.shape == b.shapex = torch.cat([a, b, a * b, a - b], dim=-1)x = self.fc(x)x = torch.tanh(x)return xclass FusionLayer(nn.Module):def __init__(self, hd_size):super(FusionLayer, self).__init__()self.m = SimpleFusionLayer(hd_size)self.g = nn.Sequential(nn.Linear(hd_size * 2, 1),nn.Sigmoid())def _single_layer(self, a, b):ma = self.m(a, b)x = torch.cat([a, b], dim=-1)ga = self.g(x)return ga * ma + (1 - ga) * adef forward(self, a, b):assert a.shape == b.shapea = self._single_layer(a, b)b = self._single_layer(b, a)return torch.cat([a, b], dim=-1)

这篇关于python 人岗推荐论文,SHPJF模型代码,人岗推荐思路和实践的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/856768

相关文章

SpringBoot基于MyBatis-Plus实现Lambda Query查询的示例代码

《SpringBoot基于MyBatis-Plus实现LambdaQuery查询的示例代码》MyBatis-Plus是MyBatis的增强工具,简化了数据库操作,并提高了开发效率,它提供了多种查询方... 目录引言基础环境配置依赖配置(Maven)application.yml 配置表结构设计demo_st

使用Python绘制蛇年春节祝福艺术图

《使用Python绘制蛇年春节祝福艺术图》:本文主要介绍如何使用Python的Matplotlib库绘制一幅富有创意的“蛇年有福”艺术图,这幅图结合了数字,蛇形,花朵等装饰,需要的可以参考下... 目录1. 绘图的基本概念2. 准备工作3. 实现代码解析3.1 设置绘图画布3.2 绘制数字“2025”3.3

python使用watchdog实现文件资源监控

《python使用watchdog实现文件资源监控》watchdog支持跨平台文件资源监控,可以检测指定文件夹下文件及文件夹变动,下面我们来看看Python如何使用watchdog实现文件资源监控吧... python文件监控库watchdogs简介随着Python在各种应用领域中的广泛使用,其生态环境也

Python中构建终端应用界面利器Blessed模块的使用

《Python中构建终端应用界面利器Blessed模块的使用》Blessed库作为一个轻量级且功能强大的解决方案,开始在开发者中赢得口碑,今天,我们就一起来探索一下它是如何让终端UI开发变得轻松而高... 目录一、安装与配置:简单、快速、无障碍二、基本功能:从彩色文本到动态交互1. 显示基本内容2. 创建链

SpringCloud集成AlloyDB的示例代码

《SpringCloud集成AlloyDB的示例代码》AlloyDB是GoogleCloud提供的一种高度可扩展、强性能的关系型数据库服务,它兼容PostgreSQL,并提供了更快的查询性能... 目录1.AlloyDBjavascript是什么?AlloyDB 的工作原理2.搭建测试环境3.代码工程1.

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

python 字典d[k]中key不存在的解决方案

《python字典d[k]中key不存在的解决方案》本文主要介绍了在Python中处理字典键不存在时获取默认值的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录defaultdict:处理找不到的键的一个选择特殊方法__missing__有时候为了方便起见,

Java中ArrayList的8种浅拷贝方式示例代码

《Java中ArrayList的8种浅拷贝方式示例代码》:本文主要介绍Java中ArrayList的8种浅拷贝方式的相关资料,讲解了Java中ArrayList的浅拷贝概念,并详细分享了八种实现浅... 目录引言什么是浅拷贝?ArrayList 浅拷贝的重要性方法一:使用构造函数方法二:使用 addAll(

使用Python绘制可爱的招财猫

《使用Python绘制可爱的招财猫》招财猫,也被称为“幸运猫”,是一种象征财富和好运的吉祥物,经常出现在亚洲文化的商店、餐厅和家庭中,今天,我将带你用Python和matplotlib库从零开始绘制一... 目录1. 为什么选择用 python 绘制?2. 绘图的基本概念3. 实现代码解析3.1 设置绘图画

Python pyinstaller实现图形化打包工具

《Pythonpyinstaller实现图形化打包工具》:本文主要介绍一个使用PythonPYQT5制作的关于pyinstaller打包工具,代替传统的cmd黑窗口模式打包页面,实现更快捷方便的... 目录1.简介2.运行效果3.相关源码1.简介一个使用python PYQT5制作的关于pyinstall