7
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

IPFactoryAdvent Calendar 2023

Day 8

pythonだけでwebアプリを作る!streamlit!

Last updated at Posted at 2023-12-07

pythonだけでwebアプリを作る!streamlit!

この記事はIPFactory Advent Calender 2023 8日目の記事です。

記事の目的等

コンテストのためにstreamlitを使ったツールを作ったので、streamlitとはなにかを軽く説明したのち、他のstreamlitの記事になかったような要素とよく使った要素を紹介していこうと思います。

この記事はあくまで補足で使ってほしいので他の方が書かれた見ておくべきおすすめの記事を合間に貼っておきます。
とりあえず大体どんな要素があるか知りたいとき
https://qiita.com/not13/items/dcd8c12d64982dc0e819

優位点

エラーすら吐かないhtml,js,phpを使わなくていい!
(pythonなので)エラーがわかりやすい!
(pythonなので)いろいろなライブラリと併用できる!

shell
pip install streamlit

で簡単に導入できる

起動方法

基本shellから起動するのですが、streamlitを扱うファイルと同じフォルダに以下のようなbatファイルを作ることで1クリックで起動することができます。
おすすめです。

streamrit_test.bat
streamlit run --server.address localhost "streamlit_test.py"
cmd /k

rem "streamlit_test.py"には起動したいファイル名を入れる

また、コードを変更して保存した際は既に開いているstreamlitのページをリロードするだけで更新されるので起動しなおす必要がないです。
ただし複数タブの起動はできないと思います。

できること

タイトル、本文

streamrit_test.py
import streamlit as st

st.title('Streamlit_Test')

st.write('Display Image')

image.png

入力フォームを作る

streamrit_test.py
import streamlit as st

st.title('Streamlit_Test')

name = st.text_input('Enter your name')

st.write('Hello', name, "!")

image.png
image.png

このようにとても簡単に要素を追加することができます。

本題

いろいろなパラメータ

streamrit_test.py
import streamlit as st


word = st.text_input(value="word", label="word",placeholder="input word")
number = st.number_input(value=1, label="number", placeholder=100, min_value=0, max_value=1000)

if st.button("submit"):
    st.write(f"word: {word}")
    st.write(f"number: {number}")

image.png

  • lavel
    • 左上に出るタイトル
  • value
    • 初期値
    • numberはここに整数を入れることで整数のみを扱い、小数をいれることで小数を扱うことができるようになる
  • placeholder
    • 入力欄が空のとき背景にうっすら出てるやつ
  • max_value,min_value
    • 数字の上限値、下限値

その他使えるパラメータは公式ドキュメントを参照してください。

st.empty()で場所を確保する

次はMBSDCCでツールを作るときによく使っていた位置を先に決めておいてその内容を置換するやつです。
Javascriptでもこんなのがあった気がします。(がまともに勉強してないのでよくわかりません)

本来st.empty()は要素を初期化するために使うものですが、要素が存在しない場合変数宣言のような挙動をとります。

streamrit_test.py
import streamlit as st

word = st.text_input(value="word", label="word")
number = st.number_input(value=1, label="number", placeholder=100)

wordPlace = st.empty()
numberPlace = st.empty()

if st.button("submit"):
    wordPlace.text(word)
    numberPlace.write(number)

if st.button("multiply"):
    wordPlace.text(word*number)
    numberPlace.write(number*number)

image.png

↑ これはsubmitを押した状態

image.png

↑ multiplyを押した場合

これはwordPlaceとnumberPlaceを先に定義し、場所を確保したうえで中身を書き換えています。
何も場所を指定しない場合、本来ならsubmitボタンよりも下に"word"や"100"は生成されるのですが、今回はinputとbuttonの間に場所を確保してあるので真ん中に表示されます。
また、multiplyボタンを押した際、wordPlaceは1行しかないので溢れています。
st.xxx_inputとst.empty()による場所指定も上から順番に入るので、先に場所を確保(して後述のsesson_stateで保存)すればinputより上に表示することもできます。

コラムを作って縦に分割する

image.png
↑ 決めた比を超えることはない

streamrit_test.py
import streamlit as st

st.title("Streamlit Test")

number = st.number_input(label="number", value=1, placeholder="Enter a number")

form = st.empty()

with form:
    name_number_about = st.empty()
    col0, col1, col2 = name_number_about.columns((3, 2, 5))
    col0.write("I love this number!")
    col1.write(number)
    col2.write("Here's some textttttttttttttttttttttttttttttttttttttttttttttttttttttttttttttttttttttt")

コラムを作ってそれらに比を割り振ることで横に様々な要素を並べることができます
今回は 3:2:5 で割り振っています
col2のように長い文章でもコラムに決められた横幅を超えることはないです

注意点

実行されるたび状況がリセットされる

streamlit最大の問題がこれで、実行(ボタンを押すなど)されるたびに
全てを1から読み込みなおします
そのためどこかに値を保持しておく必要があります

streamrit_test.py
import streamlit as st

st.title('Streamlit hatomato')

word = st.text_input('enter your favorite word')
number = st.number_input('enter your favorite number')
things = st.multiselect('enter your favorite things', ['apple', 'orange', 'banana'])

if st.button('submit'):
    st.write(word)
    st.write(number)
    st.write(things)

image.png
↑ これは値を入力してsubmitを押した直後

image.png
↑ 1文字消して更新しただけで先ほどの結果が消えてしまった

(見えないだけで値は持っている?…というか入力欄をもう一度見て値を入れ直しているのだと思う)

今はこれだけしか機能がないのでこの状態でも構いませんがこれがwebアプリを作ろうと思うとそうはいきません。

st.sesson_state[]で値を保持する

もちろん対策がないわけがなくて、sesson_stateというものを使えば状況を保持することができます。
sesson_stateは値の更新が起ころうと、sesson_stateにある値を保持し続けます。

streamrit_test.py
import streamlit as st

st.title('Streamlit hatomato')

word = st.text_input('enter your favorite word')
number = st.number_input('enter your favorite number')
things = st.multiselect('enter your favorite things', ['apple', 'orange', 'banana'])

if 'opened' not in st.session_state:
    st.session_state["opened"] = False

if st.button('submit'):
    if st.session_state["opened"] == False:
        st.session_state["opened"] = True 

if st.session_state["opened"] == True:
    st.write(word)
    st.write(number)
    st.write(things)

これは

  1. openedという値がsesson_stateになければfalseを入れる
  2. ボタンを押したときsession_state["opened"] が FalseならTrueにする
  3. もしsession_state["opened"]がTrueならst.write()の3つを実行

という形になっていて値が変わるたびに全て実行しなおしていることに変わりはないのですが、見かけ上は常にword,number,thingsの結果が見えています。

image.png
↑ 1文字消しても結果が表示され続けるようになった

他の方が書いた入力フォームとsesson_stateについての記事

ちょっと遅い

1から読み込みなおしていることから規模が大きくなると遅くなる
あとpythonゆえに遅い
悲しいね

まとめ

streamlitは

  • 簡単にwebアプリ作成を体験できる
  • 比較的簡単なかつ初学者が入りやすいpythonで書くことができる
  • エラーをちゃんと出してくれる
  • 起動がbatファイル1つでできる

いいライブラリだと思います。
ここでは紹介しませんでしたがNumpyやPandasを使ったグラフ等の表示による使われている気がします。
もしpythonでなにか作ってみたいと思ったらぜひ使ってみてください。

7
2
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
7
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?