梶研 [技育展 & カルマンフィルタについて調べた]
2023年9月26日

技育展 & カルマンフィルタについて調べた
出席率
- 3年セミナー:??%
スケジュール
短期的な予定
- 端末姿勢推定
- データを収集
- グラフを作成
- 静止状態の判別
- 端末の姿勢を推定
- 加速度を世界座標に変換
- カルマンフィルターを用いた推定
- 技育展 決勝
長期的な予定
- 9月中旬まで 端末姿勢推定(加速度, 角速度)
- 10月まで? 端末姿勢推定(カルマンフィルター)
- 10月中 Kotlin(とRust)の勉強 「いつでもセンシングアプリ」
進捗報告
技育展
結果
ファイナリストなれず
改善点
- 自分
- ノードを元にGoogle検索とQiita検索をしやすいように
- 初めて訪れた際、検索ボックスを表示するようにする
- ユーザビリティの向上(検索ボックス等)
- その他
- 2回目以降に訪れた際、最後に見たノードから始まるようにする
- 関数などより細かな情報を追加
感想等
ファイナリストには研究やOSSなど年単位で制作しているものが多く、
気軽に参加してファイナリストになれるものではないと感じた.
しかし、一部のファイナリストや企業賞を取っていた参加者の中には2ヶ月で制作したとうい人も多く、
制作期間だけではなく 技育展に適した アイデアなども重要だと思った.
端末姿勢推定
カルマンフィルタについて調べた
論文から
CiNii でカルマンフィルタを使っている研究論文を探した
GPS測位情報とセンサ情報に基づく位置推定システムに関する研究
- Dead Reckoning(DR)技術
- デッドレコニング
- カーナビに使われる
- GPSによる絶対位置推定情報とセンサによる相対移動推定情報をカルマンフィルタによって組み合わせる
- 歩数と歩幅は使わない
- 短時間フーリエ変換を用いる
- 最低2歩分の1s以上がよい
移動体の電波強度を利用した歩行者の位置精度向上手法の評価
- 電波強度にカルマンフィルタを適用
測距情報と姿勢情報と部屋情報を融合させた屋内位置推定システムに関する研究
- 端末の姿勢情報と周囲の壁面などの測距情報と,屋内の3次元部屋情報を融合する
加速度の時空間情報を考慮した進行方向推定手法の検討
論文は専門用語や未知の数学を使っていて難しかった.
Qiita 等
言いたいことは分かるが、数式が難しく計算方法がわからない
esa
相補フィルター
相補フィルターにも種類があることを知った
- 固定相補フィルタ
- どちらか一方を使う
- 可変相補フィルタ
- 重み係数でどちらも使う
- 静止時には加速度に重みを、移動時には角速度に重みをおくなど
Youtube
とても分かりやすかった.
が、プログラムに起こした時の形が見えない
Chat GPT
サンプルコードを生成してもらった.
プログラムから逆に考えてみる
生成したコード
1import numpy as np 2import pandas as pd 3 4# カルマンフィルターの初期化 5def initialize_filter(): 6 # 初期状態推定値 (クォータニオン) 7 x = np.array([1.0, 0.0, 0.0, 0.0]) # 初期の姿勢は単位クォータニオン(姿勢なし) 8 9 # 初期推定誤差共分散行列 10 P = np.eye(4) 11 12 # プロセスノイズの共分散行列 (システムノイズ) 13 Q = np.eye(4) * 0.01 # 適切な値を設定する必要があります 14 15 # 観測ノイズの共分散行列 16 R = np.eye(3) * 0.1 # 適切な値を設定する必要があります 17 18 return x, P, Q, R 19 20# クォータニオンを角速度に変換する関数 21def quaternion_to_rotation_rate(q, gyro_data): 22 q_conj = np.array([q[0], -q[1], -q[2], -q[3]]) 23 omega = 2.0 * R.from_quat(q_conj).inv().as_rotvec() * gyro_data 24 return omega 25 26# カルマンフィルターの更新ステップ 27def update_filter(x, P, Q, R, accel_data, gyro_data, dt): 28 # 予測ステップ 29 A = np.eye(4) + 0.5 * dt * np.array([ 30 [0, -gyro_data[0], -gyro_data[1], -gyro_data[2]], 31 [gyro_data[0], 0, gyro_data[2], -gyro_data[1]], 32 [gyro_data[1], -gyro_data[2], 0, gyro_data[0]], 33 [gyro_data[2], gyro_data[1], -gyro_data[0], 0] 34 ]) 35 x_pred = A.dot(x) 36 P_pred = A.dot(P).dot(A.T) + Q 37 38 # カルマンゲインの計算 39 H = np.eye(3) 40 K = P_pred.dot(H.T).dot(np.linalg.inv(H.dot(P_pred).dot(H.T) + R)) 41 42 # 角速度をクォータニオンに変換 43 omega = quaternion_to_rotation_rate(x, gyro_data) 44 45 # 観測更新 46 x_updated = x_pred + K.dot(accel_data - H.dot(x_pred)) 47 P_updated = (np.eye(4) - K.dot(H)).dot(P_pred) 48 49 return x_updated, P_updated 50 51# メインプログラム 52if __name__ == "__main__": 53 # データをDataFrameから読み込む (仮のデータ) 54 data = pd.read_csv('sensor_data.csv') # ファイル名は実際のファイルに合わせて変更 55 56 # カルマンフィルターの初期化 57 x, P, Q, R = initialize_filter() 58 59 # サンプリング時間 60 dt = data['timestamp'].diff().mean() 61 62 # データフレームをイテレートし、カルマンフィルターを更新 63 for index, row in data.iterrows(): 64 accel_data = np.array([row['accel_x'], row['accel_y'], row['accel_z']]) 65 gyro_data = np.array([row['gyro_x'], row['gyro_y'], row['gyro_z']]) 66 x, P = update_filter(x, P, Q, R, accel_data, gyro_data, dt) 67 68 # 推定された姿勢をクォータニオンからオイラー角に変換 69 euler_angles = R.from_quat(x).as_euler('xyz') 70 71 print(f"推定されたロール角度: {euler_angles[0]} rad") 72 print(f"推定されたピッチ角度: {euler_angles[1]} rad") 73 print(f"推定されたヨー角度: {euler_angles[2]} rad")
進路関係
なし
余談
心霊スポット行ってきた
春日井市の 千歳楼
サーモグラフィで撮ってみた
幽霊に温もりはあるのか