Hi 游客

更多精彩,请登录!

比特池塘 区块链技术 正文
Kaggle上的Twitter的情感分析竞赛。在这个案例中,将使用预训练的模型BERT来完成对整个竞赛的数据分析。! M: L) j4 R& E+ P$ d' U# o$ ^% d
导入需要的库0 T) \$ L1 m& g: `# E  v& H& @( \
import numpy as np
0 `2 m/ w5 s  u, B) C0 yimport pandas as pd
. m  T  t) p. H! l& u" _from math import ceil, floor& y% B& M' M4 r
import tensorflow as tf
# j: E( u  K! Z" _$ ~- zimport tensorflow.keras.layers as L
4 ^* j* ~' n+ {0 P: K) C. r( Afrom tensorflow.keras.initializers import TruncatedNormal
! _  z7 e" F. V% mfrom sklearn import model_selection
  M$ K! {# h' b" [9 D7 d& Z) gfrom transformers import BertConfig, TFBertPreTrainedModel, TFBertMainLayer$ c- D0 y4 R) P
from tokenizers import BertWordPieceTokenizer4 v* Q& Y; Z2 v+ B1 Y
读取并解释数据5 f! D. F  T* j: O  E$ r" B- ]: c
在竞赛中,对数据的理解是非常关键的。因此我们首先要做的就是读取数据,然后查看数据的内容以及特点。) P8 ?0 x; z- a
先用pandas来读取csv数据,4 @+ y! t$ D: M7 X
7 F  ?7 Q  F& \7 B
train_df = pd.read_csv('train.csv')
! Q; k' a6 r! u$ Ttrain_df.dropna(inplace=True)0 x$ x5 I, F$ v1 o$ J% G* c+ E
test_df = pd.read_csv('test.csv')# E: |1 s3 I5 C0 t: |
test_df.loc[:, "selected_text"] = test_df.text.values
! G5 j* Y; v2 y% V8 ksubmission_df = pd.read_csv('sample_submission.csv')
3 `$ y" z* s, B3 i再查看下我们的数据的数量,我们一共有27485条训练数据,3535条测试数据,6 a# b; {; T* `) h5 B
print("train numbers =", train_df.shape)
, z8 c; E2 S( e; |  uprint("test numbers =", test_df.shape)1 _6 p7 D( c- O% p
紧接着查看训练数据和测试数据前10条表单的字段跟数据,表单中包含了一下几个数据字段:
+ E5 @. i6 H) x% I; O9 H: I% f4 k7 t
textID: 文本数据记录的唯一ID;" F0 ^3 Q+ h7 k2 d

+ k' p+ O3 ]" [6 H# R' s7 Ktext: 原始语句;) Q4 j7 l( q" e. D& F

+ ]0 @, g' [$ G/ e: ^3 bselected_text: 表示情感的语句;
# T: ^# v2 _5 F' F8 j7 Z
8 M8 f1 x2 D6 @( e9 r7 x  Ksentiment: 情感类型, neutral中立, positive积极, negative消极;
$ H3 m$ l' S* B0 P% z& x
! J+ m9 P5 U3 C/ f% u
/ j5 W8 D/ V8 `# c/ K从数据中我们可以得出,目标就是根据现有的情感从原本是的语句中选出能代表这个情感的语句部分。
3 t! l3 K# S, V3 s# ~* A% dtrain_df.head(10)" p& w9 A4 A% b
test_df.head(10)) n$ V1 G9 K, R
定义常量
  t2 o& S; P+ f, B6 K# I# bert预训练权重跟数据存放的目录
5 L2 r3 `+ n) JPATH = "./bert-base-uncased/"9 F( m/ B/ f/ w$ [0 L7 k
# 语句最大长度
) C' ]! O1 p; kMAX_SEQUENCE_LENGTH = 128
8 r* S/ V. h* G' D# M载入词向量9 `5 a+ S4 f9 l( R( T0 H( G
BERT是依据一个固定的词向量来进行训练的。因此在竞赛中需要先使用BertWordPieceTokenizer来加载这些词向量,其中的lowercase=True表示所有的词向量都是小写。设置大小写不敏感可以减少模型对资源的占用。
7 |3 r$ T0 j7 ^/ x3 KTOKENIZER = BertWordPieceTokenizer(f"{PATH}/vocab.txt", lowercase=True)+ U* O4 ?; A4 N; R" ~: |2 S
定义数据加载器4 u5 @) e) l" ?
定义数据预处理函数* W/ E8 V+ u+ x) R1 s9 \
$ Y! M: i2 G! n9 B/ M2 a$ |# j
def preprocess(tweet, selected_text, sentiment):
1 O: r; [6 W5 ?* a: O    : v+ R& d* I- \; T# Y$ w5 W4 m
# 将被转成byte string的原始字符串转成utf-8的字符串
5 Y% L/ v9 t- j5 u% q5 L. _    tweet = tweet.decode('utf-8')
( o4 D1 c# J) c    selected_text = selected_text.decode('utf-8')
/ t" g) J7 r3 L* T    sentiment = sentiment.decode('utf-8')
5 h8 t8 ~& L" D% Z    tweet = " ".join(str(tweet).split())
# O3 P1 r) u) p- H& w& V    selected_text = " ".join(str(selected_text).split())
  S3 w& |2 W3 ]- \0 y0 Y4 U6 {   
( w/ R/ `* N2 ^9 w% n6 |# 标记出selected text和text共有的单词
# H  c, p* p' W8 {1 |) X    idx_start, idx_end = None, None
, R& g; h: L; V" F    for index in (i for i, c in enumerate(tweet) if c == selected_text[0]):# m% j) B1 y4 E, u7 W
        if tweet[index:index+len(selected_text)] == selected_text:  F- c6 X; B. O6 T, V# m. R
            idx_start = index0 g% [& ~9 Q( p- q+ s
            idx_end = index + len(selected_text)
0 |9 D5 `. E9 t! p( A: p3 }            break
! n  g3 W: Z3 E5 G" r0 v! }    intersection = [0] * len(tweet)
8 x0 r; y% ?$ ^. P    if idx_start != None and idx_end != None:$ f; N" I7 j' ?' M" d
        for char_idx in range(idx_start, idx_end):
7 Y  ?* `$ Y' \  f5 j7 J  Q% W            intersection[char_idx] = 1
( Z: M' ?( z4 }8 }1 l4 k9 H4 e  R   
2 P/ W" K& r; a% k  ]9 k' v# 对原始数据用词向量进行编码, 这里会返回原始数据中的词在词向量中的下标
5 Y5 T2 x9 r/ u8 r7 O# 和原始数据中每个词向量的单词在文中的起始位置跟结束位置
, L( t* {% L8 I$ [! k2 f/ y    enc = TOKENIZER.encode(tweet)
# v) p+ ]& y- \9 h: Z    input_ids_orig, offsets = enc.ids, enc.offsets
& j/ p; q$ L$ j) ]0 F1 z+ P2 E% C    target_idx = []
# Q  I( A% m1 @) r& ?( i5 Y    for i, (o1, o2) in enumerate(offsets):
4 _* Y8 e& \9 U8 C  O: A: f- E: s, g        if sum(intersection[o1: o2]) > 0:
" c$ Q8 Q& X9 M/ Q" t% M, m            target_idx.append(i). T* y, p) {8 M0 f) q8 a5 H
    target_start = target_idx[0]6 n$ T: d5 K3 ~: E
    target_end = target_idx[-1]2 j  c9 v2 _2 t$ [% O# n0 s$ _9 ?
    sentiment_map = {) J) t  ~% E% g8 N8 L- R' ?4 X
        'positive': 3893,# `4 F" `: w3 m' c& ^$ h5 ^- g
        'negative': 4997,$ l3 h3 t; B0 l- I: `6 J
        'neutral': 8699,
/ |: p3 q1 w  W' P1 l    }
: n( Z, {  J3 B; ~- ^$ L) @5 z. `% D    + a9 z, T# r% M
# 将情感标签和原始的语句的词向量组合在一起组成我们新的数据
' Q; e, Y# U9 K5 |    input_ids = [101] + [sentiment_map[sentiment]] + [102] + input_ids_orig + [102], ~6 c/ z0 Q9 O  x( p& ]
    input_type_ids = [0] * (len(input_ids_orig) + 4): f* N7 k; }+ Y
    attention_mask = [1] * (len(input_ids_orig) + 4)  ]5 n+ D( s2 {5 e1 F9 {
    offsets = [(0, 0), (0, 0), (0, 0)] + offsets + [(0, 0)]
- D' n: c6 {  F5 i    target_start += 3; O) l0 L8 F* Z5 D2 L
    target_end += 33 ?* M3 X+ j* M0 ?  @- J
# 计算需要paddning的长度, BERT是以固定长度进行输入的,因此对于不足的我们需要做pandding
9 O* V, S: i: A8 v# U    padding_length = MAX_SEQUENCE_LENGTH - len(input_ids)2 V3 ]# r+ [" u# G
    if padding_length > 0:9 r" M8 m: _! c( x
        input_ids = input_ids + ([0] * padding_length)
! y) ?4 c. u3 X( R        attention_mask = attention_mask + ([0] * padding_length)
# s2 q$ i0 O. H8 V        input_type_ids = input_type_ids + ([0] * padding_length): @! c* Z5 a6 W& p2 n( C$ `
        offsets = offsets + ([(0, 0)] * padding_length)- S9 x& _/ H& ~9 j3 A7 y5 ~* }
    elif padding_length
/ V2 y8 W& [& Z4 g* A$ ?2 ?! A6 T定义数据加载器7 ]; x/ G6 \9 z) w% v# m3 Q% s

+ l5 N- k$ ]- d( T5 Dclass TweetDataset(tf.data.Dataset):
$ `0 I% t" M  Q* U, _0 |) s9 M' g    ! l$ A8 P+ g, m( r, O8 x
    outputTypes = (
" T$ Z+ ~; B% W) x        tf.dtypes.int32,  tf.dtypes.int32,   tf.dtypes.int32, * s/ {/ Y2 X4 g) c) J
        tf.dtypes.int32,  tf.dtypes.float32, tf.dtypes.float32,9 @3 ]. D% n8 q) j% H) O
        tf.dtypes.string, tf.dtypes.string,  tf.dtypes.string,2 X9 q2 q$ Q& J
    )
. [+ I8 E" K" U2 S) y      n8 C% q* g! M9 W1 F4 E: H' b
    outputShapes = (  p, t" e# a" r  s' T! Y$ I
        (128,),   (128,), (128,),
$ ^3 v/ Z& ~' L        (128, 2), (),     (),
0 N3 l! D3 T  _; l5 I1 M/ C        (),       (),     (),$ Y9 h5 T9 C+ m
    )
2 o; b3 Y" V; p/ A9 p% g# I   
1 ^; T% b/ [  C    def _generator(tweet, selected_text, sentiment):! p0 b8 c+ N. M8 ?) \1 b
        for tw, st, se in zip(tweet, selected_text, sentiment):0 d5 k2 r* M* G4 c/ a
            yield preprocess(tw, st, se)( X/ \4 N+ i! i9 c5 c/ ^
    % w- A. b* {/ J# C( V, e+ ~8 W
    def __new__(cls, tweet, selected_text, sentiment):
# g4 u7 |/ j  `. |* w, y8 W+ `        return tf.data.Dataset.from_generator(6 x& {, r% X5 Z
            cls._generator,
! W2 M' ~( z9 p" v            output_types=cls.outputTypes,
" C1 N7 M0 y& f! L$ [4 h9 L            output_shapes=cls.outputShapes,
1 G( B7 ]: ~8 ?  X            args=(tweet, selected_text, sentiment)# f- G+ B9 Z( `. z" g6 F1 D
        ), T! w" L$ [  ]/ a# V
    ) y$ a& c* |& S1 A0 b3 |( B
    @staticmethod
0 p: i1 A2 S) `, d    def create(dataframe, batch_size, shuffle_buffer_size=-1):
5 a' [8 M4 s% K( }$ o$ @/ L9 [        dataset = TweetDataset(
) V7 [' `, f2 ^, p/ o' b- z            dataframe.text.values,
, D( E$ O+ f- O; z% x1 Y" L7 {            dataframe.selected_text.values, ' v- j# A! Q# @3 b( \$ P$ a
            dataframe.sentiment.values
, S9 }7 ~0 A. U& k        )
  v0 e9 a$ r+ t2 ]' J5 W        dataset = dataset.cache()
