1
class Hero(models.Model):
   talents = models.ManyToManyField(Talent)
   (...)

class Talent(models.Model)
   required_talents = models.ManyToManyField('self', symmetrical=False)
   (...)

我想创建方法has_required_talents(self, talent)for Hero,它将检查这Hero是否已required_talents选择Talent

我试过这个:

def has_required_talents(self, talent)
   required_list = talent.talents_required.values_list('id', flat=True)
   hero_talents = self.hero.talents.values_list('id', flat=True)
   if required_list in hero_talents:
      return True
   return False

但是,当我使用以下测试对其进行测试时,它无法正常工作:

class HeroTalents(TestCase):
 def setUp(self):
    self.hero=Hero('Duck')
    self.hero.save() 

 def test_has_required_talents(self):
    self.talent1 = Talent(name = "Overpower")
    self.talent1.save()
    self.talent2 = Talent(name = "Overpower2")
    self.talent2.save()
    self.talent2.talents_required.add(self.talent1)
    self.assertFalse(self.hero.has_required_talents(self.talent2), "test1")
    self.hero.talents.add(self.talent1)
    self.assertTrue(self.hero.has_required_talents(self.talent2), "test2")
    self.talent3 = Talent(name = "Overpower3")
    self.talent3.save()
    self.hero.talents.add(self.talent2)
    self.assertTrue(self.hero.has_required_talents(self.talent3), "test3")
    self.talent1.talents_required.add(self.talent3)
    self.assertFalse(self.hero.has_required_talents(self.talent1), "test4")

需要做些什么来完成这项工作?

4

1 回答 1

0
def has_required_talents(self, talent)
   required_list = talent.talents_required.values_list('id', flat=True)
   hero_talents = self.hero.talents.values_list('id', flat=True)
   for item in required_list:
      if not item in hero_talents:
         return False
   return True

那将是一种基于列表的方法。您还可以将它们转换为集合:

def has_required_talents(self, talent)
   required_list = set(talent.talents_required.values_list('id', flat=True))
   hero_talents = set(self.hero.talents.values_list('id', flat=True))
   if required_list.issubset(hero_talents):
      return True
   return False

但是(我不知道确切的内部结构,你可以运行一些测试)第一种方法应该更快,因为没有转换。

于 2012-11-01T12:09:42.663 回答