Xutils3的更详细的使用

2019-10-14 13:36:19

参考地址 Xutils和Xutils3的基本使用

 这几天把公司的电商软件第一版完成之后,打算把项目重新优化并排版一下,让软件整体看起来更加美观。作为一名优(lan)雅(duo)的程序猿,一定要时刻保持优雅的情操,比如FindViewById()和SetOnclick等方法,用起来是真心不优雅,打算使用Xutils3的注解功能,之前一直使用的是Xutils,但是Github上面的Xutils已经很久没有更新了,而且Xutils对Android6.0的兼容性不是特别好,于是搜了一下,决定把Xutils和Xutils3的基本知识整理一下方便更多的人学习使用。

一、Xutils简介: Xutils是前几年比较火得一个开源框架,Xutils包含了很多实用的工具类,支持大文件上传,且有更全面的http请求协议支持,拥有灵活的Orm,支持事件注解且不受代码混淆影响。Xutils刚问世的时候真是很多屌丝程序猿的福利啊,集成了这么多功能强大的模块,大家果然都爱不释手,但是随着Android版本的升级和人们开发软件思想的转变,Xutils也暴露出了很多问题,现在用Xutils的已经不是特别的多了,如果想使用的话,推荐使用比较新的Xutils3。

二、Xutils的主要模块:不管是老版本的Xutils还是新版本的Xutils3,总的来说,他们都分了四个重要的模块:ViewUtils,HttpUtils,BitmapUtils,DbUtils。附带的还有一个LogUtils,可以用来输出Log日志,用起来也是蛮方便的。不管是老版本的Xutils还是新版本的Xutils3,总的来说,他们都分了四个重要的模块:ViewUtils,HttpUtils,BitmapUtils,DbUtils。附带的还有一个LogUtils,可以用来输出Log日志,用起来也是蛮方便的。

三、首先要说的是老版本Xutils的4个模块:

ViewUtils模块     主要用来代替findViewById()的,也可以用来给控件添加点击事件

  // xUtils的view注解要求必须提供id,以使代码混淆不受影响。

@ViewInject(R.id.textView)

TextView textView;

 

//@ViewInject(vale=R.id.textView, parentId=R.id.parentView)

//TextView textView;

 

@ResInject(id = R.string.label, type = ResType.String)

private String label;

 

// 取消了之前使用方法名绑定事件的方式,使用id绑定不受混淆影响

// 支持绑定多个id @OnClick({R.id.id1, R.id.id2, R.id.id3})

// or @OnClick(value={R.id.id1, R.id.id2, R.id.id3}, parentId={R.id.pid1, R.id.pid2, R.id.pid3})

// 更多事件支持参见ViewCommonEventListener类和包com.lidroid.xutils.view.annotation.event。

@OnClick(R.id.test_button)

public void testButtonClick(View v) { // 方法签名必须和接口中的要求一致

...

}

...

//在Activity中注入:

@Override

public void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.main);

ViewUtils.inject(this); //注入view和事件

...

textView.setText("some text...");

...

}

//在Fragment中注入:

@Override

public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {

View view = inflater.inflate(R.layout.bitmap_fragment, container, false); // 加载fragment布局

ViewUtils.inject(this, view); //注入view和事件

...

}

//在PreferenceFragment中注入:

public void onActivityCreated(Bundle savedInstanceState) {

super.onActivityCreated(savedInstanceState);

ViewUtils.inject(this, getPreferenceScreen()); //注入view和事件

...

}

// 其他重载

// inject(View view);

// inject(Activity activity)

// inject(PreferenceActivity preferenceActivity)

// inject(Object handler, View view)

// inject(Object handler, Activity activity)

// inject(Object handler, PreferenceGroup preferenceGroup)

// inject(Object handler, PreferenceActivity preferenceActivity)

HttpUtils模块    主要用来进行各种常见网络请求和文件下载以及上传。

  /**普通get方法**/

HttpUtils http = new HttpUtils();

http.send(HttpRequest.HttpMethod.GET,

"http://www.lidroid.com",

new RequestCallBack<String>(){

@Override

public void onLoading(long total, long current, boolean isUploading) {

testTextView.setText(current + "/" + total);

}

 

@Override

public void onSuccess(ResponseInfo<String> responseInfo) {

textView.setText(responseInfo.result);

}

 

@Override

public void onStart() {

}

 

@Override

public void onFailure(HttpException error, String msg) {

}

});

/**使用HttpUtils上传文件 或者 提交数据 到服务器(post方法)**/

RequestParams params = new RequestParams();

params.addHeader("name", "value");

params.addQueryStringParameter("name", "value");

 

// 只包含字符串参数时默认使用BodyParamsEntity,

