Ответ 1
Я считаю, что вам нужно assign_slice_update
, описанное в билет # 206. Он пока недоступен.
UPDATE: теперь это реализовано. См. Ответ jdehesa: fooobar.com/questions/156922/...
До появления assign_slice_update
(или scatter_nd()
) вы можете создать блок нужной строки, содержащий значения, которые вы не хотите изменять вместе с требуемыми значениями для обновления, например:
import tensorflow as tf
a = tf.Variable(tf.ones([10,36,36]))
i = 3
j = 5
# Gather values inside the a[i,...] block that are not on column j
idx_before = tf.concat(1, [tf.reshape(tf.tile(tf.Variable([i]), [j]), [-1, 1]), tf.reshape(tf.range(j), [-1, 1])])
values_before = tf.gather_nd(a, idx_before)
idx_after = tf.concat(1, [tf.reshape(tf.tile(tf.Variable([i]), [36-j-1]), [-1, 1]), tf.reshape(tf.range(j+1, 36), [-1, 1])])
values_after = tf.gather_nd(a, idx_after)
# Build a subset of tensor `a` with the values that should not be touched and the values to update
block = tf.concat(0, [values_before, 5*tf.ones([1, 36]), values_after])
d = tf.scatter_update(a, i, block)
with tf.Session() as sess:
sess.run(tf.initialize_all_variables())
sess.run(d)
print(a.eval()[3,4:7,:]) # Print a subset of the tensor to verify
Пример генерирует тензор единиц и выполняет a[i,j,:] = 5
. Большая часть сложности заключается в получении значений, которые мы не хотим изменять, a[i,~j,:]
(иначе scatter_update()
заменит эти значения).
Если вы хотите выполнить T[i,k,:] = a[1,1,:]
по вашему запросу, вам нужно заменить 5*tf.ones([1, 36])
в предыдущем примере на tf.gather_nd(a, [[1, 1]])
.
Другим подходом было бы создать маску для tf.select()
желаемых элементов из нее и присвоить ее переменной, как таковой:
import tensorflow as tf
a = tf.Variable(tf.zeros([10,36,36]))
i = tf.Variable([3])
j = tf.Variable([5])
# Build a mask using indices to perform [i,j,:]
atleast_2d = lambda x: tf.reshape(x, [-1, 1])
indices = tf.concat(1, [atleast_2d(tf.tile(i, [36])), atleast_2d(tf.tile(j, [36])), atleast_2d(tf.range(36))])
mask = tf.cast(tf.sparse_to_dense(indices, [10, 36, 36], 1), tf.bool)
to_update = 5*tf.ones_like(a)
out = a.assign( tf.select(mask, to_update, a) )
with tf.Session() as sess:
sess.run(tf.initialize_all_variables())
sess.run(out)
print(a.eval()[2:5,5,:])
Он потенциально менее эффективен с точки зрения памяти, поскольку он требует в два раза больше памяти для обработки переменной a
-like to_update
, но вы можете легко изменить этот последний пример, чтобы получить операцию сохранения градиента от tf.select(...)
node. Вам также может быть интересно посмотреть на этот другой вопрос StackOverflow: Условное назначение тензорных значений в TensorFlow.
Эти неэлегантные искажения должны быть заменены на вызов правильной функции TensorFlow по мере ее появления.