質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.35%
Flask

FlaskはPython用のマイクロフレームワークであり、Werkzeug・Jinja 2・good intentionsをベースにしています。

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

2回答

3277閲覧

[Python]関数内で作成したインスタンスを関数外で利用したい

baboo

総合スコア8

Flask

FlaskはPython用のマイクロフレームワークであり、Werkzeug・Jinja 2・good intentionsをベースにしています。

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

1クリップ

投稿2021/01/18 21:27

#実現したいこと
特定の関数の中で作成したインスタンスをその関数の外で利用したいと考えています。
具体的にはFlaskを利用してルーティングを指定している中で、

0 始めるボタン(@app.route(/))
1 タスクと目標の情報を入力(@app.route(/task))
2 睡眠や食事にかかる時間を入力(@app.route(/conditions))
3 結果の表示(@app.route(/result))

としているのですが、1の「タスクと目標の情報入力」で取得した値を生成したインスタンスに格納し、
その内容を保持したまま、2の「睡眠や食事にかかる時間を入力」で取得した値と共に3の処理へ渡したいと考えています。

#ソースコード

Python

1#!/usr/bin/env python3 2# -*- coding: utf-8 -*- 3 4#モジュールのインポート 5from flask import Flask, render_template, request 6 7#インスタンス化 8app = Flask(__name__) 9 10#クラスの定義 11class Task: 12 count = 0 13 def __init__(self, task, goal): 14 self.task = task 15 self.goal = goal 16 17class Conditions: 18 def __init__(self, sleep, breakfirst, lanch, dinner, bath, work, housework, others): 19 self.sleep = sleep 20 self.breakfirst = breakfirst 21 self.lanch = lanch 22 self.dinner = dinner 23 self.bath = bath 24 self.work = work 25 self.housework = housework 26 self.others = others 27 28#ルーティング 29@app.route('/') 30def top(): 31 return render_template("index.html") 32 33@app.route('/task') 34def task(): 35 return render_template("input_task.html") 36 37@app.route('/conditions', methods = ['POST']) 38def conditions(): 39 if request.form['task1'] and request.form['goal1'] and request.form['task2'] and request.form['goal2'] and request.form['task3'] and request.form['goal3']: 40 41 #task1, task2, task3のインスタンス化 42 task1 = Task(request.form['task1'], request.form['goal1']) 43 task2 = Task(request.form['task2'], request.form['goal2']) 44 task3 = Task(request.form['task3'], request.form['goal3']) 45 return render_template("input_conditions.html") 46 47@app.route('/result', methods = ['POST']) 48def result(): 49 if request.form['sleep'] and request.form['breakfirst'] and request.form['lanch'] and request.form['dinner'] and request.form['bath'] and request.form['work'] and request.form['housework'] and request.form['others']: 50 51 #conditionsのインスタンス化 52 conditions = Conditions(request.form['sleep'], request.form['breakfirst'], request.form['lanch'], request.form['dinner'], request.form['bath'], request.form['work'], request.form['housework'], request.form['others']) 53 54 return render_template("result.html", task1 = task1, conditions = conditions)

上記のソースコードでは最終行で「task1がない」とエラーになります。
そのため、現状は関数の外で空文字を入れたtask1というインスタンスを作成した上で、task1.task = request.form['task1']といった形で
1つずつ値を格納する方法をとっています。

作成したインスタンスをグローバル変数のように関数内外からアクセスできるようにするためにはどのようにすればよいでしょうか?
お知恵お借りできますと幸いです。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

ベストアンサー

この場合 API のエンドポイントをまたいで状態を維持しなければいけないので「状態の共有」が必要になります。その手段は様々です。y_waiwai さんがおっしゃっている「グローバル変数」への直接の代入も実はお手軽に始めるという条件付きであれば悪くありません(良くもないですが)。ちなみに Flask でグローバル変数を扱うには g オブジェクトというのがあるようです。
API — Flask Documentation (1.1.x)

Flask であればはじめから session 機能が付いているので、ユーザーのブラウザにデータをクッキーで保持させておくことが可能になります。
API — Flask Documentation (1.1.x)
Flaskまとめ - Qiita
ただこれはあくまで「セッション情報」なので実際のアプリケーション情報をホイホイ入れるのには向いていません。ユーザーのアクセストークンだけ入れておいてデータベースから検索、というのがちゃんとしたやり方です。

ただそこまでするのも大変だと思うので折衷案としてこんなのはいかがでしょう。g オブジェクトにディクショナリをもたせてこれをデータベースとみなす。トークンをキーにしてデータを引っ張ってくる。キーがなかったらエラー。/login API を新たに追加し、ユーザー名をそのまま session に格納するようにする。余裕があったら /logout も作ってみる。各 API では session からトークンを取得し g からディクショナリを取得しそれらを使ってデータを取得。

こんな感じです。ちょっと大変ですがこれでだいぶ本格的なアプリになると思います。

投稿2021/01/19 06:59

A_kirisaki

総合スコア2853

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

baboo

2021/01/20 13:22

kirisakiさんいつもありがとうございます! gオブジェクト知らなかったです・・・勉強になります! おっしゃる通り、グローバル変数への代入が手軽でいいなと思っていた(というかそれしかわからなかった)のですが、 セッションを活用したやり方もあるのですね!まだご回答いただいた内容を完璧に理解できている状態ではないですが、手を動かしながら確認していきます!いつもありがとうございます!
guest

0

グローバル変数にインスタンスを代入しておきましょう

投稿2021/01/18 22:49

y_waiwai

総合スコア88042

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

baboo

2021/01/20 13:18

こちらご回答いただきありがとうございます! なぜだかpythonのグローバル変数の扱い方がよくわかっていなかったのですが、このご回答をきっかけに調べ直し、無事理解することができました!ありがとうございます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.35%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問