// 类似于UrlEncodedFormEntity("application/x-www-form-urlencoded")。

params.addBodyParameter("name", "value");

 

// 加入文件参数后默认使用MultipartEntity("multipart/form-data"),

// 如需"multipart/related",xUtils中提供的MultipartEntity支持设置subType为"related"。

// 使用params.setBodyEntity(httpEntity)可设置更多类型的HttpEntity(如:

// MultipartEntity,BodyParamsEntity,FileUploadEntity,InputStreamUploadEntity,StringEntity)。

// 例如发送json参数:params.setBodyEntity(new StringEntity(jsonStr,charset));

params.addBodyParameter("file", new File("path"));

 

HttpUtils http = new HttpUtils();

http.send(HttpRequest.HttpMethod.POST,

"uploadUrl....",

params,

new RequestCallBack<String>() {

 

@Override

public void onStart() {

testTextView.setText("conn...");

}

 

@Override

public void onLoading(long total, long current, boolean isUploading) {

if (isUploading) {

testTextView.setText("upload: " + current + "/" + total);

} else {

testTextView.setText("reply: " + current + "/" + total);

}

}

 

@Override

public void onSuccess(ResponseInfo<String> responseInfo) {

testTextView.setText("reply: " + responseInfo.result);

}

 

@Override

public void onFailure(HttpException error, String msg) {

testTextView.setText(error.getExceptionCode() + ":" + msg);

}

});

/**使用HttpUtils下载文件:支持断点续传,随时停止下载任务,开始任务**/

HttpUtils http = new HttpUtils();

HttpHandler handler = http.download("http://apache.dataguru.cn/httpcomponents/httpclient/source/httpcomponents-client-4.2.5-src.zip",

"/sdcard/httpcomponents-client-4.2.5-src.zip",

true, // 如果目标文件存在,接着未完成的部分继续下载。服务器不支持RANGE时将从新下载。

true, // 如果从请求返回信息中获取到文件名,下载完成后自动重命名。

new RequestCallBack<File>() {

 

@Override

public void onStart() {

testTextView.setText("conn...");

}

 

@Override

public void onLoading(long total, long current, boolean isUploading) {

testTextView.setText(current + "/" + total);

}

 

@Override

public void onSuccess(ResponseInfo<File> responseInfo) {

testTextView.setText("downloaded:" + responseInfo.result.getPath());

}

 

 

@Override

public void onFailure(HttpException error, String msg) {

testTextView.setText(msg);

}

});

//调用cancel()方法停止下载

handler.cancel();

BitmapUtils模块    主要用来加载ImageView的图片资源,可以本地图片,也可以是网络图片

   BitmapUtils bitmapUtils = new BitmapUtils(this);

 

 

   // 加载网络图片

   bitmapUtils.display(testImageView, "http://bbs.lidroid.com/static/image/common/logo.png");

 

 

   // 加载本地图片(路径以/开头, 绝对路径)

   bitmapUtils.display(testImageView, "/sdcard/test.jpg");

 

 

   // 加载assets中的图片(路径以assets开头)

   bitmapUtils.display(testImageView, "assets/img/wallpaper.jpg");

 

 

   // 使用ListView等容器展示图片时可通过PauseOnScrollListener控制滑动和快速滑动过程中时候暂停加载图片

   listView.setOnScrollListener(new PauseOnScrollListener(bitmapUtils, false, true));

   listView.setOnScrollListener(new PauseOnScrollListener(bitmapUtils, false, true, customListener));

DbUtils模块    主要用来操作数据库的

  DbUtils db = DbUtils.create(this);

User user = new User(); //这里需要注意的是User对象必须有id属性,或者有通过@ID注解的属性

user.setEmail("wyouflf@qq.com");

user.setName("wyouflf");

db.save(user); // 使用saveBindingId保存实体时会为实体的id赋值

 

// 查找

Parent entity = db.findById(Parent.class, parent.getId());

List<Parent> list = db.findAll(Parent.class);//通过类型查找

 

Parent Parent = db.findFirst(Selector.from(Parent.class).where("name","=","test"));

 

// IS NULL

Parent Parent = db.findFirst(Selector.from(Parent.class).where("name","=", null));

// IS NOT NULL

Parent Parent = db.findFirst(Selector.from(Parent.class).where("name","!=", null));

 

// WHERE id<54 AND (age>20 OR age<30) ORDER BY id LIMIT pageSize OFFSET pageOffset

List<Parent> list = db.findAll(Selector.from(Parent.class)

   .where("id" ,"<", 54)

   .and(WhereBuilder.b("age", ">", 20).or("age", " < ", 30))

   .orderBy("id")

   .limit(pageSize)

   .offset(pageSize * pageIndex));

 

