Android中封装自己的SDK示例和自己的总结

下面是转载于:http://blog.csdn.net/ryffic/article/details/44891899

这几天,忙的跟狗一样了。

新公司,新的责任。

最近,公司的一个新项目。我的工作,是负责安卓端网络通信层的编写。

其实,说实话,安卓项目经验也不是非常多。也是边学边成长吧。

最开始,我只是做一个简单的编写,利用Xutils框架,做了一个最简单的封装。

public void send(HttpMethod method, String url, RequestParams params) {

mHttpUtils.send(method, url, params, new RequestCallBack() {

@Override

public void onFailure(HttpException arg0, String arg1) {

}

@Override

public void onSuccess(ResponseInfo arg) {

}

});

}

说简单点,就是在框架上封装了一次,然后给业务层调用。

不过,项目经理看到这个以后,就屌我了:你这个不是网络通信层阿。只是一个方法而已。跟他进行沟通之后,了解到他的意思。他的想法是,业务层要进行网络通信时,只要调用某个业务,并传给参数给我,相应的url,请求方式t,都不是它关心的事。打个比方,现在有一个要登录的操作,业务层只要调用 login(参数)就可以了。

所以,我也在不停地尝试。

之后,回想到百度地图SDK中,有许多类似的例子。比如,现在我要做一个搜索Poi数据的操作,百度地图SDK也只有几步而已。

1:取得检索实例

2:设置监听

3:发起检索

简单的代码示例如下:

//取得Poi检索实例

mPoiSearch =PoiSearch.newInstance();

//设置监听

mPoiSearch.setOnGetPoiSearchResultListener(new OnGetPoiSearchResultListener() {

@Override

public void onGetPoiResult(PoiResult result) {

}

@Override

public void onGetPoiDetailResult(PoiDetailResult result) {

});

//发起检索

mPoiSearch.searchInCity(new PoiCitySearchOption().city("北京").keyword("餐厅").pageNum(1).pageCapacity(10));

由百度地图的SDK,我突然想到,是不是,我也能参照这样的步骤,做一个类似的SDK,给业务层调用呢。

然后,我就开始写代码。这里,我用到的是百度地图LBS云检索的API来做示例。

首先,我得有一个监听函数。那么,我就先写一个简单的接口。

/**

*@类名称:OnResultListener

*@类描述:  结果回调

*@创建人:Yul_Wu

*@创建时间:2015年4月5日 下午10:08:15

*@修改人:Yul_Wu

*@修改时间:2015年4月5日 下午10:08:15

*@修改备注:

*@version

*

*/

public interface OnResultListener {

void reDraw(String result,int error);

}

然后,参照百度SDK的,我们还有一个检索的类GetBaiduSearch(以下称为检索类)

/**

*

*

*@项目名称:Result

*@类名称:GetBaiduSearch

*@类描述:  检索类

*@创建人:Yul_Wu

*@创建时间:2015年4月5日 下午10:11:02

*@修改人:Yul_Wu

*@修改时间:2015年4月5日 下午10:11:02

*@修改备注:

*@version

*

*/

public class GetBaiduSearch {

private static GetBaiduSearch baiduSearch;

private GetBaiduSearch() {

}

/**

* 取得实例

*

* @return

*/

public static GetBaiduSearch getInstance() {

if (baiduSearch == null) {

synchronized (GetBaiduSearch.class) {

baiduSearch = new GetBaiduSearch();

}

}

return baiduSearch;

}

可以看到,这是一个非常简单的单例模式,通过getInstance()的方法,我们可以取得该类的实例。那们,我们还要有一个设置监听的方法,就叫setOnResultListener好了,那么代码就成这样了

public class GetBaiduSearch {

OnResultListener listener = null;

private static GetBaiduSearch baiduSearch;

private GetBaiduSearch() {

}

/**

* 取得实例

*

* @return

*/

public static GetBaiduSearch getInstance() {

if (baiduSearch == null) {

synchronized (GetBaiduSearch.class) {

baiduSearch = new GetBaiduSearch();

}

}

return baiduSearch;

}

public void setOnResultListener(OnResultListener listener) {

        this.listener = listener;

        this.listener.reDraw(“要回调的具体内容”, 0);

    }

 
  

这样,我们一个简单的回调函数,以及简单方法就写好了。接下来,我们来测试一下,在MainActivity中,我们做一个非常简单的测试。

mSearch= GetBaiduSearch.getInstance();

button.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

mSearch.setOnResultListener(new OnResultListener() {

@Override

public void reDraw(String result, int error) {

// TODO Auto-generated method stub

Log.v("yul_wu", result);

}

});

}

});

