4

我有一个 Keras 模型,它具有输入 x_1、...、x_n 和 d 维输出 f(x_1)、...、f(x_n)。我正在研究 d 维目标 y_1,...,y_n 的回归问题。

我想最小化损失函数:对于介于 0 和 1 之间的固定元参数 a,返回 的 a^th(经验)分位数|f(x_i)-y_i|^2

到目前为止,这是我编写的代码:

def keras_custom_loss(y_true,y_predicted):
    SEs = K.square(y_true-y_predicted)
    out_custom = tfp.stats.percentile(SEs, 50.0, interpolation='midpoint')
    return out_custom

一个问题是我想避免使用 tensorflow_probability,我想在 Keras 上完成整个实现。

但是,我无法弄清楚如何。

4

2 回答 2

2

要获取高于该百分位数的“所有元素”,您将需要一个不同的答案:

import keras.backend as K
from keras.layers import *
from keras.models import Model
import numpy as np
import tensorflow as tf

def above_percentile(x, p): #assuming the input is flattened: (n,)

    samples = K.cast(K.shape(x)[0], K.floatx()) #batch size
    p =  (100. - p)/100.  #100% will return 0 elements, 0% will return all elements

    #samples to get:
    samples = K.cast(tf.math.floor(p * samples), 'int32')
        #you can choose tf.math.ceil above, it depends on whether you want to
        #include or exclude one element. Suppose you you want 33% top,
        #but it's only possible to get exactly 30% or 40% top:
        #floor will get 30% top and ceil will get 40% top.
        #(exact matches included in both cases)

    #selected samples
    values, indices = tf.math.top_k(x, samples)

    return values

def custom_loss(p):
    def loss(y_true, y_predicted):
        ses = K.square(y_true-y_predicted)
        above = above_percentile(K.flatten(ses), p)
        return K.mean(above)
    return loss

测试:

dataX = np.array([2,3,1,4,7,10,8,5,6]).reshape((-1,1))
dataY = np.ones((9,1))


ins = Input((1,))
outs = Lambda(lambda x: x)(ins)
model = Model(ins, outs)

model.compile(optimizer='adam', loss = custom_loss(70.))
model.fit(dataX, dataY)

损失将65130/2(均值)。和130 = (10-1)² + (8-1)²,是108输入中的前两个 k。

于 2020-03-16T16:50:04.140 回答
2

对于您的特定用例,您可以使用以下功能,它是tfp.stats.percentile(他们使用Apache License 2.0)的简化版本:

import tensorflow as tf

def percentile(x, p):
    with tf.name_scope('percentile'):
        y = tf.transpose(x)  # take percentile over batch dimension
        sorted_y = tf.sort(y)
        frac_idx = tf.cast(p, tf.float64) / 100. * (tf.cast(tf.shape(y)[-1], tf.float64) - 1.)
        return 0.5 * (  # using midpoint rule
            tf.gather(sorted_y, tf.math.ceil(frac_idx), axis=-1)
            + tf.gather(sorted_y, tf.math.floor(frac_idx), axis=-1))
于 2020-03-16T15:49:26.023 回答