Android 杂记 - 存货盘点用的客户端

最近有个盘点用的东西,要放到移动设备,本来用 .Net Compact Framework,CAB 部署在 CE 系统的移动条码设备。技术太旧,我用了这个周末两天时间,把这东西在试试实现在安卓上面,给用户看看效果。

Android 杂记 - 存货盘点用的客户端Android 杂记 - 存货盘点用的客户端Android 杂记 - 存货盘点用的客户端

Android 杂记 - 存货盘点用的客户端

Android 杂记 - 存货盘点用的客户端Android 杂记 - 存货盘点用的客户端Android 杂记 - 存货盘点用的客户端

Android 杂记 - 存货盘点用的客户端

很多很多的意外,大量 Gotcha… 分享一下我的惨痛经历。以下有错误、或各位大哥有更好做法,请留言赐教。

1. IIS 上面用 ASMX,返回 JSON,原来在客户端 POST 时候是必须注明 Content-type 为 json,否则 ASMX 返回 XML。POST 没有标识为 json,即使你自己序列化,它也是返回 XML 包裹着 JSON。

ASP.NET CODE:

namespace Stocktake {
/// <summary>
/// Summary description for RecordService
/// </summary>
[WebService(Namespace = "http://xxx.com/")]
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
[System.ComponentModel.ToolboxItem(false)]
// To allow this Web Service to be called from script, using ASP.NET AJAX, uncomment the following line.
[System.Web.Script.Services.ScriptService]
public class RecordService : System.Web.Services.WebService { [WebMethod]
[ScriptMethod(ResponseFormat = ResponseFormat.Json)]
public StocktakeCode getStocktakeCode() { var result = new StocktakeCode() {
Code = "Testing",
Remark = "Some Remark"
};
return result;
}
} public class StocktakeCode {
public string Code {
get;
set;
}
public string Remark {
get;
set;
}
}
}

这段服务器端代码,我拼了命找错处,*、Google、百度,原来,是没有错的。TMD x 10000。问题是下面客户端的,之前没 addHeader()。

ANDROID CODE:

public class DataAccess {
public static String getStocktakeCode(StocktakeActivity activity)
throws IOException {
HttpHost target = new HttpHost(getHostName(activity), getServerPort(activity));
HttpClient client = new DefaultHttpClient();
HttpPost post = new HttpPost(getStockCodeWsPath(activity));
post.addHeader("Content-type", "application/json; charset=utf-8"); // THIS IS IMPORTANT
HttpEntity results = null;
try{
HttpResponse response=client.execute(target, post);
results = response.getEntity();
return EntityUtils.toString(results);
} catch (Exception e){
throw new IOException("Web Service failed.");
} finally{
if(results!=null){
try{
results.consumeContent();
} catch (IOException e){ }
}
}
}
}

2. web.config 要加 protocol … 我忘记了加。

ASP.NET web.config XML:

<webServices>
<protocols>
<add name="HttpPost"/>
</protocols>
</webServices>

不熟悉  web 开发的我,这个也很要人命。

3. MS 自动序列化为 JSON 时,多了个莫名其妙的 d:。客户端那边,必须用 getString 方式过滤掉。里面的字符才是我们要的。

ANDROID CODE:

JSONObject raw = new JSONObject(HttpPost_result);
String text = raw.getString("d");

4. Android 内,JSONObject 要取节点内的值用getString,遇上null 会抛JSONException异常。必须检查。

ANDROID CODE:

JSONObject raw = new JSONObject(HttpPost_result);
if(!raw.isNull("d")){
String text = raw.getString("d");
JSONObject newObj = new JSONObject(text);
TextView tv = (TextView)activity.findViewById(R.id.textViewStocktakeCode);
tv.setText(newObj.getString("Code"));
TextView tv2 = (TextView)activity.findViewById(R.id.textViewStocktakeRemark);
tv2.setText(newObj.getString("Remark"));
activity.setIsStocktakeAllowed(true);
} else {
//...
}

我这设计是,服务器可能返回对象(序列化为 JSON),也可能返回 null。返回 null 时候,服务器发出的响应 JSON 是 {d:null}。

5. Android 内,启动 AsyncTask 必须用 execute()。可能各位安卓的大哥觉得理所当然吧,我没怎么看 documentation 就动手,结果我直接 doInBackground,在 onPreExecute 打断点不到。就这问题,查了 TMD 四小时…。

ANDROID CODE:

错误:

String result = new StocktakeCodeWorker(StocktakeActivity.this).doInBackground((String[])null);

正确:

new StocktakeCodeWorker(StocktakeActivity.this).execute((String[])null);

Execute 是 void 没返回值,要干就要在 Override onPostExecute 内。

6. Android 内,用 intent 打开另一个程序等返回值,要指定 app 的话必须 setPackage。

ANDROID CODE:

Intent intent = new Intent("com.google.zxing.client.android.SCAN");
intent.setPackage("com.google.zxing.client.android"); // THIS IS IMPORTANT
intent.putExtra("SCAN_MODE","ONE_D_MODE");
intent.putExtra("SCAN_FORMAT", "CODE_128");
startActivityForResult(intent, SCAN_BARCODE_REQUEST);

否则,有多个程序供应比如 com.google.zxing.client.android.SCAN 的话,画面会出现 app chooser 让用户选。我测试用的手机上,刚好就有(淘宝、和 ZXING 本身)。我在好几个 QQ 群上问为何,也问为何淘宝出现在 app chooser,无回复。我猜,应该是淘宝用了 ZXING 的代码,而且连开放 INTENT 的也拿了过来的原因。

加上 setPackage 后,问题消失,app chooser 没有出现而直接运行了 ZXING。

7. Android 的 Preferences 内,用 EditTextPreferences 即使指明 input type 是 number,保存的依然是 String。除非你一直直接用代码来保存 preferences 和读取 preferences,否则,SharedPreferences.getInteger 是废的。要自己 parse。

ANDROID Preference XML:

<EditTextPreference
android:key="server_port"
android:title="@string/preference_serverPort"
android:summary="@string/preference_serverPortRemark"
android:dialogTitle="@string/preference_serverPort"
android:inputType="number"
android:defaultValue="80" />

ANDROID CODE:

public static int getServerPort(StocktakeActivity activity){
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(activity);
String textResult = sp.getString("server_port", "").trim();
return Integer.parseInt(textResult);
}

8. 用 Eclipse 图形化界面添加 Permission ,android.permission.INTERNET,加完依然无法连。搜了一下 *,试试手工直接修改 XML,把它删了再手写一次,又可以了… 原因不明。

ANDROID Manifest XML:

<uses-permission android:name="android.permission.INTERNET"/>

真心不知道什么回事。

这完全是为了做个 protocol 出来给用户试试看而已。已能用,不美观,代码不公开。其实最关键部分,已经在上面全都写了。过程没有乐趣,只有痛苦。被采纳的话,接下去给下面的人改。

上一篇:[转] Console命令详解,让调试js代码变得更简单


下一篇:SQLite使用教程10 运算符