2018年3月25日日曜日

はじめるDeep learning

そうだ、Deep learningをやろう。そんなあなたへ送る解説記事です。
そう言いながらも私自身勉強しながら書いているので誤記や勘違いなどがあるかもしれません。もし見つけたらご連絡ください。

Deep learningとは

こちらのスライドがとてもよくまとまっています。

Deep learning

つまるところ、Deep learningの特徴は「特徴の抽出までやってくれる」という点に尽きると思います。
例えば相撲取りを判定するモデルを構築するとしたら、普通は「腰回りサイズ」「マゲの有無」「和装か否か」といった特徴を定義して、それを元にモデルを構築することになります。ちょうど関数の引数を決めるようなイメージです。

ところが、Deep learningではこの特徴抽出もモデルにやらせてしまいます。というか、そのために多層、つまりDeepになっています。
具体的には頭のあたりの特徴、腰のあたりの特徴、そしてそれらを複合した上半身の特徴・・・というように、特徴の抽出を並列・多層に行って学習させて、それでもって判定させようというのが根本的なアイデアです(以下、イメージ図)。

deeplerning.PNG

図中にあるとおり、Deep learningはそれを構成する技術要素を以下のように分けることができます。

  • データの分割
    判定に使用するデータの分割方法。画像判定の場合、画像を何pixelの幅で何pixelごと分割するかなど。
  • 特徴の抽出
    • 特徴抽出に利用するモデル
      特徴の抽出に使用するモデル。代表的な例だとAutoencoderや制限ボルツマンマシンなど。これらのモデルは、通常一つ一つにつき事前に学習を行っておく(Pre training)。
    • 抽出した特徴量の伝播方法
      抽出した特徴を、どのノードに、どうやって伝えるか。例えば、抽出した特徴をすべて次のノードに伝えるとすると、逆に誤差も伝播してしまうことになるので、そこを調整するなど。
  • 判定:抽出した特徴から最終的な判定を行う
    実はここはもうニューラルネットワークではなく、SVMなど一般的な分類器でもよい

以下は、その要素ごとに解説していきます。

データの分割

データや音声など、解析対象のデータには様々な特徴が含まれます。最も理想的なのは1データ1特徴の形で分割することですが、そもそも何が特徴かわかってない状態でそれを行うのは難しいので、なるべく特徴を欠損させないようにデータを分割するのが重要になります。

例えば相撲取りの判定で最も重要なのが「マゲ」の有無だったとすると、この「マゲ」が分割した画像に収まってくれているかが非常に重要になります。仮にマゲが分割した画像の境界線上にあったりしたら判定ができなくなってしまうためです。

これに対するアプローチとして、単にデータを分割するのではなくある程度かぶせる(オーバーラップさせる)という手法がよく取られます。画像を分割する単位をウィンドウと呼びますが、このウィンドウを少しずつずらしていくというイメージです。文字列で言えば、データがABCDEでウインドウサイズが3・移動幅が1だとすると、データを「ABC」「BCD」「CDE」という風に取ることになります。
※なお、これはDeep learningに限った話ではないです

特徴の抽出

Deep learningの最大の特徴は、特徴それ自体の抽出を行える点です。では、それはどのように行われるのか。

特徴抽出に利用するモデル

基本的なアイデアは、Autoencoderというものです。Autoencoderについての解説は以下が分かりやすいです。入力と出力を同じにしたモデルを作成し、限られた特徴で入力を再現できるよう学習させるようなイメージです。

Hello Autoencoder

これと共によく利用されるのが、制限ボルツマンマシン(Restricted Boltzmann Machine)です。こちらは、ざっくり言えば入力されたデータが発生するような確率分布を推定するという、確率モデルになります。解説はこちらが分かりやすいです。

2012-12-12 ゆるふわ Restricted Boltzmann Machine

何れも「与えられたデータをうまく説明できるパラメータを探索する」という点で本質的には同じです(参考1参考2)。

Deep learningではこれらを個別に学習させて(Pre training)結合していきます。1層目は各自、2層目は1層目のパラメータを固定したうえで学習、という感じで進めていきます。イメージ的にはスタブを利用した単体テストに近いです。
この事前学習によって、層が深くなると誤差伝播がうまくいかない(上の層から伝播していくので、下の層に届きにくい)という多層ニューラルネットワークの弱点を克服しています。

抽出した特徴量の伝播方法

次に問題になるのが、次の層にどう伝えるかです。ここは、上述の通り情報の伝播と誤差の伝播がトレードオフの関係になります。
(なお、以下では各学習モデルのことをノードと記載します)。

perception.PNG

出力値(活性化関数(Activation function))

