(This repository was clone from here, and some big checkpoint files were removed of history)
Note: this tutorial assumes you are already familiar with recursive neural networks and the basics of TensorFlow programming, otherwise it may be helpful to read up on both first. The tutorial and code follow the tree-net assignment of the (fantastic) Stanford CS224D class, and would be most useful to those who have attempted it on their own.
Recursive neural networks (also known as tree-structured, not to be confused with recurrent) provide state-of-the-art results on sentiment analysis tasks, but, due to network architecture being different for every example, can be hard to implement efficiently. Here we will benchmark two possible implementations. Complete implementations are in rnn_dynamic_graph.py and rnn_static_graph.py files.
The simplest way to implement a tree-net model is by building the computational graph manually on-the-fly for every input parse-tree, starting from leaf embeddings and gradually building it up using DFS tree traversal while re-using all the weight variables.
First, a data structure to represent the tree as a graph:
class Node: # a node in the tree
def __init__(self, ...):
self.isLeaf = True / False
self.hidden_state = None
# for leaves
self.word = word
# for inner nodes
self.left = None # reference to left child
self.right = None # reference to right child
...
Define model weights (once per execution, since they will be re-used):
class RNN_Model():
def add_model_vars(self):
with tf.variable_scope('Embeddings'):
embeddings = \
tf.get_variable('embeddings', [len(self.vocab), self.config.embed_size])
with tf.variable_scope('Composition'):
W1 = tf.get_variable('W1',
[2 * self.config.embed_size, self.config.embed_size])
b1 = tf.get_variable('b1', [1, self.config.embed_size])
with tf.variable_scope('Projection'):
U = tf.get_variable('U', [self.config.embed_size, self.config.label_size])
bs = tf.get_variable('bs', [1, self.config.label_size])
Build computational graph recursively using tree traversal (once per every input example):
class RNN_Model():
...
def build_subtree_model(node):
if node.isLeaf:
# lookup word embeddings
node.hidden_state = tf.nn.embedding_lookup(embeddings,
vocab.encode(node.word))
else:
# build the model recursively and combine children nodes
left_tensor = build_subtree_model(node.left)
right_tensor = build_subtree_model(node.right)
node.hidden_state = tf.nn.relu(tf.matmul(tf.concat(1, [left_tensor, right_tensor]), W1) + b1)
return node.hidden_state
Since we add dozens of nodes to the graph for every example, we have to reset the default graph every once in a while to save RAM:
def run_training_or_inference(self, input_trees):
for i in xrange(INPUT_BATCHES):
with tf.Graph().as_default(), tf.Session() as sess:
self.add_model_vars()
saver = tf.train.Saver()
saver.restore(sess, backup_path)
for tree in trees[i * BATCH_SIZE:(i + 1) * BATCH_SIZE]:
### run training or inference ###
saver.save(sess, backup_path) # in case of training, save progress
Luckily, since TensorFlow version 0.8, there is a better option: tf.while_loop
provides an option to implement conditionals and loops as a native part of the
TensorFlow graph, rather than Python code that sits on top of it.
The first part of the implementation is similar: we define the variables, same as before (by the way, the checkpoint files for the two models are interchangeable, meaning you can train with the dynamic graph version and run inference with the static graph, or vice versa).
embeddings, W1, b1, U, bs = ...
Then we store the input tree in a list form to make it easier to process in a single while_loop (you may have to run some simple tree traversal on input examples to flatten the trees into lists).
Consider a very simple tree, (the (old cat))
, with three leaves and two inner
nodes, (old cat)
and (the (old cat))
, the root.
vocab = {'the': 0, 'old': 1, 'cat': 2}
node_words = ['the', 'old', 'cat', '', '']
is_leaf = [True, True, True, False, False]
left_children = [-1, -1, -1, 1, 0] # indices of left children nodes in this list
right_children = [-1, -1, -1, 2, 3] # indices of right children nodes in this list
node_word_indices = [vocab[word] if word else -1 for word in node_words]
Now build the main computation graph node by node using while_loop. TensorArray below is a tensor with one flexible dimension (think a C++ vector of fixed-size arrays):
node_tensors = tf.TensorArray(tf.float32, size=0, dynamic_size=True,
clear_after_read=False, infer_shape=False)
def embed_word(word_index):
with tf.device('/cpu:0'):
return tf.expand_dims(tf.gather(embeddings, word_index), 0)
def combine_children(left_tensor, right_tensor):
return tf.nn.relu(tf.matmul(tf.concat(1, [left_tensor, right_tensor]), W1) + b1)
def loop_body(node_tensors, i):
node_is_leaf = tf.gather(is_leaf, i)
node_word_index = tf.gather(node_word_indices, i)
left_child = tf.gather(left_children, i)
right_child = tf.gather(right_children, i)
node_tensor = tf.cond(
node_is_leaf,
lambda: embed_word(node_word_index),
lambda: combine_children(node_tensors.read(left_child),
node_tensors.read(right_child)))
node_tensors = node_tensors.write(i, node_tensor)
i = tf.add(i, 1)
return node_tensors, i
loop_cond = lambda node_tensors, i: \
tf.less(i, tf.squeeze(tf.shape(is_leaf)))
node_tensors, _ = tf.while_loop(loop_cond, loop_body, [node_tensors, 0],
parallel_iterations=1)
This tiny code sample is fully working and builds a tree-net for our phrase.
Feel free to paste it into your terminal and run to understand the basics of how
while_loop
works. Run print sess.run(node_tensors.pack())
to see the output.
The actual code is a bit more complex (you would need to define placeholders for input data and feed that into the model, otherwise the graph would be re-created every time from scratch again), so take a look at the full implementation.
Dynamic graph: 1.43 trees/sec for training, 6.52 trees/sec inference.
Static graph: 23.3 trees/sec for training, 48.5 trees/sec inference.
Conclusion: training 16x faster, inference 8x faster!
Another benefit of building the graph statically is the possibility to use more advanced optimiziation algorithms like Adam.
If you build the graph on the fly, attempting to simply switch
tf.train.GradientDescentOptimizer(self.config.lr).minimize(loss_tensor)
to tf.train.AdamOptimizer(self.config.lr).minimize(loss_tensor)
would crash
the training code:
FailedPreconditionError (see above for traceback): Attempting to use
uninitialized value beta1_power
[[Node: beta1_power/read = Identity[T=DT_FLOAT, _class=["loc:@Embeddings/embeddings"], _device="/job:localhost/replica:0/task:0/gpu:0"](beta1_power)]]
...
This happens because Adam creates custom variables to store momentum
estimates, and their number depends on the structure of the graph. As a result,
there would have to be a re-initialization op for the new variables before every
train_op
, making the training process extremely slow.
However, for the static graph version swapping one optimizer for another works just fine. Note how much faster Adam converges here (though it starts overfitting by epoch 4).
Epoch | SGD train set loss | Adam train set loss | SGD train set accuracy | Adam train set accuracy | SGD dev set accuracy | Adam dev set accuracy |
---|---|---|---|---|---|---|
1 | 4.857 | 3.934 | 0.52 | 0.90 | 0.46 | 0.62 |
2 | 4.613 | 1.784 | 0.5 | 0.98 | 0.5 | 0.75 |
3 | 4.482 | 0.750 | 0.5 | 0.99 | 0.5 | 0.82 |
4 | 4.304 | 0.603 | 0.5 | 0.99 | 0.5 | 0.76 |
5 | 4.058 | 0.411 | 0.5 | 0.99 | 0.5 | 0.72 |
It should not be too hard to add batching to the static graph implementation, speeding it up even further.
We would have to pad the placeholders up to the length of the longest tree in the batch, and in the loop body replace tf.cond(...)
on a single value with tf.select(...)
on the whole batch.