LoginSignup
32
32

More than 5 years have passed since last update.

自分用TensorFlowメモ

Last updated at Posted at 2016-04-30

今さらながら、TensorFlowの勉強をしています。
自分用にメモをしていたのですが、「しょぼくても公開したほうがイイヨ」とアドバイスを貰ったのでそうします。

書いてる内容は、全部どこかの他の記事の内容ですが、なるべくそれに自分の理解を追加して書いていきたいです(マサカリ投げてもらうため)。
間違ってる内容があれば指摘してもらえると喜びます。

以下メモをコピペ。


TensorFlowを試す&勉強するために読んだ記事やそこで得た知見を簡単にまとめる。
新しい知識を得るたび追記する。

チュートリアルを動かすまでに試した手順

導入

pipで問題なく導入できる。
ただし、導入時にTensorFlowのバージョン、python2か3か、CPU版かGPU版か、LinuxかMax OSかを指定してインストールをする必要があり、いずれかによってpipのコマンドが変わる。
詳細はここ を参照。下記は実際に使ったコマンド。

# Ubuntu/Linux 64-bit, GPU enabled:
$ sudo pip install --upgrade https://storage.googleapis.com/tensorflow/linux/gpu/tensorflow-0.7.1-cp27-none-linux_x86_64.whl

どこか詰まったら、大抵公式ページのIssueに解決方法が乗ってる。必ず確認。
GPUを使う場合は、CUDAへのPATH、LD_LIBRALY_PATHを通しておく必要がある(他のライブラリ、例えばChainerなどで必要になるものと同様)。

# こんな感じの内容を.bashrcとかに書いておく
$ export LD_LIBRARY_PATH="$LD_LIBRARY_PATH:/usr/local/cuda/lib64"
$ export CUDA_HOME=/usr/local/cuda

動作確認

Ipythonを起動して下記のコードを1行ずつ確認。
これは公式のイントロダクションにあったコード。
コードを読むに、線形変換のパラメータを目的関数(二乗平均誤差)にしたがって最適化する処理っぽい。

test.py
import tensorflow as tf
import numpy as np

# Create 100 phony x, y data points in NumPy, y = x * 0.1 + 0.3
x_data = np.random.rand(100).astype(np.float32)
y_data = x_data * 0.1 + 0.3

# Try to find values for W and b that compute y_data = W * x_data + b
# (We know that W should be 0.1 and b 0.3, but Tensorflow will
# figure that out for us.)
W = tf.Variable(tf.random_uniform([1], -1.0, 1.0))
b = tf.Variable(tf.zeros([1]))
y = W * x_data + b

# Minimize the mean squared errors.
loss = tf.reduce_mean(tf.square(y - y_data))
optimizer = tf.train.GradientDescentOptimizer(0.5)
train = optimizer.minimize(loss)

# Before starting, initialize the variables.  We will 'run' this first.
init = tf.initialize_all_variables()

# Launch the graph.
sess = tf.Session()
sess.run(init)

# Fit the line.
for step in xrange(201):
    sess.run(train)
    if step % 20 == 0:
        print(step, sess.run(W), sess.run(b))

# Learns best fit is W: [0.1], b: [0.3]

TensorFlowでは、numpy形式のデータを基にしたVariableというクラスで変数を管理する。
この変数の演算で、最適化したい式を表現する(上記コードのy)。

誤差関数はreduce_meanのようである。
二乗誤差を引数として与えて、それをlossにすると定義している。
他にも誤差関数のためのメソッドは用意されていそう。

最適化のためのアルゴリズムもたくさん実装サれているようで、ここでは最もよく使われるGradient Descent(SGDではない?後で調べる)が使われている。
0.5は恐らくlearning rate。
これをoptimizerとして定義し、このminimizeメソッドに先ほどのlossを与えている。

恐らく最も大事なのはsessionの概念。
TensorFlowでは事前に行いたい計算をまとめておき、それをセッションに投げることで処理を行う。
今回はtrainやinitがそれ。それぞれ、パラメータチューニングを行いたい計算、パラメータの初期化。

VariableやSessionといった概念はここここの記事を読んで勉強する。

MNISTを試す

公式のgitをcloneしてその中(tensorflow/examples/tutorials/mnist)にあるチュートリアルを試す。
チュートリアルのコードを一部変更