まず次の層(ノード)に伝播する値の計算方法を決めます。一般的なニューラルネットワークと同様シグモイド関数が使用されることが多いですが、Rectified Linear Unitなども最近は使われるようです。
他にもmaxoutやLp Poolingなども利用されます。これらは複数の出力をまとめて値を決めるもので、maxoutはその名の通り最大値、Lp Poolingは・・・ちょっと一言では言えないいい感じの値を出力します。これについては、Lp poolingがとても分かりやすいのでご参照ください。 画像の細いエッジなどはそれを検知できるノードが少なくなり情報が伝播しにくくなりますが、これらはそうした微細な情報を取り洩らさないようにするためのテクニックになります(ざっくり言えば)。

伝播方法

伝播先をどのように設定するのかも検討事項です。一番簡単なのはニューラルネットワーク同様、次の層の全ノードに伝播することです。ただ、これだと当然誤差の伝播も発生し過学習などが発生することもあります。
これに対応する方法として、接続ノードを制限する局所受容野(Local receptive feild)や、ランダムに伝播しないノードを決定するDropoutといった方法が考案されています。

これらを決定した上で各ノードを結合していくことで、最終的に「事前学習済みの特徴抽出モデル」(ノード)が層状に組みあがります。これで抽出した特徴を利用し、最後の判定を行っていくことになります。

判定

最後判定を行うことになりますが、この最後の判定のための学習をFine trainingと呼びます。ここは、当然教師有学習になります。
最後までニューラルネットワークで組んで誤差伝播法で調整したり、SVMを上にくっつけて学習させるなどします。

以上がDeep learningの仕組みの解説になります。

Deep learningを使う

もう気づいている方がいると思いますが、実はDeep learningは新技術というわけではなく、既存のニューラルネットワークをうまく組み合わせたものになります。そのため実装もやろうと思えば自力でできる・・・と思うのですが、さすがにつらいのでライブラリがあった方がいいと思います。

ライブラリ

実装については、こちらにとてもよくまとまっています。
実装ディープラーニング

代表的なライブラリは以下になります。

GoogleがTensorFlowを発表し、これにかなり集約されていくのかなという感はあります。KerasがTensorFlowへの対応を行い、またscikit-learnライクに使えるラッパー(skflow)が登場するなど、周辺の開発も活発になっています。
画像ではCaffe、というのが以前の通説でしたがChainerではCaffeのモデル(Model Zooで学習済みのモデルが公開されている)のインポート機能が実装されるなど、この辺りの優位性はなくなってくるかもしれません。Caffeはとかくインストールの手間(Issueの多くがインストール関連)やPython3非対応など周辺状況がよろしくないこともあり、モデルのインポート機能や他ライブラリでの学習済みモデルの配布が進むにつれ、移行が進む可能性は高いと思います。

なお、気になるPython3対応は、以下のようになっています。

他は調べきれてないですが、多くのライブラリは対応をしています。TensorFlowで真っ先に立ったIssueがPython3対応だったことからも、Python2 onlyというのでは今後普及しないと思います。

ライブラリでなく自分で実装してみたい、という場合多くのライブラリがベースにしているTheano、またRestricted Boltzmann machinesを搭載しているscikit-learnが役立つと思います。yusugomori/DeepLearningはいろんな言語でDeep learningの実装をしているリポジトリで、こちらのソースコードも参考になると思います。

学習データ

学習用データをどこから持ってくるのかも、機械学習における問題の一つです。自分で数万件のデータを取ってくるのは大変ですし、教師有学習のためにラベルを付けるとなったら気が遠くなります。
そんなわけで学習用データを提供してくれているサイトがあるので、ありがたく使わせていただきましょう。
※ただ、研究はともかくアプリケーションに活かそうとする場合はデータは自前で用意しないといけない場合が多く、ここでは地道かつ泥臭い作業が発生します。

  • MNIST
    手書き文字のサンプルデータです。
  • CIFAR-10
    10クラス(airplane, automobileなど)にラベル付された画像集。CIFAR-100というより詳細なラベル付けがされたものもあります。
  • The Oxford-IIIT Pet Dataset
    CIFAR-10と同様、ラベル付きのデータ。その名の通り動物系です。
  • kaggle
    データ解析のコンペティションサイト。モデルの精度を競い合うことができ、データも提供されています。Kaggle Datasetsでデータの検索、また公開もできるようになりました。
  • Microsoft Azure Marketplace
    NFLの試合結果や人口統計など、様々なデータが提供されています(有料なものもありますが、無料も多いです)。
  • UC Irvine Machine Learning Repository
    機械学習のためのデータセットを集めているサイト。
  • Harvard Dataverse
    ハーバード大学が公開しているデータセット。Dataverseというソフトウェアを使って公開していて、このサイトでは同様にこのソフトウェアを使ってデータを公開しているところを見ることができる。
  • COCO
    Microsoftが公開している画像認識・また画像に対するキャプション等のデータセット(画像にキャプションを付けるNeuraltalk2ではこれが利用されている)
  • nico-opendata
    研究者を対象にニコニコ静止画・動画のコメントなどのデータを提供している
  • DATA GO JP
    日本政府が公開している、オープンデータのカタログサイト。こちらには、各自治体の公開しているデータもまとまっている。
  • Data.gov
    本家アメリカのオープンデータのカタログサイト
  • ShapeNet
    3Dモデルのデータセット。家具から飛行機までと、色々な種類のモデルがそろっている。
  • DCASE
    自然音の分類を行うタスク(公園の音、オフィスの音など)で、学習・評価用データが公開されている。