6 U# X' B7 Q- l7 v8 |" ]6 \& K* i7 `+ w        if shuffle_buffer_size != -1:
* a1 ]# o& U/ H. y  h1 `            dataset = dataset.shuffle(shuffle_buffer_size)
" ?. G3 ~& m! {( r2 O% [$ i9 {6 B        dataset = dataset.batch(batch_size)  s# K/ }% o: T+ t
        dataset = dataset.prefetch(tf.data.experimental.AUTOTUNE). w" x7 K7 O6 q2 A- |
        return dataset
& t6 {: v% u6 ^. _9 ]定义模型
. F; ^- m+ U; \: a1 ?' r我们使用BERT模型来进行这次竞赛,这里对BERT模型做一些简单的介绍。
" v* A& e7 j! q& W; W+ HBERT的全称是Bidirectional Encoder Representation from Transformers,即双向Transformer的Encoder,因为decoder是不能获要预测的信息的。% v2 j  _7 T6 z; c/ t1 g6 J
模型的主要创新点都在pre-train方法上,即用了Masked LM和Next Sentence Prediction两种方法分别捕捉词语和句子级别representation。; m: n' B; R7 `9 N) ?- u& I* q( b' M
BERT主要特点如下:
; @8 V% l  @: x: y* e; S! u1 W; x# a5 N9 U+ h
使用了Transformer作为算法的主要框架,Trabsformer能更彻底的捕捉语句中的双向关系;
: s. p& Z+ z3 T$ x. z. J, k; S  \, J
使用了Mask Language Model 和 Next Sentence Prediction的多任务训练目标;
# l: M, \( s! {% K! F  r" k1 w" s8 ~3 |
使用更强大的机器训练更大规模的数据,Google开源了BERT模型,我们可以直接使用BERT作为Word2Vec的转换矩阵并高效的将其应用到自己的任务中。4 E1 T  S. n% `+ t

/ a! k& `0 L5 B, a9 ~! X; J( O0 `* r5 Y  I- [
BERT的本质是在海量的语料基础上,运行自监督学习方法让单词学习得到一个较好的特征表示。
$ I* w- A0 y& w! d* j+ b在之后特定任务中,可以直接使用BERT的特征表示作为该任务的词嵌入特征。所以BERT提供的是一个供其它任务迁移学习的模型,该模型可以根据任务微调或者固定之后作为特征提取器。# I5 n' A" Z* S& n
在竞赛中,我们定义了一个BertModel类,里面使用TFBertPreTrainedModel来进行推理。
) x5 o5 n, I+ j, e, @; D, @, K! \BERT的输出我们保存在hidden_states中,然后将这个得到的hidden_states结果在加入到Dense Layer,最后输出我们需要提取的表示情感的文字的起始位置跟结束位置。
" v; R  T# \  l" A2 b( o" L这两个位置信息就是我们需要从原文中提取的词向量的位置。0 K# r! l' s6 }! k" v- J6 I
3 ~4 v9 h: ~0 X" n9 q* ]% n6 v
class BertModel(TFBertPreTrainedModel):
5 C" G) [% B' |: K7 a$ i: k/ u6 `   
# m! `6 s$ J. P  T# drop out rate, 防止过拟合, _9 T' D1 e; F+ R. e3 @5 H
    dr = 0.1
/ Z9 a4 j! s9 K  D+ ^: S# hidden state数量
# L8 k& I9 d( Z0 F: b    hs = 2
4 S& E: Y3 h+ h+ J' s' U2 D   
1 H  E: l' c' k9 ~3 W    def __init__(self, config, *inputs, **kwargs):
* T7 O) f4 c4 p2 ]0 v( t        super().__init__(config, *inputs, **kwargs)
( l( M. W; a/ f( T5 k  j        $ |2 b* [  l' S3 V# t
        self.bert = TFBertMainLayer(config, name="bert")
( ^: }% p5 K6 y& [6 d8 R        self.concat = L.Concatenate()& l* T0 D9 G: @
        self.dropout = L.Dropout(self.dr)4 C6 O$ r; i# w. I+ x  o( [7 D
        self.qa_outputs = L.Dense(4 G+ x4 W# p, Y" w8 r
            config.num_labels,
) @6 Y( J6 C9 j& ~8 V            kernel_initializer=TruncatedNormal(stddev=config.initializer_range),
  ^7 ~7 Y  W# ^% ]* R4 ]& ]( @            dtype='float32',
; ?9 Q  c3 x2 ]& v            name="qa_outputs")
" v9 q2 O1 ~' ~        . [/ [4 _/ [; q: g
    @tf.function
. d" h" n$ N" t    def call(self, inputs, **kwargs):
" ~0 b. A$ j( u, j# o+ ]  {        _, _, hidden_states = self.bert(inputs, **kwargs)4 y1 b; w$ i( a1 p
        
* ~0 Y  T( Q: @4 b- n        hidden_states = self.concat([
% I) G  F8 h3 ]2 C            hidden_states[-i] for i in range(1, self.hs+1)) H( \5 T# v/ H* |  s
        ])
+ K0 e: H+ J1 B7 C1 J; h6 a( Q  Q        
5 i1 R% \# ~# v        hidden_states = self.dropout(hidden_states, training=kwargs.get("training", False))0 z+ v5 i, K* w2 H6 ?2 k4 \" g
        logits = self.qa_outputs(hidden_states)3 [1 y& `5 F# S& k' w% p
        start_logits, end_logits = tf.split(logits, 2, axis=-1)- m4 _+ z! {: [$ J' m
        start_logits = tf.squeeze(start_logits, axis=-1)0 T3 O6 w% h, ]. a
        end_logits = tf.squeeze(end_logits, axis=-1)
) O. _) k2 R( R' h        
( E  U) }8 z( f/ q% j9 ]0 z  i        return start_logits, end_logits
9 }' ?# V. T4 x1 E9 m定义训练函数2 }( I: x$ O9 x
3 h# j; _! j: r
def train(model, dataset, loss_fn, optimizer):- A( b3 f. V0 C/ n6 f. e2 ^
    5 `0 x2 Q# T4 G. `5 T, N8 m
    @tf.function: K" \: j, D6 J5 w! H
    def train_step(model, inputs, y_true, loss_fn, optimizer):
2 P+ r  ?) t: j  B        with tf.GradientTape() as tape:
$ {) R9 S5 G7 `; z  d. F, z' L  X            y_pred = model(inputs, training=True)
9 \* Z: f' Y/ p5 R' q            loss  = loss_fn(y_true[0], y_pred[0])
1 O2 r' V6 t5 L            loss += loss_fn(y_true[1], y_pred[1])
5 _4 Z) G, t' ?9 N( Q* y            scaled_loss = optimizer.get_scaled_loss(loss)+ a4 ?) h, x% {/ B
   
+ _2 k4 b1 z; u6 j        scaled_gradients = tape.gradient(scaled_loss, model.trainable_variables). I( K/ r$ a$ G; }$ U/ y
        gradients = optimizer.get_unscaled_gradients(scaled_gradients)
: i1 G! }# o3 p; H' U# y5 {& p2 ~        optimizer.apply_gradients(zip(gradients, model.trainable_variables))
, h# |% w8 T% Q: D, Y% K8 `2 x        return loss, y_pred
# T7 u2 U+ ~( v3 i6 L6 d! t6 n% v# l    epoch_loss = 0.) M, c& i+ s1 p% i7 y% M
    for batch_num, sample in enumerate(dataset):
( y/ G. ]; t$ o        loss, y_pred = train_step(model, sample[:3], sample[4:6], loss_fn, optimizer)/ f+ b# P6 u# C8 k4 t
        epoch_loss += loss% s9 c9 ~5 @6 S: p
        print(8 F% w4 {9 [. f3 C' i3 ]
            f"training ... batch {batch_num+1:03d} : "- S. G. A8 V6 ]" |
            f"train loss {epoch_loss/(batch_num+1):.3f} ",
$ v! S& e+ {6 D( t            end='\r')
  X; |& T; k1 {7 o1 N定义预制函数0 F5 j( I$ O2 B9 [3 d" w
def predict(model, dataset, loss_fn, optimizer):
$ c' A5 Z6 ?6 K  ?    0 |0 j# m1 T5 M7 m7 t3 O/ e
    @tf.function
9 r# q- j3 @; r    def predict_step(model, inputs):7 l5 l4 o; \" H3 s) v( c! _- d
        return model(inputs)9 n2 o. T+ L* G2 @" x5 L* u' w
        
6 x/ Y' c% q5 Q7 y2 S4 I4 {* E    def to_numpy(*args):5 d7 Y: Z1 p$ V% \+ ?
        out = []
3 f0 `7 K7 X6 n4 V9 I        for arg in args:
3 K' ~* m4 r; }; h! j* Q! D* E5 e            if arg.dtype == tf.string:" c6 a/ T0 M; Q# G  j" E: B- J8 o
                arg = [s.decode('utf-8') for s in arg.numpy()]
& I8 l1 X" ~" z+ L' M& K                out.append(arg)
+ r2 G' c; ]! o4 y) ^            else:
7 I8 Y7 Z- {1 Q2 D/ R+ p                arg = arg.numpy()& F3 ]1 S$ |' D9 {
                out.append(arg)/ @& Z& z, A7 P" c9 }
        return out, N* f3 {( I2 M4 b4 a
   
1 N0 q& |& N& F9 e  @$ C+ f    offset = tf.zeros([0, 128, 2], dtype=tf.dtypes.int32)8 q* J$ y& o; P* i% @3 Q
    text = tf.zeros([0,], dtype=tf.dtypes.string)
8 x4 h7 C, r- B; C/ J" u0 P& K6 D0 v0 v/ ?    selected_text = tf.zeros([0,], dtype=tf.dtypes.string): l* i' e8 V+ C  m$ r  z9 w
    sentiment = tf.zeros([0,], dtype=tf.dtypes.string)
/ L7 h3 H8 F9 U3 Q/ k    pred_start = tf.zeros([0, 128], dtype=tf.dtypes.float32)+ r+ T7 j3 G$ q+ ?% t0 E2 r
    pred_end = tf.zeros([0, 128], dtype=tf.dtypes.float32)" C" T- Y% u( e) {& k' c, P; e
   
/ \- ]  S% [% n( T( G    for batch_num, sample in enumerate(dataset):
( I7 L5 y# K  M3 K- `# }        2 B$ a+ ^! A* v) O0 B9 G
        print(f"predicting ... batch {batch_num+1:03d}"+" "*20, end='\r')+ w- n) l% q! {* b
        
. d5 U% {$ d  V, B8 E        y_pred = predict_step(model, sample[:3])4 [$ L1 h, ?4 y% \
        " x2 q, Y+ D- ?9 [. M# c3 ^
        # add batch to accumulators
- q( [, Q! r9 p) ^* ^$ b7 s" A        pred_start = tf.concat((pred_start, y_pred[0]), axis=0)8 k; m9 a5 j/ ^0 ~4 @4 h; U; Z( y! g
        pred_end = tf.concat((pred_end, y_pred[1]), axis=0)
* M1 K+ U, a& X1 }: J' t        offset = tf.concat((offset, sample[3]), axis=0)
1 r, i$ R" r7 ~1 m# N: I        text = tf.concat((text, sample[6]), axis=0)
" R& u6 ?: S9 M% {" ^        selected_text = tf.concat((selected_text, sample[7]), axis=0)
1 R. g9 j$ g- _( Z' \1 s        sentiment = tf.concat((sentiment, sample[8]), axis=0)
6 a! @; o9 q4 V* @   
8 r& G  X$ v0 D- F+ q8 F# ]    pred_start, pred_end, text, selected_text, sentiment, offset = \
* h: b6 `+ k  n5 p$ X% B/ L( C        to_numpy(pred_start, pred_end, text, selected_text, sentiment, offset)' B+ C9 Z. I! H# M
   
9 j& ?" p  O+ s/ i    return pred_start, pred_end, text, selected_text, sentiment, offset, m0 M* p( t' B. @8 |* C
判断函数
7 D' q  P! |7 J5 S这个竞赛采用单词级Jaccard系数,计算公式如下. O9 Z4 Z3 u7 _, v: ]
$ ?' e1 F4 c. n; F  ?6 ]
Jaccard系数计算的是你预测的单词在数据集中的个数,! c! v0 ~9 n; y  F5 H% r
def jaccard(str1, str2):
; f( }7 v! l& M# a/ e$ o0 [    a = set(str1.lower().split())5 P8 E6 |# t4 P6 L4 Z# [1 Q* N
    b = set(str2.lower().split())
/ C# w; U) j5 g1 O5 Z    c = a.intersection(b)6 l( v& M9 ^, Y/ ~# @3 |# I
    return float(len(c)) / (len(a) + len(b) - len(c))
2 v( A1 s. z$ `  H定义预测结果解码函数
; f! o) ~. k+ W! D解码函数通过模型预测拿到的start和end的index位置信息,然后和之前拿到的词向量在样本句子中的位置进行比较,将这个区间内的所有的单词都提取出来作为我们的预测结果。
8 m) R' b5 m% g1 F( u
% L: r5 m4 K7 u1 H( udef decode_prediction(pred_start, pred_end, text, offset, sentiment):* W, {- K3 S2 \& ^) w9 }! K
    2 Y; |2 S/ D$ [; s
    def decode(pred_start, pred_end, text, offset):$ m, }; b  c/ x0 Q, c6 ~
        decoded_text = ""' [3 m7 v: e% E/ W* U
        for i in range(pred_start, pred_end+1):
6 p1 j, q4 s; T1 [/ t6 c$ o8 s            decoded_text += text[offset[0]:offset[1]]! w* }* P1 u+ I7 A7 n  @  b4 Q% _0 _
            if (i+1)  idx_end:1 C# O( \2 {" G: X, ?' O, ?
                idx_end = idx_start
/ e9 L' n" f( Q- G5 ^3 G            decoded_text = str(decode(idx_start, idx_end, text, offset))3 a. s9 E# V6 x* B( f5 u
            if len(decoded_text) == 0:
  k2 G9 r4 Z* L  ?                decoded_text = text
: w+ C8 @$ a& L7 {  k        decoded_predictions.append(decoded_text)* ]8 l4 h4 Q- J# F, g4 r& k, L
    ' q9 r) D, l3 j4 c. X; G. i
    return decoded_predictions  g( s" T/ @! C! g9 E2 C
开始训练7 n% [! Z  q' Y
将训练数据分成5个folds,每个folds训练5个epoch,使用adam优化器,learning rate设置成3e-5,batch size使用32。1 r7 l  O# }# F9 F
. H  V/ @8 P, J2 C
num_folds = 5
' l5 ~! i9 Y! v) W3 Knum_epochs = 5$ O; c! E; l; ]) \
batch_size = 32' d0 t- D% r/ [. t
learning_rate = 3e-5
. h  P. c1 ~; V8 u0 w8 J  [optimizer =  tf.keras.optimizers.Adam(learning_rate)
% i+ t. e' i5 S6 r+ ?optimizer = tf.keras.mixed_precision.experimental.LossScaleOptimizer(( H  q; j' p9 |4 b! V2 T
    optimizer, 'dynamic')! ^" |% i! U- [8 P. `! t
config = BertConfig(output_hidden_states=True, num_labels=2)
" r6 E! c8 i2 v+ _" x3 [. o# _! [model = BertModel.from_pretrained(PATH, config=config); v6 L9 ]0 _( m1 ?' P( l
loss_fn = tf.keras.losses.SparseCategoricalCrossentropy(from_logits=True)' c( ^0 J' L+ W. _; H4 @7 t
kfold = model_selection.KFold(
. w- q) i9 V- P( B, Z; p$ U9 g    n_splits=num_folds, shuffle=True, random_state=42)2 V& D, i0 M0 n6 I. }- `( d
test_preds_start = np.zeros((len(test_df), 128), dtype=np.float32)
3 F! c* U* B: W% `( q# x( R$ Vtest_preds_end = np.zeros((len(test_df), 128), dtype=np.float32)
: k! c' o$ \' p$ K' M5 {+ @for fold_num, (train_idx, valid_idx) in enumerate(kfold.split(train_df.text)):4 o8 B' [% y4 ~1 v+ Q6 w1 R5 `% T
    print("\nfold %02d" % (fold_num+1))) z$ O. t; \, _; t# n8 W
    , M0 C: R- Q! q' d" N1 _2 v) Z
# 创建train, valid, test数据集" `& Z+ B( K: {: q
    train_dataset = TweetDataset.create(* R: U' d( y; ^$ b; T
        train_df.iloc[train_idx], batch_size, shuffle_buffer_size=2048)
  {# `% |% J# z1 J, \    valid_dataset = TweetDataset.create(
7 U1 V. r, V0 T' U- @        train_df.iloc[valid_idx], batch_size, shuffle_buffer_size=-1)
% J! \# ~* m/ g8 I  q. Z1 A    test_dataset = TweetDataset.create(
5 `! b) |( ?- ^% i/ v        test_df, batch_size, shuffle_buffer_size=-1)% @1 G  E2 K  h$ _
    % T, T$ ?. a# Z  l( q6 m
    best_score = float('-inf')
2 b* b+ U* f# N6 t0 G+ z/ r* Y. d    for epoch_num in range(num_epochs):. u' e) Y, z7 i! b
        print("\nepoch %03d" % (epoch_num+1))
6 z1 o4 y6 u7 c        1 w/ |6 ~/ z7 G* P
        train(model, train_dataset, loss_fn, optimizer)- v) |8 g4 D4 V% w% i5 i8 M4 x
        6 {' N$ @2 @8 w- n
        pred_start, pred_end, text, selected_text, sentiment, offset = \
. h0 S% Y8 d9 M# s  ~            predict(model, valid_dataset, loss_fn, optimizer)/ Y# o- \* `5 J# O- K$ V
        ) E: H. X6 O" c
        selected_text_pred = decode_prediction(
; z4 F# X# d* s( w" R& N            pred_start, pred_end, text, offset, sentiment)0 O6 T, n, ~9 w
        jaccards = []8 T* s8 k1 e8 T, O- l5 |) d
        for i in range(len(selected_text)):
+ S3 b" o5 z3 z. P  d. `            jaccards.append(
  X5 }  V1 T/ f% ~& S                jaccard(selected_text, selected_text_pred))1 s0 ^  `  F* g! h
        & D! y6 a1 h) P$ u0 M. i
        score = np.mean(jaccards)
6 e( R3 O8 A8 G0 U7 V9 C        print(f"valid jaccard epoch {epoch_num+1:03d}: {score}"+" "*15)
1 J& M" q; E+ R) P1 [5 Y        ! p( T1 F/ X! R; e- W
        if score > best_score:. s6 a. O7 o( }, w
            best_score = score5 r7 i) I, V8 H, Y- {2 K
            
( R; u9 {% P0 a: r& y2 }3 @) o% a# predict test set
* z# d* y3 i3 E) D* b+ b8 |            test_pred_start, test_pred_end, test_text, _, test_sentiment, test_offset = \
; F8 ]% z6 w/ c                predict(model, test_dataset, loss_fn, optimizer)
( P, u+ O9 Y% p, t    ' h8 ~7 z8 Q$ Y
    test_preds_start += test_pred_start * 0.2
5 Y+ m9 |0 Q5 n3 s. N    test_preds_end += test_pred_end * 0.21 E5 ^% B; o- ~. r9 {
    / p' P( O- O! q9 p: b0 a
# 重置模型,避免OOM
+ G" s6 b4 _8 z! T    session = tf.compat.v1.get_default_session()3 p  O6 o  `) R& I/ p2 c5 W2 j% q6 t
    graph = tf.compat.v1.get_default_graph()  h9 ]  c$ w- V8 G& t1 g1 |% T
    del session, graph, model
, z1 U: a, z9 s    model = BertModel.from_pretrained(PATH, config=config)1 M9 L, o4 x. D+ l, H. K
预测测试数据,并生成提交文件0 i0 Q! C9 j4 r" d3 z* `4 C
selected_text_pred = decode_prediction(
' E" B3 K1 f! e$ b    test_preds_start, test_preds_end, test_text, test_offset, test_sentiment)
# [( g3 T' n% `: w- cdef f(selected):/ A0 @% G- m! b
    return " ".join(set(selected.lower().split()))
% L1 ]) O& `/ {$ k* M9 z6 N* usubmission_df.loc[:, 'selected_text'] = selected_text_pred2 H9 R) C$ N7 `1 y+ f) @
submission_df['selected_text'] = submission_df['selected_text'].map(f)
* X2 q: q* a! L0 b* G; M; Hsubmission_df.to_csv("submission.csv", index=False): s+ [. _# }) B
这个方案在提交的时候在553个队伍中排名153位, 分数为0.68。8 O6 B5 I: @% z& R
BitMere.com 比特池塘系信息发布平台,比特池塘仅提供信息存储空间服务。
声明:该文观点仅代表作者本人,本文不代表比特池塘立场,且不构成建议,请谨慎对待。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

成为第一个吐槽的人

lf0517 小学生
  • 粉丝

    0

  • 关注

    0

  • 主题

    1