fully_connected_feed.py(一部抜粋)
# これを
#from tensorflow.examples.tutorials.mnist import input_data
#from tensorflow.examples.tutorials.mnist import mnist

# こうじゃ
import input_data
import mnist

どうやら、プロジェクトのルートから実行する前提のコードらしい。チュートリアルのディレクトリまで移動してから実行する場合は上のように変更が必要。

それでは、mnistディレクトリまで移動して以下のコマンドを実行。

$ python fully_connected_feed.py 

MNISTの学習が始まるはず。しばらく待つと終わって最終的な精度が表示される。
このチュートリアルはMLPを用いたもの。CNNを用いたものもある。

TensorFlowのキホン, データ構造・メソッド

最低限知っておくべき機能やデータ構造、メソッドについて書く。
内容はほぼ公式ページのHow toなどを読んで自分が解釈した内容。
間違いに気づく度に更新する。

VariableとPlaceholderの違い

TensorFlowで扱う変数にはVariableとPlaceholderの二つがある。

Variableは、ニューラルネットが持っている(入力層、隠れ層の重み等)パラメータを定義・保持しておくためのデータ構造。
誤差関数を最適化するようにチューニングしたいパラメータはVariableで定義するイメージ。

Placeholderは、ニューラルネットの入力として与えるデータの形を定義しておくためのデータ構造。
ネットワークを定義するときに、どのような入力がどのような形式で来るのかをPlaceholderを使って宣言しておき、実際の訓練や動作時に対応するデータを流し込むイメージ。

使用するデバイスの設定

どのデバイス(どのCPU?どのGPU?)を使うかは tf.deviceッメソッドで設定する。

# Pin a variable to CPU.
with tf.device("/cpu:0"):
  v = tf.Variable(...)

# Pin a variable to GPU.
with tf.device("/gpu:0"):
  v = tf.Variable(...)

# Pin a variable to a particular parameter server task.
with tf.device("/job:ps/task:7"):
  v = tf.Variable(...)

モデルのセーブ&ロード

モデルのセーブ&ロードのためのクラスとして、tf.train.Saverというものがある。
pickleを使っても保存できるとは思うが、特に事情がない限り用意されてるものを使うのが一番だと思う。

モデルのセーブ

Saverを定義してsaveメソッドを呼べば良い。
引数はセッションと保存先のディレクトリ+ファイル名の文字列。

example_save
# Create some variables.
v1 = tf.Variable(..., name="v1")
v2 = tf.Variable(..., name="v2")
...
# Add an op to initialize the variables.
init_op = tf.initialize_all_variables()

# Add ops to save and restore all the variables.
saver = tf.train.Saver()

# Later, launch the model, initialize the variables, do some work, save the
# variables to disk.
with tf.Session() as sess:
  sess.run(init_op)
  # Do some work with the model.
  ..
  # Save the variables to disk.
  save_path = saver.save(sess, "/tmp/model.ckpt")
  print("Model saved in file: %s" % save_path)

モデルを保存する場合、必ず各変数に名前をつけておく。
名前は、nameプロパティで付けられる。

モデルのロード

Saverを定義してrestoreメソッドを呼べば良い。
引数はセッションと保存先のディレクトリ+ファイル名の文字列。

example_load
# Create some variables.
v1 = tf.Variable(..., name="v1")
v2 = tf.Variable(..., name="v2")
...
# Add ops to save and restore all the variables.
saver = tf.train.Saver()

# Later, launch the model, use the saver to restore variables from disk, and
# do some work with the model.
with tf.Session() as sess:
  # Restore variables from disk.
  saver.restore(sess, "/tmp/model.ckpt")
  print("Model restored.")
  # Do some work with the model
  ...

一部のパラメータのみ保存

変数に名前を付けておくと、一部の変数のみ保存、ということが出来る。
コレを利用すると"モデルのサブセットのパラメータを保存して、別のモデルに転用する"という事ができる(?)。
Fine Tuningしたい場合は使うことになりそう。

example_save_subset
# Create some variables.
v1 = tf.Variable(..., name="v1")
v2 = tf.Variable(..., name="v2")
...
# Add ops to save and restore only 'v2' using the name "my_v2"
saver = tf.train.Saver({"my_v2": v2})
# Use the saver object normally after that.
...

参考資料

32
32
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
32
32