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

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

新規登録して質問してみよう
ただいま回答率
85.48%
Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python 3.x

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

Q&A

解決済

1回答

1376閲覧

models.Manager でQueryのだしわけ

Kentaro0919

総合スコア258

Django

DjangoはPythonで書かれた、オープンソースウェブアプリケーションのフレームワークです。複雑なデータベースを扱うウェブサイトを開発する際に必要な労力を減らす為にデザインされました。

Python 3.x

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

0グッド

0クリップ

投稿2017/07/24 02:52

Djangoで 次のようなモデルがあるのですが、適切なqueryが思いつかず、悩んでいます。

お客様は複数の契約を持っています。

class Customer(models.Model):
customer_number = models.CharField(max_length=200)
customer_name = models.CharField(max_length=200)

class Contract(models.Model):
customer = models.ForeignKey(Customer, on_delete=models.CASCADE, blank=True, null=True)
contract_name = models.CharField(max_length=200)

class ProductMaster(models.Model):
product_id = models.CharField(max_length=200)
product_name = models.CharField(max_length=200)

各月の請求金額のモデル
class Billing(models.Model):
billing_product = models.ForeignKey(BillingProduct, on_delete=models.CASCADE)
value = models.FloatField()

契約によりプロダクトが決まるのですが、同じプロダクトでも契約ごとに他のプロダクトと
合算で請求するのか個別で請求すのか変わってきます。

class BillingProduct(models.Model):
contract = models.ForeignKey(Contract, on_delete=models.CASCADE)
product = models.ForeignKey(ProductMaster, on_delete=models.CASCADE)
billing_with = models.ForeignKey('self', on_delete=models.CASCADE, blank=True, null=True)

今は出しわけがうまく思いつかずfor文でぐるぐるして解決しています。

単体のプロダクト のパターン -> 自身の名称ででBilling.value を返す

この処理がすっきりしないのですが...
代表にぶら下がるパターン -> 代表の名称ででBilling.value を返す
代表になるパターン -> 自身の名称ででBilling.value を返す

代表の名称で合計してその値を返す

契約で請求対象になっているリストに対して
BillingProduct.objects.get().billingproduct_set.all()
を実行すると請求の代表を返します

契約で請求対象になっているリストに対して
BillingProduct.objects.filter().exclude(billing_with= None)
を実行するとぶら下がっているプロダクトの返します

ここまではDjangoのQueryすっきり結果が返ってきたのですが、models.Managerとかですっきり
できないでしょうか?

コンピュートプロパティーを返す別モデルを作ればいいのでしょうか??

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

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

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

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

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

guest

回答1

0

自己解決

Billingの中に計算を返すフィールドを作成することで少しはすっきりして
modelの中だけで閉じるようにできました。

def _get_BP_billing_set(self):
if len(self.billing_product.billingproduct_set.all()) == 0:
return None
else:
return self.billing_product.billingproduct_set.all()

def _get_kakin_with_children(self):
if self.billing_with == None and self.billing_set == None:
return self.kakin
elif self.billing_set:
children = [Billing.objects.get(billing_product=k, billing_date=self.billing_date, statisticmaste=self.statisticmaste) for k in self.billing_set]
return self.kakin + sum([k.kakin for k in children])
elif self.billing_with == None:
return None

kakin = property(_get_kakin)
new_kakin = property(_get_kakin_with_children)

投稿2017/07/25 02:31

Kentaro0919

総合スコア258

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問