その他、Deeplearning情報サイトのDatasetでもいろいろなデータが紹介されています。画像系についてはResource - List of Interesting Very Large Datasets of Imagesによくまとめられています。

自然言語系はスクレイピングを行うことも多いですが、その際は倫理的な問題を頭に入れておきましょう(Webスクレイピングの注意事項一覧)。

実践

今回はメジャーどころなpylearn2を使ってみます。Windowsの場合、恒例ですがインストールが地獄門なのでこちらをご参照ください(今回ばかりはこの地獄を抜けられないと一瞬思った)。

pylearn2が用意できたところで、まずは欲張らずにチュートリアルを動かしてみます。
Quick-startにあるのはgrbm_smdというモデルで、これはDeep learningを構成する制限ボルツマンマシン(Restricted Boltzmann Machine)に該当します。

  • grbmRBM with Gaussian unitsの略です。RBMでは通常ノードが取れる値は0か1のバイナリ値ですが、これだと表現力に乏しいため、独立した分散を持つ線形関数(Gaussian unit)へ拡張したモデルになります。詳しくはこちらを参照してください(A Practical Guide to Training Restricted Boltzmann Machines 13.2の項)。この文書には、他にもRBMを実際に使う際のテクニックについてよくまとめられています。
  • smddenoising score matchingの意味で、これは入力データを意図的に欠損させ(ノイズを与える)、それがどれくらいうまく復元できたかをモデルの指標(=目的関数)とする方法です(Denoising autoencoder)。Deep learningの中にこうした役割の層を組み込むことで、例えば手書き文字で途中がかすれているような場合に対応力を上げることができます。

実際のモデルを図にしてみたものが以下になります。pylearn2では通常モデルをYAML形式のファイルで設定するので(今回のtutorialではcifar_grbm_smd.yaml)、そちらの方も見てみてください。

pylearn_quick.PNG

では、実際にやってみます。手順はREADMEに書かれているので、そちらをなぞっていく形になります。

  1. CIFAR-10データのダウンロード
    事前にpylearn2で使用するデータを格納するディレクトリを表す環境変数PYLEARN2_DATA_PATHを設定し、その後Gitのshellから
    pylearn2/scripts/datasets/download_cifar10.sh
    を実行します。
  2. make_dataset.pyの実行
    pylearn2/scripts/tutorials/grbm_smdフォルダに移動し、データ作成のスクリプトを実行します(仮想環境の有効化を忘れずに)。
    python make_dataset.py
  3. train.pyの実行
    以上でデータが用意できたので、学習を行います(本当はpylearn2/scriptsをパスに入れるんですが、面倒なので相対パスで実行しています)。
    ../../train.py cifar_grbm_smd.yaml
  4. 実行結果確認
    これで学習済みのモデルができたはずなので、その中身を見てみます。matplotlibが使用されますが、これはcondaでインストールしないとうまく動きませんでした(Tcl/TckのGUI表示やPIL周りなど)。
    ../../show_weights.py cifar_grbm_smd.pkl
    learning1.PNG
    これがどうなっていればいいのかは謎ですが・・・少なくとも重みの大きさを反映しており、欠損はランダムに入ることを考えると、示されている色・濃さは復元しにくい色/箇所であることを表している?(補完の必要がなければ重みは必要ないはずなので)。
    また、学習によって誤り率が下がっていく状況を以下コマンドで簡単に確認できます(プロット対象にb,L,Mを指定)。
    ../../plot_monitor.py cifar_grbm_smd.pkl
    learning2.PNG

QuickStartは以上です。
難しい内容も含みましたが、ここまでで少なくともモデルの概念と実行環境は手に入ったのではないかなと思います。Deep learningは文字通りDeepな世界ですが、この記事がその理解へのとっかかりになればと思います。

参考資料

Deep learning
Hello Autoencoder
2012-12-12 ゆるふわ Restricted Boltzmann Machine
Lp pooling
ニューラルネットの逆襲
Deep Learning技術の今
実装ディープラーニング
ねこと画像処理 part 3 – Deep Learningで猫の品種識別
Convolutional Neural Network
Convolution Neural Network for speech recognition
Building High-level Features Using Large Scale Unsupervised Learning




0 件のコメント:

コメントを投稿