7

在 Apache HTTP 客户端库的帮助下从 Android 设备调用 Rails Web 服务时,我想克服以下情况

设想 :

客户端(Android):发起 HTTP POST 请求(使用 Apache HTTP 客户端,JSON 作为交换格式)将少量记录插入服务器数据库(MySQL)。

(SocketTimeOutException)服务器(Rails 中的 Web 服务):处理请求并将记录成功插入数据库,但同时在客户端发生超时。

客户端(Android):超时时,重试执行相同的 HTTP POST 请求,并再次将相同的记录插入数据库并发生数据库不一致。

任何人都可以帮助我克服这种情况。

提前致谢

编辑:

我目前的情况:

1) 用户尝试通过填写和提交数据使用 Android 应用进行注册。所以基本上 POST 服务包含发送到服务器的所有必需用户(Rails 和 MySql)

2) POST 请求由服务器处理,该过程主要负责将记录插入数据库,并使用插入数据的 USER_ID 向客户端提供响应。

3) 轰隆隆!!数据已插入 MySql 数据库,但由于某种原因存在SocketTimeOutException于 Android 端,因此 Android 端没有 USER_ID,用户尝试再次注册........ 数据库不一致,我很伤心:(

注意:步骤 3) 的发生非常罕见。但是当服务器太忙或有一些处理负载时,这种情况很常见。

4

3 回答 3

1

对于您上面描述的特定情况,我建议您在程序中添加一个会话 cookie。当用户发布数据时,您检查该会话是否已经存在 USER_ID,如果是,则返回它,如果没有,则执行数据库插入。因此,即使初始插入发生,然后在用户获得响应之前超时,您也可以在用户重试注册时提供响应。没有让你的数据库被破坏。

您还可以在数据库中查找相同(或其他唯一)记录(最近创建)并在第二个请求中返回该 USER_ID。这将取决于用户发送的数据。但是,如果它是注册并且您在短时间内连续收到两篇具有相同电子邮件的帖子,那么可以肯定它来自同一个用户......

编辑前:

由于您使用的是 Rails,您可以简单地利用create_or_update方法来创建记录。基本上,您检查记录是否已经存在,然后更新它们,而不是盲目地插入新记录。

这是一个您可能无论如何都必须处理的问题,因为即使您没有转发问题,两个客户也可能同时发布相同的信息。

如果您想更详细地说明其工作原理,则必须提供有关帖子包含内容的更多信息。在这种情况下,键和唯一字段很有趣,以及有关记录是否绑定到特定用户等的任何信息......

于 2013-02-07T12:51:35.137 回答
0

#rule始终允许 php 和 mysql 脚本在您的客户端(Android 或任何其他)之前超时。您可以通过在系统的所有端点上适当地设置超时来轻松确保这一点。

专门针对您的方案,您可以向您的用户表添加一个布尔值并将其命名为verified

  1. 用户注册时,verified=false默认未设置验证( )。
  2. 当用户第一次登录时,已验证设置为 ( verified=true)。
  3. 当用户注册并且username指定的已经存在时,如果验证==false则允许注册,否则不允许注册。
  4. 成本)用户注册时,他应该登录。最好删除注册页面并向用户显示一个视觉指示器,表明注册已完成,现在我们正在登录。我通常通过在注册时显示一个进度对话框来显示“注册...”然后进入登录页面并向用户显示一个进度对话框,上面写着“正在登录...”。

这应该确保问题不再发生。如果登录屏幕上发生错误(超时),用户应留在该屏幕上以便他可以重试。这确实是 android (Intent) 上的正常流程。


  • 您当然可以强制登录在注册页面上保持静音并继续重试直到它工作。
  • 您还可以删除额外的登录步骤,并在第一次交易时验证用户,但这会花费很多(您必须检查每个请求)
于 2013-03-04T15:45:23.013 回答
0

请尝试使用以下代码

HttpParams httpParameters = new BasicHttpParams();
            int timeoutConnection = 30000;
            HttpConnectionParams.setConnectionTimeout(httpParameters, timeoutConnection);
            int timeoutSocket = 30000;
            HttpConnectionParams.setSoTimeout(httpParameters, timeoutSocket);
            HttpClient httpclient = new DefaultHttpClient(httpParameters);
            HttpPost httppost = new HttpPost(url);
            StringEntity se = new StringEntity((JSONRequest));
            httppost.setEntity(se);
            httppost.setHeader("Accept", "application/json");
            httppost.setHeader("Content-type", "application/json; charset=utf-8");
            ResponseHandler<String> responseHandler = new BasicResponseHandler();
            strResponse = httpclient.execute(httppost, responseHandler);

根据您的要求设置超时。

于 2013-03-04T10:54:12.143 回答