2017-04-23 90 views
4

我是Keras的新手,我試圖定義我自己的指標。它計算一致性指數,這是衡量回歸問題的指標。Keras自定義指標迭代

def cindex_score(y_true, y_pred): 
    sum = 0 
    pair = 0  
    for i in range(1, len(y_true)): 
     for j in range(0, i): 
      if i is not j: 
       if(y_true[i] > y_true[j]): 
        pair +=1 
        sum += 1* (y_pred[i] > y_pred[j]) + 0.5 * (y_pred[i] == y_pred[j]) 
    if pair is not 0: 
     return sum/pair 
    else: 
     return 0 


def baseline_model(hidden_neurons, inputdim): 
    model = Sequential() 
    model.add(Dense(hidden_neurons, input_dim=inputdim, init='normal', activation='relu')) 
    model.add(Dense(hidden_neurons, init='normal', activation='relu')) 
    model.add(Dense(1, init='normal')) #output layer 

    model.compile(loss='mean_squared_error', optimizer='adam', metrics=[cindex_score]) 
    return model 

def run_model(P_train, Y_train, P_test, model): 
    history = model.fit(numpy.array(P_train), numpy.array(Y_train), batch_size=50, nb_epoch=200) 
    plotLoss(history) 
    return model.predict(P_test) 

baseline_model,run_model和cindex_score功能在one.py和下面的函數是在two.py,我叫模型,

def experiment(): 
    hidden_neurons = 250 
    dmodel=baseline_model(hidden_neurons, train_pair.shape[1]) 
    predicted_Y = run_model(train_pair,train_Y, test_pair, dmodel) 

,但我得到了下面的錯誤,「對象類型'Tensor'沒有len()「。它也不適用於shape屬性。例如,y_true表示爲Tensor(「dense_4_target:0」,shape =(?,?),dtype = float32),其形狀爲Tensor(「strided_slice:0」,shape =(),dtype = INT32)。

請問如何在Tensor對象內迭代?

最佳,

+0

你爲什麼選中「如果我不是J」 ......是不是真的所有的時間? – Pedia

回答

1

,那麼你可以嘗試使用此代碼來代替:

def cindex_score(y_true, y_pred): 

    g = tf.subtract(tf.expand_dims(y_pred, -1), y_pred) 
    g = tf.cast(g == 0.0, tf.float32) * 0.5 + tf.cast(g > 0.0, tf.float32) 

    f = tf.subtract(tf.expand_dims(y_true, -1), y_true) > 0.0 
    f = tf.matrix_band_part(tf.cast(f, tf.float32), -1, 0) 

    g = tf.reduce_sum(tf.multiply(g, f)) 
    f = tf.reduce_sum(f) 

    return tf.where(tf.equal(g, 0), 0.0, g/f) 

以下是驗證兩種方法均爲eq的代碼uivalent:

def _ref(J, K): 
    _sum = 0 
    _pair = 0 
    for _i in range(1, len(J)): 
     for _j in range(0, _i): 
      if _i is not _j: 
       if(J[_i] > J[_j]): 
        _pair +=1 
        _sum += 1* (K[_i] > K[_j]) + 0.5 * (K[_i] == K[_j]) 
    return 0 if _pair == 0 else _sum/_pair 

def _raw(J, K): 

    g = tf.subtract(tf.expand_dims(K, -1), K) 
    g = tf.cast(g == 0.0, tf.float32) * 0.5 + tf.cast(g > 0.0, tf.float32) 

    f = tf.subtract(tf.expand_dims(J, -1), J) > 0.0 
    f = tf.matrix_band_part(tf.cast(f, tf.float32), -1, 0) 

    g = tf.reduce_sum(tf.multiply(g, f)) 
    f = tf.reduce_sum(f) 

    return tf.where(tf.equal(g, 0), 0.0, g/f) 


for _ in range(100): 
    with tf.Session() as sess: 
     inputs = [tf.placeholder(dtype=tf.float32), 
        tf.placeholder(dtype=tf.float32)] 
     D = np.random.randint(low=10, high=1000) 
     data = [np.random.rand(D), np.random.rand(D)] 

     r1 = sess.run(_raw(inputs[0], inputs[1]), 
         feed_dict={x: y for x, y in zip(inputs, data)}) 
     r2 = _ref(data[0], data[1]) 

     assert np.isclose(r1, r2) 

請注意,這隻適用於一維張量(很少情況下,你會在keras中)。

+0

感謝您的回答。我對tensorflow一點都不舒服。我試圖理解你的翻譯。雖然,看到上面的代碼工作,我包括「導入tensorflow爲tf」我得到這兩個連續的錯誤,第一個是「一個」變量沒有定義,然後「返回tf.where(tf.equal(g,0 ),0.0,g/f)TypeError:其中()需要1到2個位置參數,但3個被給出「 –

+0

您有哪個版本的tensorflow? – Pedia

+0

另外,你在哪一行得到未定義的變量? – Pedia

1

更換len(y_true)y_true.shape[0]

如果舊版本TensorFlow的如果您正在使用tensorflow舒適的使用y_true.get_shape()

+0

我有以下錯誤,ylen = y_true.shape [0] AttributeError:'Tensor'對象沒有'形狀'屬性。 –

+0

看到這個答案,你可能會在TF的舊版本。 http://stackoverflow.com/questions/38666040/tensorflow-attributeerror-tensor-object-has-no-attribute-shape/ 「在TensorFlow 1.0之前的版本tf.Tensor沒有.shape屬性。應該使用Tensor.get_shape()「 – pyCthon

+0

對不起,我反應太快了,我糾正爲get_shape(),但我現在有這個錯誤,」我在範圍內(1,ylen): TypeError:__index__返回非int (類型爲NoneType)「 –

2

我用@Pedia代碼爲三維張量來計算的多標籤分類排名的損失:

def rloss(y_true, y_pred): 
    g = tf.subtract(tf.expand_dims(y_pred[1], -1), y_pred[1]) 
    g = tf.cast(g == 0.0, tf.float32) * 0.5 + tf.cast(g > 0.0, tf.float32) 
    f = tf.subtract(tf.expand_dims(y_true[1], -1), y_true[1]) > 0.0 
    f = tf.matrix_band_part(tf.cast(f, tf.float32), -1, 0) 
    g = tf.reduce_sum(tf.multiply(g, f)) 
    f = tf.reduce_sum(f) 
return tf.where(tf.equal(g, 0), 0.0, g/f) 


model = Sequential() 
model.add(Dense(label_length, activation='relu')) 
model.add(Dense(label_length, activation='relu')) 
model.add(Dense(label_length, activation='sigmoid')) 
model.summary() 


adgard = optimizers.Adagrad(lr=0.01, epsilon=1e-08, decay=0.0) 
model.compile(loss='binary_crossentropy', 
      optimizer=adgard, metrics=[rloss]) 
model.fit(X_train, y_train, 
     batch_size=batch_size, 
     epochs=n_epoch, 
     validation_data=(X_test, y_test), 
     shuffle=True)