我对Django很新,我试图弄清楚如何解决这个问题。 这是我的模特:
class Product(models.Model):
code = models.CharField(max_length=50)
name = models.CharField(max_length=50)
class ProductDetail(models.Model):
product = models.ForeignKey(Product, on_delete=models.CASCADE, editable=False)
price = models.DecimalField(max_digits=5, decimal_places=2, default=Decimal('999.99'))
available = models.BooleanField(default=True)
validity_tms = models.DateTimeField(default=timezone.now)
基本上,对于每个产品,我想跟踪价格和可用性的变化。因此,每个产品都有许多ProductDetail。
如果需要,我需要为每个产品获取代码,名称,可用和价格字段,但只与每个产品的最大validity_tms的ProductDetail相关。
此QuerySet仅包含我需要的部分信息:
ProductDetail.objects.values('product').annotate(max_date=Max('validity_tms'))
如何检索价格和可用字段?
有没有办法解决这个问题?我在模型中做错了吗?
答案 0 :(得分:3)
@Todor我需要一个
最大的那个validity_tms
然后您基本上在寻找greatest-1-per-group查询。
但是这不在ORM的capabilities内,只能用一个查询完成。
<强>更新强>
这个问题让我做了一些挖掘,我只是想通了我们可以用来自Django 1.11
的新员工来构建这样的查询。我们使用Subquery和OuterRef执行此操作,方法如下。
latest_product_details = ProductDetail.objects.filter(
validity_tms=Subquery(
(ProductDetail.objects
.filter(product=OuterRef('product'))
.values('product')
.annotate(max_date=Max('validity_tms'))
.values('max_date')[:1]
)
)
)
#now you can choose to work with ProductDetail itself:
for product_detail in latest_product_details.select_related('product'):
product = product_detail.product
print (product, product_detail)
#or you can work with Product objects and prefetch those latest details
products = Product.objects.my_complex_filtering().prefetch_related(
models.Prefetch('productdetail_set',
queryset=latest_product_details,
to_attr='latest_details'
)
)
for product in products:
#get the first element in the list or None if Empty
last_detail = next(iter(product.latest_details), None)
print (product, last_detail)
END OF UPDATE。
旧答案仍然适用于Django 1.11
之前的版本
我将为您提供2个查询的替代方法。这种方法不适用于每组具有多个对象的情况没有任何将该组过滤为仅几个元素的机会。这是因为,当你每组只需要1个时,所有对象都将被加载到内存中。
products = (Product.objects
.annotate(max_date=Max('productdetail__validity_tms'))
.prefetch_related(models.Prefetch('productdetail_set',
#this will fetch all related ProductDetails for a product
#if one product has many of them, consider further limiting
#the queryset with some additional filter
#e.g. only details from the last year or something like that.
#the idea is to lower the memory footprint, since you need only the first one
queryset=ProductDetail.objects.order_by('-validity_tms'),
to_attr='details'
))
)
#usage
for product in products:
print({
'product': product.id,
'max_date': product.max_date,
'detail_price': product.details[0].price if product.details else None,
'detail_available': product.details[0].available if product.details else None,
})
答案 1 :(得分:0)
尝试一下:
FINDREPLACE fr = {sizeof(fr), 0};
...
fr.Flags = FR_ENABLEHOOK;
fr.lpfnHook = &FRHookProc;
...
HWND hDlg = ReplaceText(&fr);