1

所以我现在正在构建我的第一个 Django 项目,一个双人国际象棋网站。我快完成了,现在我正在编写测试。该项目有两个原始模型:玩家和游戏。Player 模型与 User 模型具有一对一的字段 Player.user 和 Game 的外键 Player.current_game,它指示玩家当前所在的游戏实例。 Game.draw_offered CharField 采用值 "" (默认值)、“w”或“b”,指示是否已提供平局以及提供平局的玩家的颜色。我有一个执行以下操作的测试用例:

class Draw_ViewTestCase(TestCase):

    def setUp(self):
        self.user1 = User.objects.create_user(username="Test_User_1", password="test1")
        self.user2 = User.objects.create_user(username="Test_User_2", password="test2")
        self.factory = RequestFactory()
        self.game = Game.objects.create()
        self.player1 = Player.objects.create(user=self.user1, current_game=self.game, color="w")
        self.player2 = Player.objects.create(user=self.user2, current_game=self.game, color="b")

    def test_draw(self): 
        request = self.factory.get(reverse('draw'))
        request.user = self.user1
        #The draw view changes the draw_offered field of the player's current game to the player's color, and saves the current_game to the database
        response = draw(request)
        self.game.refresh_from_db()
        assert self.game.draw_offered == self.player1.color
        assert self.game == self.player2.current_game
        #self.player2.current_game.refresh_from_db()
        assert self.game.draw_offered == self.player2.current_game.draw_offered

除了最后一个,所有的断言都通过了,但是如果你取消倒数第二行的注释,它就会通过。

这是怎么回事?据我了解,当您引用外键属性 self.player2.current_game 时,Django 会执行数据库查找并返回带有最新字段的 Game 实例。由于 self.game 和 self.player2.current_game 对应数据库中相同的 Game 记录,而 self.game.refresh_from_db() 只是被调用,你会认为 self.game.draw_offered == self.player2.current_game.draw_offered将评估为 True——两个 Game 实例都引用相同的数据库记录并具有最新的字段。我必须调用 self.player2.current_game.refresh_from_db() 以使断言通过这一事实对我来说没有意义——根据我对 Django Foreign Keys 的理解,self.player2.current_game 应该自动更新与数据库。

4

2 回答 2

2

这与外键无关。

您缺少的是,仅仅因为两个实例指向同一个数据库记录,并不意味着它们是同一个对象。正如您必须刷新self.game才能查看对视图中的基础记录所做的更改一样draw,您还需要刷新self.player2.current_game以便它也获得这些更新的值。

如果 Django 认为它已经拥有外键,它不会进行数据库调用来查找外键:确实如此,因为您最初创建self.player1self.player2传入整个 Game 对象。所以是的,您需要显式刷新 current_game 对象以查看所做的更改。

如果您真的想要,您可以通过传递 ID 而不是完整对象来创建对象而不这样做self.player2 = Player.objects.create(...current_game_id=self.game.id...):这样,Django 实际上不会缓存对象,因此需要在测试结束时查询它。

于 2017-09-03T19:31:28.807 回答
0

测试运行器将在每次测试之前运行setUp 方法,因此您在运行代码之前就在实例上运行。文档中对此进行了很好的解释Unitest

您没有player2从数据库访问新实例,只是加载它,如方法中定义的那样setUp

于 2017-09-03T18:50:12.520 回答