Кусочно-активационная функция в тензорном потоке и математическая операция вещания - PullRequest
0 голосов
/ 15 января 2019

Я пытаюсь реализовать и протестировать функцию активации, которую я прочитал в статье.

Я использую Keras с бэкэндом тензорного потока и хочу передать функцию активации методу подгонки моей модели. Вот математическая форма функции:

Кусочная формула

Я пытался реализовать это двумя способами:

def function_1(x):

    cond1 = tf.greater(x , 2.0)
    cond2 = tf.logical_and(tf.less_equal(x, 2.0), tf.greater_equal(x, 0.0))
    cond3 = tf.logical_and(tf.less(x, 0.0), tf.greater_equal(x, -2.0))
    cond4 = tf.less(x, -2.0)

    y = tf.where(cond1, tf.constant(1.0) , tf.where(cond2,
    x - 0.25*tf.square(x), tf.where(cond3, x + 0.25*tf.square(x), 
    tf.where(cond4, tf.constant(-1.0), tf.constant(-1.0)))))

    return y

def function_2(x):

    cond1 = tf.greater(x , 2.0)
    cond2 = tf.logical_and(tf.less_equal(x, 2.0), tf.greater_equal(x, 0.0))
    cond3 = tf.logical_and(tf.less(x, 0.0), tf.greater_equal(x, -2.0))
    cond4 = tf.less(x, -2.0)

    y = tf.case({cond1: lambda x: tf.constant(1.0), cond2: lambda x: x - 
    0.25*tf.square(x), cond3: lambda x: x + 0.25*tf.square(x),
    cond4: lambda x: tf.constant(-1.0)}, exclusive = True)

    return y

В обоих случаях я получаю одинаковую ошибку:

InvalidArgumentError: Фигуры должны иметь одинаковый ранг, но они равны 0 и 2 для 'density_22 / Select' (op: 'Select') с входными фигурами: [?, 5], [], [].

Как правильно это сделать и что не так с моим кодом?

Ответы [ 2 ]

0 голосов
/ 15 января 2019

Проблема в том, что вы сравниваете тензор с формой [None, 5] (ранг 2) со скейлером (ранг 0), что невозможно при tf.greater или tf.less. Вместо этого вы можете использовать tf.math..., поддерживающий вещание.

Вот одно из возможных решений для реализации этой функции:

import tensorflow as tf

x = tf.placeholder(dtype=tf.float32, shape=[1, 5])

cond1 = tf.cast(tf.math.greater(x, 2.0), tf.float32)
cond2 = tf.cast(tf.math.logical_and(tf.math.less_equal(x, 2.0), tf.math.greater_equal(x, 0.0)), tf.float32)
cond3 = tf.cast(tf.math.logical_and(tf.math.less(x, 0.0), tf.math.greater_equal(x, -2.0)), tf.float32)
cond4 = tf.cast(tf.math.less(x, -2.0), tf.float32)

a = tf.math.multiply(cond1, 1.0)
b = tf.math.multiply(cond2, (x - tf.square(x) / 4))
c = tf.math.multiply(cond3, (x + tf.square(x) / 4))
d = tf.math.multiply(cond4, -1.0)

f = a + b + c + d

with tf.Session() as sess:
    sess.run(tf.global_variables_initializer())
    print(sess.run(f, feed_dict={x: [[-1.0, -5, 1.5, -1.5, 5]]})) 
0 голосов
/ 15 января 2019

Вы можете попробовать слой lambda, который очень удобен для добавления нашего пользовательского слоя или функции в сеть. Вот пример:

dense1 = Dense(dims)(input)   
act1 = Lambda(customFunc, output_shape)(dense1)

def customFunc(x):
    """-----operations----""""
    # final output y of shape defined as above in layer
    y = conditioned-output
    return y

Вот ссылка для получения дополнительной информации info Это еще одна полезная ссылка , поясненная на примере.

...