一. 前言
背景
一开始笔者在研究数据发送与接收的时候,看到Wear数据类DataMap除了可以put基本类型外,还有个fromBundle方法来构建一个DataMap对象。所以一口气的将原本功能上的序列化对象传递过去,结果手表端一直都收不到消息。
问题原因
后来查阅了开发者文档,发现这个Bundle并没有我们在Intent中传递数据的那么强大,它并不支持序列化!先附上开发者文档的说明:
Returns a DataMap from a Bundle. The input Bundle is expected to contain only elements supported by DataMap. Any elements in the Bundle not supported by DataMap will be dropped.
http://developer.android.com/reference/com/google/android/gms/wearable/DataMap.html#fromBundle(android.os.Bundle)
关键的一句是"The input Bundle is expected to contain only elements supported by DataMap",只有DataMap本身支持的类型才能被传递到Bundle中,那么也就是说我们自己构建的数据Bean并不能够直接序列化传进Bundle中,这导致结果是"Any elements in the Bundle not supported by DataMap will be dropped",即这些数据不会被传递,而是会被抛弃。
问题来了
那么,问题来了!怎么样才能够优雅地传递数据Bean呢?
接下来笔者将给出自己的解决方案。
二. 解决方案
1. 使用公共工程
- 创建公共工程:因为数据在Phone工程和Wear工程是共用,为了避免在这两个工程中都各自持有数据类,最好的方式自然是将这部分抽出来。创建完毕后,打钩"Is Library",将公共工程设置为库工程。
- 导入公共工程:Phone工程和Wear工程都引用公共工程。
- 公共工程模板:以下用一个城市数据作为样例。注意:要使用DataMap这个类,必须引用5.0版本以上的GooglePlayServices。
import com.google.android.gms.wearable.DataMap; public class WearCityBean { public static final String KEY = "/bean/wear/city"; private static final String M_CITY_ID = "mCityId"; private static final String M_CITY_NAME = "mCityName"; private static final String M_COUNTRY_NAME = "mCountryName"; // 城市ID private String mCityId; // 城市名 private String mCityName; // 国家名 private String mCountryName; public WearCityBean() { } public WearCityBean(DataMap map) { setCityId(map.getString(M_CITY_ID)); setCityName(map.getString(M_CITY_NAME)); setCountryName(map.getString(M_COUNTRY_NAME)); } public DataMap getDataMap() { DataMap map = new DataMap(); map.putString(M_CITY_ID, mCityId); map.putString(M_CITY_NAME, mCityName); map.putString(M_COUNTRY_NAME, mCountryName); return map; } public String getCityId() { return mCityId; } public void setCityId(String cityId) { mCityId = cityId; } public String getCityName() { return mCityName; } public void setCityName(String cityName) { mCityName = cityName; } public String getCountryName() { return mCountryName; } public void setCountryName(String countryName) { mCountryName = countryName; } @Override public String toString() { return "WearCityBean [mCityId=" + mCityId + ", mCityName=" + mCityName + ", mCountryName=" + mCountryName + "]"; } }
关键代码分析 - 发送数据:要发送数据的一端,将一个DataMap对象传入到这个方法中,将Bean自身的基本数据传递到DataMap中。这样,要发送数据的那一端就不用存在很多零散的代码,而且如果Phone端和Wear端都要实现数据发送的话,就不用重复两份代码了。
public DataMap getDataMap() { DataMap map = new DataMap(); map.putString(M_CITY_ID, mCityId); map.putString(M_CITY_NAME, mCityName); map.putString(M_COUNTRY_NAME, mCountryName); return map; }
发送端代码:下面通过先初始化一个WearCityBean对象后,再从中获取DataMap对象,最后再放到数据请求对象PutDataMapRequest中,完成发送的封装。
private void packageCityData(PutDataMapRequest dataMap) { //通过getWearCityBean()构建并初始化数据对象 WearCityBean wearCityBean = getWearCityBean(); DataMap map = wearCityBean.getDataMap(); dataMap.getDataMap().putDataMap(WearCityBean.KEY, map); }
关键代码分析 - 接收数据:将收到的DataMap用来构建一个WearCityBean对象,即可完成数据的解析。
public WearCityBean(DataMap map) { setCityId(map.getString(M_CITY_ID)); setCityName(map.getString(M_CITY_NAME)); setCountryName(map.getString(M_COUNTRY_NAME)); }
- 接收端的代码过于简单,就不再附上了。
后话
- 以上是笔者在API不够满足需求的情况下想出的一套解决方案,经过多次测试,运行良好。
- 除了直接的将基本数据类型传入DataMap中之外,还可以用Json或者其他方式将数据封装成一个String,再传入到DataMap中。这也是一个不错的方案。