summaryrefslogtreecommitdiffhomepage
path: root/vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml
diff options
context:
space:
mode:
authornsfisis <nsfisis@gmail.com>2023-09-07 22:27:48 +0900
committernsfisis <nsfisis@gmail.com>2023-09-07 22:35:53 +0900
commit994e0114d76ae19768d5c303874a968cf6369fd0 (patch)
tree5fd3f8b169eea00084b24fbae820f75273864d2a /vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml
parent57f015992f678bfd7281f171fb9d71349c96a1a0 (diff)
downloadnsfisis.dev-994e0114d76ae19768d5c303874a968cf6369fd0.tar.gz
nsfisis.dev-994e0114d76ae19768d5c303874a968cf6369fd0.tar.zst
nsfisis.dev-994e0114d76ae19768d5c303874a968cf6369fd0.zip
meta: migrate to monorepo
Diffstat (limited to 'vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml')
-rw-r--r--vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml87
1 files changed, 87 insertions, 0 deletions
diff --git a/vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml b/vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml
new file mode 100644
index 00000000..eb8aa452
--- /dev/null
+++ b/vhosts/blog/content/posts/2021-10-02/python-unbound-local-error.xml
@@ -0,0 +1,87 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<article xmlns="http://docbook.org/ns/docbook" xmlns:xl="http://www.w3.org/1999/xlink" version="5.0">
+ <info>
+ <title>【Python】 クロージャとUnboundLocalError: local variable 'x' referenced before assignment</title>
+ <abstract>
+ Python における UnboundLocalError の理由と対処法。
+ </abstract>
+ <keywordset>
+ <keyword>python</keyword>
+ <keyword>python3</keyword>
+ </keywordset>
+ <revhistory>
+ <revision>
+ <date>2021-10-02</date>
+ <revremark>Qiita から移植</revremark>
+ </revision>
+ </revhistory>
+ </info>
+ <note>
+ この記事は Qiita から移植してきたものです。
+ 元 URL: <link xl:href="https://qiita.com/nsfisis/items/5d733703afcb35bbf399">https://qiita.com/nsfisis/items/5d733703afcb35bbf399</link>
+ </note>
+ <para>
+ 本記事は Python 3.7.6 の動作結果を元にして書かれている。
+ </para>
+ <para>
+ Python でクロージャを作ろうと、次のようなコードを書いた。
+ </para>
+ <programlisting language="python" linenumbering="unnumbered">
+ <![CDATA[
+ def f():
+ x = 0
+ def g():
+ x += 1
+ g()
+
+ f()
+ ]]>
+ </programlisting>
+ <para>
+ 関数 <literal>g</literal> から 関数 <literal>f</literal> のスコープ内で定義された変数 <literal>x</literal> を参照し、それに
+ 1 を足そうとしている。 これを実行すると <literal>x += 1</literal>
+ の箇所でエラーが発生する。
+ </para>
+ <blockquote>
+ <para>
+ UnboundLocalError: local variable `x' referenced before assignment
+ </para>
+ </blockquote>
+ <para>
+ local変数 <literal>x</literal> が代入前に参照された、とある。これは、<literal>f</literal> の <literal>x</literal>
+ を参照するのではなく、新しく別の変数を <literal>g</literal> 内に作ってしまっているため。
+ 前述のコードを宣言と代入を便宜上分けて書き直すと次のようになる。<literal>var</literal>
+ を変数宣言のための構文として擬似的に利用している。
+ </para>
+ <programlisting language="python" linenumbering="unnumbered">
+ <![CDATA[
+ # 注: var は正しい Python の文法ではない。上記参照のこと
+ def f():
+ var x # f の local変数 'x' を宣言
+ x = 0 # x に 0 を代入
+ def g(): # f の内部関数 g を定義
+ var x # g の local変数 'x' を宣言
+ # たまたま f にも同じ名前の変数があるが、それとは別の変数
+ x += 1 # x に 1 を加算 (x = x + 1 の糖衣構文)
+ # 加算する前の値を参照しようとするが、まだ代入されていないためエラー
+ g()
+ ]]>
+ </programlisting>
+ <para>
+ 当初の意図を表現するには、次のように書けばよい。
+ </para>
+ <programlisting language="python" linenumbering="unnumbered">
+ <![CDATA[
+ def f():
+ x = 0
+ def g():
+ nonlocal x ## (*)
+ x += 1
+ g()
+ ]]>
+ </programlisting>
+ <para>
+ <literal>(*)</literal> のように、<literal>nonlocal</literal> を追加する。これにより一つ外側のスコープ (<literal>g</literal>
+ の一つ外側 = <literal>f</literal>) で定義されている <literal>x</literal> を探しに行くようになる。
+ </para>
+</article>