我们运行程序,点击Button,可以看到logcat上的输出:


这说明我们的数据成功返回了。

那么,接下来呢,我们再给检索类加上一个方法,也就是我们检索的主入口。

public void searchGeoTable() {

//这里用的是百度地图LBS云检索的例子,把通过API获得的数据返回给我们的监听函数

HttpUtils utils = new HttpUtils();

RequestParams params = new RequestParams();

String url = "http://api.map.baidu.com/geodata/v3/geotable/list";

params.addQueryStringParameter("ak", "百度地图的ak");

utils.send(HttpMethod.GET, url, params, new RequestCallBack() {

@Override

public void onFailure(HttpException arg0, String result) {

}

@Override

public void onSuccess(ResponseInfo result) {

if (result.statusCode == 0 || result.statusCode == 200) {

}

}

});

}

那么,我们修改一下MainActivity中的代码:

mSearch= GetBaiduSearch.getInstance();

mSearch.setOnResultListener(new OnResultListener() {

@Override

public void reDraw(String result, int error) {

Log.v("yul_wu", result);

}

});

button.setOnClickListener(new OnClickListener() {

@Override

public void onClick(View v) {

mSearch.searchGeoTable();

}

});

可以看到,我们做完这些修改以后,是不是就跟百度SDK中的代码操作是不是一样了。

那么,接下来,我们怎么把通过http请求的数据回传给我们的调用处呢。最初的想法,我是在检索类中,我们定义一个变量result,然后再上面的OnSuccess中,去设置result的值,改完之后的代码如下:

private void setResult(String str){

this.result = str;

}

public void setOnResultListener(OnResultListener listener) {

this.listener = listener;

this.listener.reDraw(this.result, 0);

}

public void SearchGeoTable() {

HttpUtils utils = new HttpUtils();

RequestParams params = new RequestParams();

String url = "http://api.map.baidu.com/geodata/v3/geotable/list";

params.addQueryStringParameter("ak", "你的百度AK");

utils.send(HttpMethod.GET, url, params, new RequestCallBack() {

@Override

public void onFailure(HttpException arg0, String result) {

}

@Override

public void onSuccess(ResponseInfo result) {

if (result.statusCode == 0 || result.statusCode == 200) {

setResult(result.result);

}

}

});

}

也就是我们在网络请求中,设置了result的值,然后再回传。接下来我们运行,发现,报空指针了.....



看这个错误日志,是说Log里的result是空的,而且,在我还没有点击按钮的时候,就已经报错了。这是为什么呢。其实,是我自己发蒙了。因为设置监听函数的时候,已经执行了redeaw这个方法了。所以,这个方法不能放在这里。那应该放在哪呢。(当然放在onSuccess方法中阿,这个简单你都不知道,博主是不是傻了。)

其实,我真是傻了。因为,代码写太多,脑子都乱掉了。我们再做一下修改,把redraw方法放在网络请求成功后的方法里,把没用的删掉,就成这样了。

public class GetBaiduSearch {

OnResultListener listener = null;

private static GetBaiduSearch baiduSearch;

private GetBaiduSearch() {

}

/**

* 取得实例

*

* @return

*/

public static GetBaiduSearch getInstance() {

if (baiduSearch == null) {

synchronized (GetBaiduSearch.class) {

baiduSearch = new GetBaiduSearch();

}

}

return baiduSearch;

}

/**

* 设置监听

* @param listener

*/

public void setOnResultListener(OnResultListener listener) {

this.listener = listener;

}

public void searchGeoTable() {

HttpUtils utils = new HttpUtils();

RequestParams params = new RequestParams();

String url = "http://api.map.baidu.com/geodata/v3/geotable/list";

params.addQueryStringParameter("ak", "你的百度AK");

utils.send(HttpMethod.GET, url, params, new RequestCallBack() {

@Override

public void onFailure(HttpException arg0, String result) {

}

@Override

public void onSuccess(ResponseInfo result) {

if (result.statusCode == 0 || result.statusCode == 200) {

listener.reDraw(result.result, 0);

}

}

});

}

}

然后,我们运行程序,程序没有报错吧。接下来,我们点击Buttou,是不是结果成功返回了!


如果做过百度LBS云开发的朋友,看到这个字符串肯定不陌生哈。说明我们的数据已经成功能过我们的回调函数返回了。这样,我们就完成了一个简单的模型。

