質問するログイン新規登録

回答編集履歴

2

追記と真実

2017/03/05 17:29

投稿

sharow
sharow

スコア1151

answer CHANGED
@@ -3,4 +3,31 @@
3
3
  ```python
4
4
  __all__ = [ModuleA.__name__]
5
5
  ```
6
- のようにするとか
6
+ のようにするとか
7
+
8
+ ----
9
+ 追記:
10
+ 質問タイトルから乖離した回答だったので、補足です。この方法はflake8の警告を消す方法ではなくて、文字列リテラルを避ける方法です。以下の理由は私個人のもので、一般的ではありません。
11
+
12
+ ### 文字列リテラルを避けたい
13
+ どんな言語でも文字列リテラルの中身が正しいかどうかは検証できないので、もし簡単に文字列リテラルの使用を避けられるのなら避ける、というのが私個人のやり方です。名前空間をもつオブジェクトが文字列リテラルでなくて`__name__`で参照されれば、flake8でも検証できます。ただ、flake8で警告が抑止されるのは結果であって、そのために文字列リテラルを避けてるのではありません。flake8を使わなくとも文字列リテラルはなるべく避ける、それだけです(例えば可能な場合はdictよりもnamedtupleを使う、など)。
14
+
15
+ ### importに対するnoqaを避けたい
16
+ もう一つの理由はF401を抑止する#noqaがイケてないと思っているからです。他の#noqaはその行に対する警告を抑止します。抑止したい警告はその行にあるので、もしその行を編集して、警告が出ない状態になれば#noqaコメントを消します。#noqaが抑止している警告はその行に依存してる、と言うことができます。視覚的にはとても近いので普通は気付けます。
17
+
18
+ 一方、importのF401に対する#noqaはソースコードのその行以下全部に依存しています。ソースコードに変更が入ってF401警告が出ない状態になったときに、import部分まで戻って#noqaを消す、という手順になります。これは視覚の範囲外であることが多いので、編集している人の頭に頼ったやりかたです。時間が空いたり、チームで作業していたりしてこれを忘れると、1.不要になった#noqaを消し忘れる、2.本当に使っていないのに#noqaで抑止されている、の2つケースに陥ります。
19
+
20
+ そもそも、importに対する#noqaをやらなければこんなことは考えなくていい、そしてimportに対する#noqaが必要になる理由は、文字列リテラルを列挙する形で`__all__`を定義しているからだ、というのが私の回答の背景でした(過去形)。
21
+
22
+
23
+ ## 真実
24
+ ここまで書いておいて、**こんな結末はないだろ**という感じもしますが、質問者さんの`__all__`の値は「文字列のタプル」ではなくてただの「文字列」になっています。
25
+
26
+ ```python
27
+ __all__ = ("ModuleA") # -> == "ABC" 誤
28
+ __all__ = ("ModuleA",) # -> 正
29
+ ```
30
+
31
+ ちゃんとタプルかリストにすればflake8(ver 3.3.0)はimportされたものが使われている、と認識してくれるようです。
32
+
33
+ これを最初に指摘できなくてごめんなさい・・・。

1

修正

2017/03/05 17:29

投稿

sharow
sharow

スコア1151

answer CHANGED
@@ -1,6 +1,6 @@
1
1
  あるいは
2
2
 
3
3
  ```python
4
- __all__ = (ModuleA.__name__)
4
+ __all__ = [ModuleA.__name__]
5
5
  ```
6
6
  のようにするとか