现在我正在使用django 1.6
我有两个与OneToOneField
相关的模型。
class A(models.Model):
pass
class B(models.Model):
ref_a = models.OneToOneField(related_name='ref_b', null=True)
首先看一下指出问题的代码:
a1 = A.objects.create()
a2 = A.objects.create()
b1 = B.objects.create()
b2 = B.objects.create(ref_a=a2)
# then I call:
print(a1.ref_b) # DoesNotExist Exception raised
print(a2.ref_b) # returns b2
print(b1.ref_a) # returns None
print(b2.ref_a) # returns a2
现在问题是,如果我想检查一个A
对象,判断它是否存在引用它的B
个对象。我该怎么办?
我尝试的有效方法只是尝试捕捉异常,但还有其他更漂亮的方法吗?
我的努力:
1 - 以下代码有效,但太难看了!
b = None
try:
b = a.ref_b
except:
pass
2 - 我还尝试检查a中的属性,但不能正常工作:
b = a.ref_b if hasattr(a, 'ref_b') else None
朋友,你遇到同样的问题吗?请指点我,谢谢!
答案 0 :(得分:20)
所以你至少有两种检查方法。
首先是创建try / catch块以获取属性,其次是使用hasattr
。
class A(models.Model):
def get_B(self):
try:
return self.b
except:
return None
class B(models.Model):
ref_a = models.OneToOneField(related_name='ref_b', null=True)
请尽量避免裸except:
条款。它可以隐藏一些problems。
第二种方式是:
class A(models.Model):
def get_B(self):
if(hasattr(self, 'b')):
return self.b
return None
class B(models.Model):
ref_a = models.OneToOneField(related_name='ref_b', null=True)
在这两种情况下,您都可以使用它,没有任何例外:
a1 = A.objects.create()
a2 = A.objects.create()
b1 = B.objects.create()
b2 = B.objects.create(ref_a=a2)
# then I call:
print(a1.get_b) # No exception raised
print(a2.get_b) # returns b2
print(b1.a) # returns None
print(b2.a) # returns a2
没有其他办法,因为抛出异常是Django One to One relationships的默认行为。
这是从官方文档处理它的例子。
>>> from django.core.exceptions import ObjectDoesNotExist
>>> try:
>>> p2.restaurant
>>> except ObjectDoesNotExist:
>>> print("There is no restaurant here.")
There is no restaurant here.
答案 1 :(得分:7)
单个模型类提供了一个名为DoesNotExist的更具体的异常,它扩展了ObjectDoesNotExist。我的偏好是这样写:
b = None
try:
b = a.ref_b
except B.DoesNotExist:
pass
答案 2 :(得分:1)
hasattr
与Django1.11兼容!
您可以将getattr
用于较短的版本:
getattr(self, 'field', default)
以您的情况
b = getattr(a, 'ref_b', None)