// op为"in"时,最后一个参数必须是数组或Iterable的实现类(例如List等)

Parent test = db.findFirst(Selector.from(Parent.class).where("id", "in", new int[]{1, 2, 3}));

// op为"between"时,最后一个参数必须是数组或Iterable的实现类(例如List等)

Parent test = db.findFirst(Selector.from(Parent.class).where("id", "between", new String[]{"1", "5"}));

 

DbModel dbModel = db.findDbModelAll(Selector.from(Parent.class).select("name"));//select("name")只取出name列

List<DbModel> dbModels = db.findDbModelAll(Selector.from(Parent.class).groupBy("name").select("name", "count(name)"));

 

List<DbModel> dbModels = db.findDbModelAll(sql); // 自定义sql查询

db.execNonQuery(sql) // 执行自定义sql

LogUtils工具类 打印输出日志 

  // 自动添加TAG,格式: className.methodName(L:lineNumber)

// 可设置全局的LogUtils.allowD = false,LogUtils.allowI = false...,控制是否输出log。

// 自定义log输出LogUtils.customLogger = new xxxLogger();

LogUtils.d("打印测试");

ok,关于Xutils的使用就简单地介绍这么多,ViewUtils,HttpUtils,BitmapUtils,DbUtils这4个模块的使用其实还是有一定规律的,除了DbUtils使用create创建对象外,其余的3个模块都是使用new来创建对象的。



四、接下来再说一下Xutils3的4个模块:


xUtils3变化较多所以建立了新的项目不在旧版(github.com/wyouflf/xUtils)上继续维护, 相对于旧版本:

HTTP实现替换HttpClient为UrlConnection, 自动解析回调泛型, 更安全的断点续传策略.

支持标准的Cookie策略, 区分domain, path...

事件注解去除不常用的功能, 提高性能.

数据库api简化提高性能, 达到和greenDao一致的性能.

图片绑定支持gif(受系统兼容性影响, 部分gif文件只能静态显示), webp; 支持圆角, 圆形, 方形等裁剪, 支持自动旋转..

以上是官方的说明,下面来看具体使用。


  Xutils3的使用方式和Xutils有很大的改变,所以请新建一个工程,按照下面的步骤操作:


首先新建一个Application类,在里面对Xutils3做初始化工作:

  // 在application的onCreate中初始化

@Override

public void onCreate() {

super.onCreate();

x.Ext.init(this);

x.Ext.setDebug(BuildConfig.DEBUG); // 是否输出debug日志, 开启debug会影响性能.

}

xutils3和Xutils使用方式有一些区别,以前是使用new得到各模块的对象,现在使用:x.模块名()得到对象,回调接口也使用统一的一个,回调接口支持泛型。


ViewUtils的使用基本没有变,只是绑定点击事件的时候有些出入:

  //使用ViewUtils绑定控件

@ViewInject(R.id.txtv)

private TextView txtv;

@ViewInject(R.id.imgv)

private ImageView imgv;

绑定事件:


  /**

* 使用ViewUtils设置按钮的点击事件,方法必须要私有化, 参数格式和type的参数一致,为了混淆方便,方法名要以Event或者Click结尾

* type可以不写,默认是点击事件类型

*/

@Event(value = R.id.btn, type = View.OnClickListener.class)

private void btnClick(View view) {}

HttpUtlis的使用:

  // 请求参数

RequestParams params = new RequestParams("http://www.baidu.com");

x.http().get(params, new CommonCallback<String>() {

 

@Override

public void onCancelled(CancelledException arg0) {

 

}

 

@Override

public void onError(Throwable arg0, boolean arg1) {

 

}

 

@Override

public void onFinished() {

 

}

 

@Override

public void onSuccess(String arg0) {

// 成功下载,显示到txtv上面

txtv.setText(arg0);

}

});

BitmapUtils的使用:

  // 设置加载图片的参数

ImageOptions options = new ImageOptions.Builder()

// 是否忽略GIF格式的图片

.setIgnoreGif(false)

// 图片缩放模式

.setImageScaleType(ScaleType.CENTER_CROP)

// 下载中显示的图片

.setLoadingDrawableId(R.drawable.ic_launcher)

// 下载失败显示的图片

.setFailureDrawableId(R.drawable.ic_launcher)

// 得到ImageOptions对象

.build();

// 加载图片

x.image().bind(imgv, imagUrl, options, new CommonCallback<Drawable>() {

@Override

public void onSuccess(Drawable arg0) {

LogUtil.e("下载成功");

}

 

@Override

public void onFinished() {

LogUtil.e("下载完成");

}

 

@Override

public void onError(Throwable arg0, boolean arg1) {

 

LogUtil.e("下载出错," + arg0.getMessage());

}

 

@Override

public void onCancelled(CancelledException arg0) {

LogUtil.e("下载取消");

}

});

