1

情况

嗨,我正在制作一个数据库驱动的游戏,其中有一个问题,如果用户找到正确的答案,他们有权更改问题(即写一个新问题)。

但这有一个重要的问题,我认为它被称为竞争条件。问题是这样的:

  • 用户 1 找到答案
  • 提示用户 1 输入新问题
  • 在用户 1 提交新问题之前,用户 2 也找到了新答案
  • 现在只有运气了​​,谁提交了新问题。如果用户 2 在用户 1 之后提交问题,则用户 2 的问题会覆盖用户 1 的问题。

现在,我已经通过使用某种信号量解决了这个问题,方法是保留它的timestamp. 总结它的工作原理是这样的:

  • 用户 1 找到答案
  • 信号量被保存到数据库中['user_ip' = 'x.x' , 'time'='y.y' , 'locked'='true']
  • 信号量在 t 秒后过期
  • 仅当信号量过期时才会接受新问题。

因此,当提交新问题时,应用程序仅在信号量锁定且当前时间t比信号量的日期大 (15) 秒时才会接受它。

但是这个也有问题,提示User 2

“你找到了答案,但有人在你之前找到了答案。如果他们没有在 15 秒内提交新问题,你可以更改问题。”

所以新的获胜者必须等待 15 秒,这可能是不必要的。

所以,

我想做的事

我想用秒表(计时器)提示“获胜者”一个表格。当它达到零时,它会解锁信号量,以便新的获胜者可以提交他们的问题。

但是这个解决方案会有这个问题:

如果获胜者在计时器归零之前关闭页面,那么 AJAX 请求将永远不会完成,信号量将被无限期锁定。

userLeave();因此,如果用户关闭页面,我必须确保我已经调用了该函数。

谷歌建议但人们onbeforeunloadonunload抱怨这些功能不起作用。那么,如果用户(获胜者)离开页面,是否有 100% 确定的方式来发送此 AJAX 调用?

4

1 回答 1

2

如果用户(获胜者)离开页面,是否有 100% 确定的方式来发送此 AJAX 调用?

永远不能100%依赖客户做任何事情。想想网络连接丢失、浏览器崩溃、不可用的功能(从不兼容 HTML5 的浏览器到禁用的 javascript)、脚本错误、以意想不到的方式调用(或不调用)您的 API 的错误配置代理或黑客故意这样做......</ p>

您对客户的唯一期望是他什么都不做——这称为超时。

如果你想让你的信号量在某个时间过期,你必须在服务器端创建计时器。如果您愿意,您还可以在客户端发出信号表示他不会提交答案(以任何方式)时立即中止计时器并过期。如果您想尝试关闭页面,您可以使用unload事件中的同步 (A)JAX。

顺便说一句:虽然我不了解你的游戏,但如果你只是接受这两个新问题会更容易,可能更舒服:-)

于 2013-09-17T01:39:58.627 回答