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

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

新規登録して質問してみよう
ただいま回答率
85.50%
Python 3.x

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

Q&A

解決済

3回答

8041閲覧

Python クラス内でクラスメソッドを使用したい

_Victorique__

総合スコア1392

Python 3.x

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

1グッド

2クリップ

投稿2017/10/12 13:11

題意の通り、クラス内でクラスメソッドを使用したいのですができません。下記に例を示しています。
これを実現するにはどうすれば良いでしょうか?

python

1class Test: 2 3 _num = cls.__set_num() 4 5 @classmethod 6 def __set_num(cls): 7 return 1 8 9#clsは定義されていませんとなる
LouiS0616👍を押しています

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

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

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

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

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

miyahan

2017/10/12 13:40

自分の知る限り方法が思いつかないのですが、なにかクラス変数の定義時にクラスメソッドを呼ぶ具体的なシチュエーションを教えていただけると問題解決に協力できるかもしれません
_Victorique__

2017/10/12 13:49

ユーティリティクラスとして実装しているのですが、クラス変数として持たせたい情報をファイルから読み込まなきゃいけなく、このような事態になっています。何か別のアプローチにした方が良いでしょうか?
guest

回答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
miyahan

総合スコア3095

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

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

_Victorique__

2017/10/12 15:23

私も案3でいいかなーって思い始めました! でもなんか悔しいですよね( ;∀;) たくさん考えていただきありがとうございました!参考になりました!
guest

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
quickquip

総合スコア11029

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

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

quickquip

2017/10/12 15:16

正直モジュールメソッドにしない理由がわからなかったりしますが。
_Victorique__

2017/10/12 15:24

回答ありがとうございます!結果モジュールメソッドにすることになりました!
guest

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つにしました。

参考
https://stackoverflow.com/questions/13900515/how-can-i-access-a-classmethod-from-inside-a-class-in-python

投稿2017/10/12 13:33

編集2017/10/12 13:53
karamarimo

総合スコア2551

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.50%

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

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

質問する

関連した質問