// 加载本地图片

// x.image().bind(imgv, "assets://test.gif", options);

// x.image().bind(iv_big_img, new

// File("/sdcard/test.gif").toURI().toString(), imageOptions);

// x.image().bind(iv_big_img, "/sdcard/test.gif", imageOptions);

// x.image().bind(iv_big_img, "file:///sdcard/test.gif", imageOptions);

// x.image().bind(iv_big_img, "file:/sdcard/test.gif", imageOptions);

DbUtils的使用:

先建一个类,用来生成数据库中的表,这里新建一个StudentInfo类(注意里面的@标签的使用,使用这些标签生成表的结构信息)内容如下:



@Table(name = "info")

public class StudentInfo {

@Column(name = "id", isId = true)

private int id;

@Column(name = "name")

private String name;

@Column(name = "age")

private int age;

 

public int getId() {

return id;

}

 

public void setId(int id) {

this.id = id;

}

 

public String getName() {

return name;

}

 

public void setName(String name) {

this.name = name;

}

 

public int getAge() {

return age;

}

 

public void setAge(int age) {

this.age = age;

}

 

}

然后:


DbManager.DaoConfig daoConfig = new DaoConfig()

// 数据库的名字

.setDbName("SudentInfo")

// 保存到指定路径

// .setDbDir(new

// File(Environment.getExternalStorageDirectory().getAbsolutePath()))

// 数据库的版本号

.setDbVersion(1)

// 数据库版本更新监听

.setDbUpgradeListener(new DbUpgradeListener() {

@Override

public void onUpgrade(DbManager arg0, int arg1, int arg2) {

LogUtil.e("数据库版本更新了!");

}

});

DbManager manager = x.getDb(daoConfig);

try {

//创建对象,用来生成表

StudentInfo info = new StudentInfo();

info.setAge(16);

info.setName("小花");

//创建表

manager.saveOrUpdate(info);

} catch (DbException e) {

e.printStackTrace();

}




  • 2020-11-22 21:12:30

    Dependency Injection with Dagger2,Fragment

    標註@Provides的method若有parameter的話,Dagger會找出其擁有的該型態物件來使用。我們在Module內新增了DataModel將其列入Dagger的管理下,接著在provideFactory()增加parameter變成provideFactory(DataModel dataModel),Dagger就會找出其管理的DataModel給provideFactory使用。

  • 2020-11-22 22:58:52

    Android LiveData Transformations

    有时候有这样的需求,需要在LiveData将变化的数据通知给观察者前,改变数据的类型;或者是返回一个不一样的LiveData。

  • 2020-11-22 23:00:16

    androidx中的lifecycle组件

    Lifecycle-aware components生命周期感知组件执行操作,以响应另一个组件生命周期状态的更改,例如Activity和Fragment。这些组件可以帮助您生成更有组织、更容易维护的轻量级代码。

  • 2020-11-22 23:02:50

    Android数据存储之DataBase的Room

    Room是Google在AndroidX中提供的一个ORM(Object Relational Mapping,对象关系映射)库。它是在SQLite上提供的一个抽象层,可以使用SQLite的全部功能,同时可以更好更便捷流畅地访问数据库。(关于AndroidX可以参考

  • 2020-11-22 23:04:39

    Android组件 LiveData与MutableLiveData教程

    LiveData与ViewMode是经常搭配在一起使用的,但是为了不太混乱,我还是拆分开来说明,此篇博客只讲解 LiveData 与 MutableLiveData的概念与使用方式(但是会涉及到ViewMode的部分代码).

  • 2020-11-22 23:14:52

    Dagger 2 在 Android 上的用法

    在前面的文章我们介绍了Dagger2 中的大部分注解的使用,接下来我们从源码角度分析下第一篇文章中例子的原理。

  • 2020-11-22 23:18:59

    Android开发从Dagger2迁移至Kodein的感受

    最近个人在尝试构建 Kotlin版本 的Android MVVM开发框架,在依赖注入框架的选型上,我最终选择了 Kodein 。这是一个非常轻量级的DI框架,相比于配置繁琐的Dagger(繁琐的配置也是导致Dagger学习成本一直居高不下的原因!),它的配置过程更清晰且简单,并且,这个库的源码也是 Kotlin 的。

  • 2020-11-22 23:25:56

    Dagger2源码解析inject过程

    添加inject后,通过编译生成的DaggerMainComponent类来导入,说明编译以后生成了一些类,那到底生成了什么类呢。 Module和Component又是什么,该怎么里理解 在这篇文章后里将一一讨论。