接下来,我们再做一步封装,把参数封装成百度SDK的样子,

public class BaiduSearchOptions {

private String ak;

public BaiduSearchOptions() {

}

public BaiduSearchOptions ak(String ak) {

this.ak = ak;

return this;

}

}

这样,我们再把检索类中的方法修改一下,

public void SearchGeoTable(BaiduSearchOptions option) {

HttpUtils utils = new HttpUtils();

//通过反射,取得参数里的值和字段名

RequestParams params = option.reflect(option);

String url = "http://api.map.baidu.com/geodata/v3/geotable/list";

utils.send(HttpMethod.GET, url, params, new RequestCallBack() {

@Override

public void onFailure(HttpException arg0, String result) {

}

@Override

public void onSuccess(ResponseInfo result) {

if (result.statusCode == 0 || result.statusCode == 200) {

listener.reDraw(result.result, 0);

}

}

});

}

}

看一下我们的reflect方法:

public RequestParams reflect(Object obj) {

RequestParams params = new RequestParams();

String key =null;

String value =null;

if (obj == null)

params= null;

Field[] fields = obj.getClass().getDeclaredFields();

for (int j = 0; j < fields.length; j++) {

fields[j].setAccessible(true);

// 字段名

key = fields[j].getName();

System.out.print(fields[j].getName() + ":");

// 字段值

if (fields[j].getType().getName()

.equalsIgnoreCase("java.lang.String")) {

try {

value= fields[j].get(obj).toString();

System.out.print(fields[j].get(obj) + "    ");

} catch (Exception e) {

e.printStackTrace();

}

}

}

params.addQueryStringParameter(key,value);

return params;

}

那么,我们检索的时候,是不是就变成这样了:

BaiduSearchOptions options = new BaiduSearchOptions().ak("你的百度AK");

mSearch.SearchGeoTable(options);

现在看一下,是不是就跟百度SDK检索的操基本一样了。这样,以后业务层调用的时候,只要给出参数,我们就可以把结果回传给调用者。调用于就不用去考虑Http请求那些东西了。当然,你还可以根据不同的业务,再次把返回的result进行封装,这里我就不多做介绍了。

好了,那么关于回调函数的内容就到这里了!

---------------------分割线----------------------------

另外附上关于SDK开发一位大神的总结:

作者:neevek

链接:https://www.zhihu.com/question/36520512/answer/68177050

来源:知乎

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

SDK 一般以一个或多个 jar 文件加上资源文件的形式对外开放,当然也可以把资源文件打包到 jar 中,这样别人接入你的 SDK 所需要的配置就非常简单了,只需要引入一个 jar 文件就可以了。 (我自己用的是第二种方式,资源文件包括图片、XML文件、动态链接库等)。

多数 SDK 都是需要 UI 的,有两种方式在一个 jar 里面实现 UI。第一种是用 Java 实现所有界面布局,很多 SDK 这样做,但这并不是最好的方法,这样实现起来麻烦,一个简单的界面要写很多的代码,维护肯定也不简单。第二种方法是像平常写 app 一样用 XML 写布局,然后用 aapt 编译这些 XML(不同于代码编译,实际上是把 XML 转换成另外一种 Android 的布局系统更容易解析的一种格式 :They call it compiled XML),在代码中通过反射使用 XmlPullParser,inflate 出 XML 中的布局,这样比 Java 实现要简单得多,代码也更容易维护。

目前应该也可以把 SDK 打包成 aar,那样应该就可以完全跟开发 app 一样了,不需要像上面那么复杂,但我没试过,不太清楚。目前市场上的 SDK 大多都还是 jar 的形式。

剩下的是一些个人实践中的一些总结,不一定对,仅供参考:

1. 暴露的接口尽可能少,最大程度减少 SDK 接入方需要了解的细节。

2. 统一所有接口调用方式,我实现过的一个 SDK 其中的一个接口签名:SDK.exec(Context, Action, Callback),Action 可以有每个不同业务的不同参数。

---------------------转载结束,下面关于SDK开发我自己的总结---------------------------

我觉的SDK开发经常用到的东西有如下:

1. java中的泛型

2.反射

3.接口回调

3.设计模式(常用的单例,工厂,观察者,代理,适配器,装饰器)

4.就是写UI布局(如上面大神说的)

你可能感兴趣的:(Android中封装自己的SDK示例和自己的总结)