題意の通り、クラス内でクラスメソッドを使用したいのですができません。下記に例を示しています。
これを実現するにはどうすれば良いでしょうか?
python
1class Test: 2 3 _num = cls.__set_num() 4 5 @classmethod 6 def __set_num(cls): 7 return 1 8 9#clsは定義されていませんとなる
気になる質問をクリップする
クリップした質問は、後からいつでもMYページで確認できます。
またクリップした質問に回答があった際、通知やメールを受け取ることができます。
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2017/10/12 13:49
回答3件
0
ベストアンサー
> クラス変数として持たせたい情報をファイルから読み込まなきゃいけなく
なるほど。共通のやや重い前処理があるという感じですね。
まずクラス変数の定義で自身のメソッドを呼ぶのは言語仕様上おそらく無理だと思います。代替案を考えてみたのですが、スマートに実装する方法がなかなか思いつきませんでした。
案1:まず クラスに閉じたい && クラスメソッドとして呼びたいんだ!というモチベーションに全振りした実装。
python
1class TestStatic(object): 2 @classmethod 3 def __get_num(cls): 4 if not hasattr(cls, '__num'): 5 print('1回だけやる処理') 6 cls.__num = 123456 7 return cls.__num 8 9 @classmethod 10 def say_hello(cls): 11 num = cls.__get_num() 12 print('こんにちは、{}回お会いしましたね。'.format(num)) 13 14""" 15>>> TestStatic.say_hello() 161回だけやる処理 17こんにちは、123456回お会いしましたね。 18>>> TestStatic.say_hello() 191回だけやる処理 20こんにちは、123456回お会いしましたね。 21"""
クラス変数は無理っぽいのでクラスメソッドで代替。
案2:次に クラスに閉じたい && インスタンス化してもいいや なパターン。
python
1from werkzeug.utils import cached_property 2 3class TestInstance(object): 4 @cached_property 5 def __num(self): 6 print('1回だけやる処理') 7 return 123456 8 9 def say_hello(self): 10 print('こんにちは、{}回お会いしましたね。'.format(self.__num)) 11 12 13""" 14>>> test = TestInstance() 15>>> test.say_hello() 161回だけやる処理 17こんにちは、123456回お会いしましたね。 18>>> test.say_hello() 19こんにちは、123456回お会いしましたね。 20"""
プロパティが使えるのでコードはすっきり、でも呼び出し側でインスタンスを管理しなければならずイケてない。
なお前処理は werkzeug.utils.cached_property を使って結果をキャッシュし2回目以降はそれを返すようにしています。
ちなみに案1と案2のいいとこどりをした「クラスプロパティ」なるものを作った方もいらっしゃいます:python メタクラスを使ってクラスプロパティを実装する - fakatatuku’s blog
案3:最後にクラス使わない式。
python
1from functools import lru_cache 2 3@lru_cache(maxsize=1) 4def __get_num(): 5 print('1回だけやる処理') 6 return 123456 7 8def say_hello(): 9 print('こんにちは、{}回お会いしましたね。'.format(__get_num())) 10 11""" 12>>> say_hello() 131回だけやる処理 14こんにちは、123456回お会いしましたね。 15>>> say_hello() 16こんにちは、123456回お会いしましたね。 17"""
こちらは、functools.lru_cach を使って前処理をキャッシュしています。
私も(PHPから入門したためか)グローバルより狭くローカル変数より広い → クラス変数という考えがありましたが、Python にはスクリプトファイルが「モジュール」という名前空間のような概念になっているので、わざわざ class に詰め込む必要もないのかなーと最近感じています。
投稿2017/10/12 15:06
編集2017/10/12 15:17総合スコア3095
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2017/10/12 15:23
0
追記
間違いでした。削除したいところですがサイトの削除ポリシーで許されていないのでこのままで失礼します。
単に定義する前に呼び出しているのが問題です(defは文であって、実行された時に名前空間に関数の名前を束縛するのです)。
python
1class Test: 2 @classmethod 3 def __set_num(cls): 4 return 1 5 6 print(Test.__set_num())
投稿2017/10/12 15:02
編集2017/10/12 15:13総合スコア11029
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2017/10/12 15:16
2017/10/12 15:24
0
cls
は__set_num
の引数であって、その外では定義されていないので当然使えません。
質問者さんがやりたいことがいまひとつ分からないですが、こうすればエラーはでないです。
lang
1class Test: 2 @classmethod 3 def _set_num(cls): 4 return 1 5 6Test._num = Test._set_num()
クラスの定義内でクラスメソッドにアクセスすることはできません。
定義したあとにメンバーを編集することはできます。
def __set_num(cls):
とアンダースコアを2つ先頭につけると、name manglingにより外からアクセスしにくくなるように自動的に名前が変えられます。
なので外からクラスメソッドをアクセスできるように上のコードではアンダースコアを1つにしました。
投稿2017/10/12 13:33
編集2017/10/12 13:53総合スコア2551
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
あなたの回答
tips
太字
斜体
打ち消し線
見出し
引用テキストの挿入
コードの挿入
リンクの挿入
リストの挿入
番号リストの挿入
表の挿入
水平線の挿入
プレビュー
質問の解決につながる回答をしましょう。 サンプルコードなど、より具体的な説明があると質問者の理解の助けになります。 また、読む側のことを考えた、分かりやすい文章を心がけましょう。