Tentei definir uma função métrica personalizada (F1-Score) em Keras (back-end do Tensorflow) de acordo com o seguinte:
def f1_score(tags, predicted):
tags = set(tags)
predicted = set(predicted)
tp = len(tags & predicted)
fp = len(predicted) - tp
fn = len(tags) - tp
if tp>0:
precision=float(tp)/(tp+fp)
recall=float(tp)/(tp+fn)
return 2*((precision*recall)/(precision+recall))
else:
return 0
Até agora, tudo bem, mas quando tento aplicá-lo na compilação do modelo:
model1.compile(loss="binary_crossentropy", optimizer=Adam(), metrics=[f1_score])
dá erro:
TypeError Traceback (most recent call last)
<ipython-input-85-4eca4def003f> in <module>()
5 model1.add(Dense(output_dim=10, activation="sigmoid"))
6
----> 7 model1.compile(loss="binary_crossentropy", optimizer=Adam(), metrics=[f1_score])
8
9 h=model1.fit(X_train, Y_train, batch_size=500, nb_epoch=5, verbose=True, validation_split=0.1)
/home/buda/anaconda2/lib/python2.7/site-packages/keras/models.pyc in compile(self, optimizer, loss, metrics, sample_weight_mode, **kwargs)
522 metrics=metrics,
523 sample_weight_mode=sample_weight_mode,
--> 524 **kwargs)
525 self.optimizer = self.model.optimizer
526 self.loss = self.model.loss
/home/buda/anaconda2/lib/python2.7/site-packages/keras/engine/training.pyc in compile(self, optimizer, loss, metrics, loss_weights, sample_weight_mode, **kwargs)
664 else:
665 metric_fn = metrics_module.get(metric)
--> 666 self.metrics_tensors.append(metric_fn(y_true, y_pred))
667 if len(self.output_names) == 1:
668 self.metrics_names.append(metric_fn.__name__)
<ipython-input-84-b8a5752b6d55> in f1_score(tags, predicted)
4 #tf.convert_to_tensor(img.eval())
5
----> 6 tags = set(tags)
7 predicted = set(predicted)
8
/home/buda/anaconda2/lib/python2.7/site-packages/tensorflow/python/framework/ops.pyc in __iter__(self)
493 TypeError: when invoked.
494 """
--> 495 raise TypeError("'Tensor' object is not iterable.")
496
497 def __bool__(self):
TypeError: 'Tensor' object is not iterable.
Qual é o problema aqui? O fato de minhas entradas de função f1_score não serem matrizes de Tensorflow? Em caso afirmativo, onde / como posso convertê-los corretamente?
tensorflow
keras
evaluation
Hendrik
fonte
fonte
eval
quando você quer dizereval()
Respostas:
Você precisa usar as funções de back-end do Keras . Infelizmente, eles não suportam o
&
operador, de modo que você precisa criar uma solução alternativa: Geramos matrizes da dimensãobatch_size x 3
, onde (por exemplo, para positivo verdadeiro) a primeira coluna é o vetor de verdade fundamental, a segunda a previsão real e a terceira é tipo de coluna auxiliar de rótulo, que contém apenas os verdadeiros positivos. Depois, verificamos quais instâncias são positivas, são preditas como positivas e o auxiliar de etiqueta também é positivo. Esses são os verdadeiros positivos.Podemos fazer esse análogo com falsos positivos, falsos negativos e verdadeiros negativos com alguns cálculos reversos dos rótulos.
Sua métrica F1 pode ter a seguinte aparência:
Como a calculadora Keras-back-end retorna nan para divisão por zero, não precisamos da instrução if-else para a declaração de retorno.
Edit: Encontrei uma boa idéia para uma implementação exata. O problema com a nossa primeira abordagem é que ela é apenas "aproximada", pois é calculada em lotes e subsequentemente calculada a média. Também se poderia calcular isso após cada época com os
keras.callback
s. Encontre a ideia aqui: https://github.com/fchollet/keras/issues/5794Um exemplo de implementação seria:
Para fazer a rede chamar essa função, basta adicioná-la a retornos de chamada como
Então você pode simplesmente acessar os membros da
metrics
variável.fonte