Якщо я вас правильно зрозумів, ви хочете помилитися з боку переоцінки. Якщо це так, вам потрібна відповідна асиметрична функція витрат. Один з простих кандидатів - це виправити втрати в квадраті:
L:(x,α)→x2(sgnx+α)2
де - параметр, який можна використовувати для вимкнення штрафу заниження і завищення. Позитивні значення штрафують завищення, тому вам потрібно встановити негатив. У python це виглядає такα α−1<α<1ααdef loss(x, a): return x**2 * (numpy.sign(x) + a)**2
Далі генеруємо деякі дані:
import numpy
x = numpy.arange(-10, 10, 0.1)
y = -0.1*x**2 + x + numpy.sin(x) + 0.1*numpy.random.randn(len(x))
Нарешті, ми зробимо наш регрес у tensorflow
бібліотеці машинного навчання від Google, яка підтримує автоматизовану диференціацію (спрощуючи оптимізацію таких проблем на градієнті). Я буду використовувати цей приклад як вихідний пункт.
import tensorflow as tf
X = tf.placeholder("float") # create symbolic variables
Y = tf.placeholder("float")
w = tf.Variable(0.0, name="coeff")
b = tf.Variable(0.0, name="offset")
y_model = tf.mul(X, w) + b
cost = tf.pow(y_model-Y, 2) # use sqr error for cost function
def acost(a): return tf.pow(y_model-Y, 2) * tf.pow(tf.sign(y_model-Y) + a, 2)
train_op = tf.train.AdamOptimizer().minimize(cost)
train_op2 = tf.train.AdamOptimizer().minimize(acost(-0.5))
sess = tf.Session()
init = tf.initialize_all_variables()
sess.run(init)
for i in range(100):
for (xi, yi) in zip(x, y):
# sess.run(train_op, feed_dict={X: xi, Y: yi})
sess.run(train_op2, feed_dict={X: xi, Y: yi})
print(sess.run(w), sess.run(b))
cost
є регулярною помилкою у квадраті, тоді acost
як згадана асиметрична функція втрат.
Якщо ви використовуєте, cost
ви отримуєте
1.00764 -3.32445
Якщо ви використовуєте, acost
ви отримуєте
1.02604 -1.07742
acost
явно намагається не недооцінювати. Я не перевіряв конвергенцію, але ви розумієте.