xUtils简介

xUtils是基于Afinal开发的目前功能比较完善的一个Android开源框架,最近又发布了xUtil3.0,在增加新功能的同时又提高了框架的性能,下面来看看官方(https://github.com/wyouflf/xUtils3)对xUtils3的介绍:

  • xUtils包含了很多实用的android工具;
  • xUtils支持超大文件(超过2G)上传,更全面的http请求协议支持(11种谓词),拥有更加灵活的ORM,更多的事件注解支持且不受混淆影响;
  • xUtils 最低兼容Android 4.0 (api level 14);
  • xUtils3变化较多所以建立了新的项目不在旧版(github.com/wyouflf/xUtils)上继续维护, 相对于旧版本:
    • HTTP实现替换HttpClient为UrlConnection, 自动解析回调泛型, 更安全的断点续传策略;
    • 支持标准的Cookie策略, 区分domain, path;
    • 事件注解去除不常用的功能, 提高性能;
    • 数据库api简化提高性能, 达到和greenDao一致的性能;
    • 图片绑定支持gif(受系统兼容性影响, 部分gif文件只能静态显示), webp; 支持圆角, 圆形, 方形等裁剪, 支持自动旋转。

目前xUtils主要有四大模块:

ViewUtils模块:

  • android中的ioc(控制倒转)框架,完全注解方式就可以进行UI,资源和事件绑定;
  • 新的事件绑定方式,使用混淆工具混淆后仍可正常工作;
  • 目前支持常用的20种事件绑定,参见ViewCommonEventListener类和包com.lidroid.xutils.view.annotation.event。

HttpUtils模块:

  • 支持同步,异步方式的请求;
  • 支持大文件上传,上传大文件不会oom;
  • 支持GET,POST,PUT,MOVE,COPY,DELETE,HEAD,OPTIONS,TRACE,CONNECT请求;
  • 下载支持301/302重定向,支持设置是否根据Content-Disposition重命名下载的文件;
  • 返回文本内容的请求(默认只启用了GET请求)支持缓存,可设置默认过期时间和针对当前请求的过期时间。

BitmapUtils模块:

  • 加载bitmap的时候无需考虑bitmap加载过程中出现的oom和android容器快速滑动时候出现的图片错位等现象;
  • 支持加载网络图片和本地图片;
  • 内存管理使用lru算法,更好的管理bitmap内存;
  • 可配置线程加载线程数量,缓存大小,缓存路径,加载显示动画等…

DbUtils模块:

  • android中的orm(对象关系映射)框架,一行代码就可以进行增删改查;
  • 支持事务,默认关闭;
  • 可通过注解自定义表名,列名,外键,唯一性约束,NOT NULL约束,CHECK约束等(需要混淆的时候请注解表名和列名);
  • 支持绑定外键,保存实体时外键关联实体自动保存或更新;
  • 自动加载外键关联实体,支持延时加载;
  • 支持链式表达查询,更直观的查询语义,参考下面的介绍或sample中的例子。

项目中快速配置xUtils3

Eclipse用户导入最新jar包和so文件,下载链接
添加网络访问权限

<uses-permission android:name="android.permission.INTERNET"/>

在Application中初始化xUtils

    @Overridepublic void onCreate() {super.onCreate();x.Ext.init(this);x.Ext.setDebug(true); //是否输出debug日志,开启debug会影响性能。}

xUtils3功能介绍

ViewUtils注解模块的使用

  • 完全注解方式就可以进行UI绑定和事件绑定。
  • 无需findViewById和setClickListener等。

1)Activity的注解的使用如下:

@ContentView(R.layout.activity_main)
public class MainActivity extends Activity {@ViewInject(R.id.button1)private Button button1;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);x.view().inject(this);...}
}

2)Fragment的注解的使用如下:

@ContentView(R.layout.fragment_http)
public class HttpFragment extends Fragment {@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {return x.view().inject(this, inflater, container);}@Overridepublic void onViewCreated(View v, Bundle savedInstanceState) {super.onViewCreated(v, savedInstanceState);}
}

3)ViewHolder的注解的使用如下:

    public class MyAdapter extends BaseAdapter{ImageOptions options = new ImageOptions.Builder().setFadeIn(true).build();@Overridepublic int getCount() {return listUrl.size();}@Overridepublic String getItem(int position) {return listUrl.get(position);}@Overridepublic long getItemId(int position) {return position;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {ViewHolder holder=null;if(convertView == null){convertView=LayoutInflater.from(BitmapUtilsActivity.this).inflate(R.layout.itemone, null);holder = new ViewHolder();x.view().inject(holder,convertView);convertView.setTag(holder);} else {holder=(ViewHolder) convertView.getTag();}x.image().bind(holder.image, getItem(position), options);           return convertView;}class ViewHolder{@ViewInject(R.id.image)private ImageView image;}

4)为按钮设置点击事件

@ViewInject(R.id.button1)
private Button button1;
...
@Override
protected void onCreate(Bundle savedInstanceState) {...
}
/*** 用注解的方式为按钮添加点击事件,方法声明必须为private* type默认View.OnClickListener.class,故此处可以简化不写*/
@Event(type = View.OnClickListener.class,value = R.id.button1)
private void testHttpUtils(View v){Intent it = new Intent(MainActivity.this, HttpUtilsActivity.class);startActivity(it);
}
/*** 长按事件*/
@Event(type = View.OnLongClickListener.class,value = R.id.button2)
private boolean testOnLongClickListener(View v){Toast.makeText(this,"testOnLongClickListener",Toast.LENGTH_SHORT).show();return true;
}/*** 同时为多个按钮添加点击事件,type用默认*/
@Event(value = {R.id.button1, R.id.button2, R.id.button3})private void getEvent(View view){switch(view.getId()){case R.id.button1:Intent it1 = new Intent(MainActivity.this, HttpUtilsActivity.class);startActivity(it1);break;case R.id.button2:Intent it2 = new Intent(MainActivity.this, BitmapUtilsActivity.class);startActivity(it2);break;case R.id.button3:Intent it3 = new Intent(MainActivity.this, DbUtilsActivity.class);startActivity(it3);break;}}

注意:
1. 使用IOC必须全部为私有,不然无效,这里就不做演示了,不信你可以把用到IOC框架的注解的成员变量及方法全部换成public ,那么全部会无效,当然除了ContentView例外。
2. 所有用到IOC成员变量,使用的时候,必须在x.view().inject(this)后,如果写在前面,那么程序会崩溃。

HttpUtils网络模块的使用

xUtils3网络模块大大方便了在实际开发中网络模块的开发,xUtils3网络模块大致包括GET请求、POST请求、上传文件、下载文件等功能,下面将做一一说明:
1)GET请求

@Event(R.id.get)
private void get(View v){final ProgressDialog progressDialog = new ProgressDialog(getActivity());progressDialog.setMessage("请稍候...");RequestParams params = new RequestParams("http://weather.51wnl.com/weatherinfo/GetMoreWeather");params.addQueryStringParameter("cityCode","101020100");params.addQueryStringParameter("weatherType","0");Callback.Cancelable cancelable = x.http().get(params, new Callback.CommonCallback<String>() {@Overridepublic void onSuccess(String result) {Toast.makeText(HttpUtilsActivity.this, result, Toast.LENGTH_SHORT).show();progressDialog.cancel();}//请求异常后的回调方法@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}//主动调用取消请求的回调方法@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {progressDialog.cancel();}});//主动调用取消请求
//    cancelable.cancel();
}

2)POST请求

@Event(R.id.post)
private void post(View v){RequestParams params = new RequestParams("http://61.129.89.191/SoarAPI/api/SoarTopic");params.addParameter("topicId","1002");params.addParameter("maxReply","-1");params.addParameter("reqApp","1"); //添加请求参数params.addBodyParameter("username","abc"); //添加一个请求体params.addHeader("head","android"); //添加一个请求头x.http().post(params, new Callback.CommonCallback<String>() {@Overridepublic void onSuccess(String result) {MainActivity.showlog("onSuccess-->result="+result);Toast.makeText(HttpUtilsActivity.this, result, Toast.LENGTH_SHORT).show();         }@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}});
}

3)上传文件

    @Event(R.id.btn_test3)private void upload(View v){RequestParams params = new RequestParams("http://192.168.1.112:8080/TopNews_WebProject/UploadFileServlet");params.setMultipart(true);//这里可以进行多文件上传params.addBodyParameter("file",new File("/sdcard/temp/file.jpg"));params.addBodyParameter("photo",new File("/sdcard/temp/file0.jpg"),"image/jpeg","hx.jpg");//还传入文件类型和新文件名x.http().post(params, new Callback.CommonCallback<String>() {@Overridepublic void onSuccess(String result) {MainActivity.showlog("onSuccess-->result="+result);Toast.makeText(HttpUtilsActivity.this, result, Toast.LENGTH_SHORT).show();}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {MainActivity.showlog("onError-->Throwable="+ex.getMessage());}@Overridepublic void onCancelled(CancelledException cex) {MainActivity.showlog("onCancelled-->CancelledException="+cex.getMessage());}@Overridepublic void onFinished() {MainActivity.showlog("onFinished");}});}

4)下载文件
这里以下载图片为例进行说明(带下载进度提示),图片下载完成后,自动加载到ImageView。

@Event(R.id.download)
private void download(View v){String url = editText.getText().toString();RequestParams params = new RequestParams(url);//自定义保存路径params.setSaveFilePath("/sdcard/temp/");//自动为文件命名params.setAutoRename(true);x.http().post(params, new Callback.ProgressCallback<File>() {@Overridepublic void onSuccess(File result) {try {Bitmap bm = BitmapFactory.decodeStream(new FileInputStream(file));image.setImageBitmap(bm);} catch (FileNotFoundException e) {e.printStackTrace();}}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}//网络请求之前回调@Overridepublic void onWaiting() {}//网络请求开始的时候回调@Overridepublic void onStarted() {}//下载的时候不断回调的方法@Overridepublic void onLoading(long total, long current, boolean isDownloading) {//当前进度和文件总大小progressBar.setMax((int)total);progressBar.setProgress((int)current); }});
}

5)直接返回Json对象
可以看到上面onSuccess回调返回的都是String,你再仔细看一下发现Callback.CommonCallback接口是支持泛型的,那么怎么让他直接返回对象呢?两个步骤:

  • 泛型参数T写成对象类
  • 对象类注解使用parser将返回的Json字符串解析为对象

看对象类bean文件写法:

@HttpResponse(parser = JsonResponseParser.class)
public class Weather implements Serializable {private static final long serialVersionUID = 1L;public WeatherInfo weatherinfo;
}

定义JsonResponseParser解析类:

public class JsonResponseParser implements ResponseParser {//检查服务器返回的响应头信息@Overridepublic void checkResponse(UriRequest request) throws Throwable {}/*** 转换result为resultType类型的对象** @param resultType  返回值类型(可能带有泛型信息)* @param resultClass 返回值类型* @param result      字符串数据* @return* @throws Throwable*/@Overridepublic Object parse(Type resultType, Class<?> resultClass, String result) throws Throwable {return new Gson().fromJson(result, resultClass);}
}

使用:

            RequestParams params = new RequestParams("http://weather.51wnl.com/weatherinfo/GetMoreWeather");params.addQueryStringParameter("cityCode","101020100");params.addQueryStringParameter("weatherType","0");Callback.Cancelable cancelable = x.http().get(params, new Callback.CommonCallback<Weather>() {public void onSuccess(Weather result) {super.onSuccess(result);Toast.makeText(HttpUtilsActivity.this, result.weatherinfo.toString(), Toast.LENGTH_SHORT).show();MainActivity.showlog(result.weatherinfo.toString());}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}});

6)HttpUtils封装
这里我对HttpUtils的四个方法进行了封装,使用起来更加方便

public class XUtil {/*** 发送get请求* @param <T>*/public static <T> Cancelable Get(String url,Map<String,String> map,CommonCallback<T> callback){RequestParams params=new RequestParams(url);if(null!=map){for(Map.Entry<String, String> entry : map.entrySet()){params.addQueryStringParameter(entry.getKey(), entry.getValue());}}Cancelable cancelable = x.http().get(params, callback);return cancelable;}/*** 发送post请求* @param <T>*/public static <T> Cancelable Post(String url,Map<String,Object> map,CommonCallback<T> callback){RequestParams params=new RequestParams(url);if(null!=map){for(Map.Entry<String, Object> entry : map.entrySet()){params.addParameter(entry.getKey(), entry.getValue());}}Cancelable cancelable = x.http().get(params, callback);return cancelable;}/*** 上传文件* @param <T>*/public static <T> Cancelable UpLoadFile(String url,Map<String,File> map,CommonCallback<T> callback){RequestParams params=new RequestParams(url);if(null!=map){for(Entry<String, File> entry : map.entrySet()){params.addBodyParameter(entry.getKey(), entry.getValue());}}params.setMultipart(true);Cancelable cancelable = x.http().get(params, callback);return cancelable;}/*** 下载文件* @param <T>*/public static <T> Cancelable DownLoadFile(String url,String filepath,CommonCallback<T> callback){RequestParams params=new RequestParams(url);//设置断点续传params.setAutoResume(true);
//      params.setAutoRename(true);params.setSaveFilePath(filepath);Cancelable cancelable = x.http().get(params, callback);return cancelable;}
}

使用封装类:

String url="http://weather.51wnl.com/weatherinfo/GetMoreWeather";if (useEnclosure) {Map<String,String> map=new HashMap<String,String>();map.put("cityCode", "101020100");map.put("weatherType", "0");//这里我用的自定义MyCallBack类,实现了Callback.CommonCallback接口,可以不用重写CommonCallback全部的回调方法,更加方便XUtil.Get(url, map, new MyCallBack<Weather>(){@Overridepublic void onSuccess(Weather result) {super.onSuccess(result);Toast.makeText(HttpUtilsActivity.this, result.weatherinfo.toString(), Toast.LENGTH_SHORT).show();MainActivity.showlog(result.weatherinfo.toString());}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {super.onError(ex, isOnCallback);MainActivity.showlog(ex.getMessage());}});

BitmapUtils图片模块的使用

现在我们需要设置两个权限,如下:

<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

xUtils3图片模块,重点在于加载图片的4个bind方法,loadDrawable与loadFile用法和ImageOptions用法。
1)首先获取ImageView控件

@ViewInject(R.id.image01)
ImageView image01;
@ViewInject(R.id.image02)
ImageView image02;
@ViewInject(R.id.image03)
ImageView image03;
...

2)得到网络图片的地址

String[] urls={"http://img.android.com/a.jpg","http://img.android.com/b.jpg""http://img.android.com/c.jpg"...
};

3)xUtils3显示网络图片方法setPic()如下:

private void setPic() {/*** 通过ImageOptions.Builder().set方法设置图片的属性*/ImageOptions options = new ImageOptions.Builder().setFadeIn(true).build(); //淡入效果//ImageOptions.Builder()的一些其他属性://.setCircular(true) //设置图片显示为圆形//.setSquare(true) //设置图片显示为正方形//setCrop(true).setSize(200,200) //设置大小//.setAnimation(animation) //设置动画//.setFailureDrawable(Drawable failureDrawable) //设置加载失败的动画//.setFailureDrawableId(int failureDrawable) //以资源id设置加载失败的动画//.setLoadingDrawable(Drawable loadingDrawable) //设置加载中的动画//.setLoadingDrawableId(int loadingDrawable) //以资源id设置加载中的动画//.setIgnoreGif(false) //忽略Gif图片//.setParamsBuilder(ParamsBuilder paramsBuilder) //在网络请求中添加一些参数//.setRaduis(int raduis) //设置拐角弧度//.setUseMemCache(true) //设置使用MemCache,默认true/*** 加载图片的4个bind方法*/x.image().bind(image01, urls[0]);x.image().bind(image02, urls[1], options);x.image().bind(image03, urls[2], new Callback.CommonCallback<Drawable>() {@Overridepublic void onSuccess(Drawable result) {}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}});x.image().bind(image04, urls[3], options, new Callback.CommonCallback<Drawable>() {@Overridepublic void onSuccess(Drawable result) {}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}});/*** loadDrawable()方法加载图片*/Callback.Cancelable cancelable = x.image().loadDrawable(urls[0], options, new Callback.CommonCallback<Drawable>() {@Overridepublic void onSuccess(Drawable result) {image03.setImageDrawable(result);}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}});//主动取消loadDrawable()方法//cancelable.cancel();/*** loadFile()方法* 应用场景:当我们通过bind()或者loadDrawable()方法加载了一张图片后,* 它会保存到本地文件中,那当我需要这张图片时,就可以通过loadFile()方法进行查找。* urls[0]:网络地址*/x.image().loadFile(urls[0],options,new Callback.CacheCallback<File>(){@Overridepublic boolean onCache(File result) {//在这里可以做图片另存为等操作Log.i("JAVA","file:"+result.getPath()+result.getName());return true; //相信本地缓存返回true}@Overridepublic void onSuccess(File result) {Log.i("JAVA","file");}@Overridepublic void onError(Throwable ex, boolean isOnCallback) {}@Overridepublic void onCancelled(CancelledException cex) {}@Overridepublic void onFinished() {}});
}

3)xUtils3显示本地图片方法:

// 加载本地图片
x.image().bind(img, "assets://test.gif", imageOptions);
x.image().bind(img, "/sdcard/test.gif", imageOptions);
x.image().bind(img, "file:///sdcard/test.gif", imageOptions);
x.image().bind(img, new File("/sdcard/test.gif").toURI().toString(), imageOptions);

DbUtils数据库模块的使用

把以下几个问题搞清楚,基本上就覆盖了DbUtils的全部用法了:

  • 如何创建数据库和删除数据库
  • 如何创建删除一张表
  • 如何对表进行增删查改操作
  • 如何建立一表对一表,多表对一表,多表对多表的外键操作

1)创建数据库和删除数据库
首先进行配置DaoConfig:

/*** DaoConfig配置* /DbManager.DaoConfig daoConfig = new DbManager.DaoConfig()//设置数据库名,默认xutils.db.setDbName("watson.db")//设置是否允许事务,默认true//.setAllowTransaction(true)//设置数据库路径,默认安装程序路径下.setDbDir(new File(Environment.getExternalStorageDirectory().getAbsolutePath()))//设置数据库的版本号.setDbVersion(1)    //设置数据库更新的监听.setDbUpgradeListener(new DbManager.DbUpgradeListener() {@Overridepublic void onUpgrade(DbManager db, int oldVersion, int newVersion) {MainActivity.showlog("数据库版本更新了!oldVersion="+oldVersion+" newVersion="+newVersion);}});

然后获取到DbManager:

DbManager db = x.getDb(daoConfig);

DbManager这个类主要做以下几件事情:

  • 1.getDaoConfig 获取数据库的配置信息
  • 2.getDatabase 获取数据库实例
  • 3.saveBindingId saveOrUpdate save 插入数据的3个方法(保存数据)
  • 4.replace 只有存在唯一索引时才有用 慎重
  • 5.delete操作的4种方法(删除数据)
  • 6.update操作的2种方法(修改数据)
  • 7.find操作6种方法(查询数据)
  • 8.dropTable 删除表
  • 9.addColumn 添加一列
  • 10.dropDb 删除数据库

其实在这个时候数据库就已经创建了,因为我们还没有添加表,所以这时候数据库中只有一个默认的表android_metadata

那么删除数据库的操作呢?

//删除数据库
@Event(R.id.del_db)
private void delDB(View v) throws DbException {db.dropDb();
}

经测试发现此方法并不能删除db文件,只是把数据库中所有的表全部删除。

2)创建表和删除表
创建表之前需要先创建对象,看一下bean文件写法吧:

@Table(name = "studentinfo")
public class StudentInfo {/*** name = "id":数据库表中的一个字段* isId = true:是否是主键* autoGen = true:是否自动增长* property = "NOT NULL":添加约束*/@Column(name = "id", isId = true, autoGen = true, property = "NOT NULL")private int id;@Column(name = "name")private String name;@Column(name = "age")private int age;//默认的构造方法必须写出,如果没有,这张表是创建不成功的public StudentInfo() {}public StudentInfo(String name, int age) {this.name = name;this.age = 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;}@Overridepublic String toString() {return "[id:"+id+" name:"+name+" age:"+age+"]";}
}

一个表对应的实体bean需要注意以下几点:

  • 在类名上面加入@Table标签,标签里面的属性name的值就是以后生成的数据库的表的名字。
  • 实体bean里面的属性需要加上@Column标签,这样这个标签的name属性的值会对应数据库里面的表的字段。
  • 实体bean里面的普通属性,如果没有加上@Column标签就不会在生成表的时候在表里面加入字段。
  • 实体bean中必须有一个主键,如果没有主键,表以后不会创建成功,@Column(name = “id”, isId = true, autoGen = true, property = “NOT NULL”)这个属性name的值代表的是表的主键的标识,isId这个属性代表的是该属性是不是表的主键,autoGen代表的是主键是否是自增长,如果不写autoGen这个属性,默认是自增长的属性。property是添加约束条件,这里不能为空。
  • 默认的构造方法必须写出,如果没有,这张表是创建不成功的

那么具体我们什么时候创建表呢?
数据库里面表的创建时间,只有在你对数据库里面的操作涉及到这张表的操作时,会先判断当前的表是否存在,如果不存在,才会创建一张表,如果存在,才会进行相应的CRUD操作。

3)对表进行增删查改(CRUD)操作
只要我们想进行一张表的CRUD操作,必须先拿到DbManager这个对象。

  • 插入操作
    private void createTable() {StudentInfo student = new StudentInfo("watson", 28);try {db.save(student);//db.saveOrUpdate(student);//db.saveBindingId(student);} catch (DbException e) {e.printStackTrace();}}

3种插入操作所需要的参数都是一个实体bean。save和saveOrUpdate的区别就是当一个实体里面的主键一样时如果使用saveOrUpdate会将当前主键对应的这条数据进行替换,而如果你使用了save就会报错。saveBindingId主要是存进去的数据如果当前表有主键会和主键进行绑定关联。
当你执行完这个方法后,你会看到数据库里面studentinfo表里面多了一条数据.

如果你想在表里一次插入多列呢?当然也是可以做到的:

private void insertInfo() {//用集合向studentinfo表中插入多条数据ArrayList<StudentInfo> studentInfos = new ArrayList<StudentInfo>();studentInfos.add(new StudentInfo("zhangsan", 20));studentInfos.add(new StudentInfo("lisi", 21));studentInfos.add(new StudentInfo("wangwu", 22));studentInfos.add(new StudentInfo("zhaoliu", 23));studentInfos.add(new StudentInfo("qianqi", 24));studentInfos.add(new StudentInfo("sunba", 25));//db.save()方法不仅可以插入单个对象,还能插入集合try {db.save(studentInfos);//db.saveOrUpdate(studentInfos);//db.saveBindingId(studentInfos);} catch (DbException e) {e.printStackTrace();}}

  • 查询操作

1.findById的使用
该方法主要是通过主键的值来进行查找表里面的数据。
需求:查找上方studentinfo表里面id为3的数据

    private void query(){try {StudentInfo student = db.findById(StudentInfo.class, "3");MainActivity.showlog("student:"+student.toString());} catch (DbException e) {e.printStackTrace();}}

2.findFirst的使用
该方法主要是返回当前表里面的第一条数据。
需求:查找上方studentinfo表里面的第一条数据

private void query() {try {StudentInfo student = db.findFirst(StudentInfo.class);MainActivity.showlog("student:"+student.toString());} catch (DbException e) {e.printStackTrace();}}

3.findAll的使用
该方法主要是返回当前表里面的所有数据。
需求:查找studentinfo表里面的所有数据

private void query() {try {List<StudentInfo> students = db.findAll(StudentInfo.class);MainActivity.showlog("students:"+students.toString());} catch (DbException e) {e.printStackTrace();}}

4.selector的使用
该方法主要是用来进行一些特定条件的查找。
需求:查找studentinfo表里面age大于22小于26的学生数据

private void query() {try {List<StudentInfo> students = db.selector(StudentInfo.class).where("age", ">", 22).and("age", "<", "26").findAll();for(StudentInfo student:students){MainActivity.showlog("student:"+student.toString());}} catch (DbException e) {e.printStackTrace();}}

5.findDbModelFirst的使用
该方法返回一个DbModel对象,那么该对象是什么呢?DbModel本质就是一个key为当前表的字段,value为当前某条记录的值的一个HashMap。注意sqlInfo对象创建的构造参数只需要传入一个sql语句即可。
需求:查找person表中第一条数据的那个人的name和age各是多少

    private void query() {try {DbModel model = db.findDbModelFirst(new SqlInfo("select * from studentinfo"));MainActivity.showlog("name:"+model.getString("name"));MainActivity.showlog("age:"+model.getString("age"));} catch (DbException e) {e.printStackTrace(); }}

6.findDbModelAll的用法
该方法的用途就是返回满足sqlInfo信息的所有数据的字段的一个集合。
需求:查找studentinfo表中年龄age大于24里面的所有人的姓名

    private void query() {try {List<DbModel> models = db.findDbModelAll(new SqlInfo("select * from studentinfo where age > 24"));for(DbModel model : models){MainActivity.showlog("name:"+model.getString("name"));}} catch (DbException e) {e.printStackTrace();}}

上面基本把查询的6种方式都说了一遍,当然上面的6种需求不一定完全用上面的查询方法可以查出结果,我这么查询的目的主要是带领大家熟悉一下XUtils3的6种查询方法是如何使用的,会了上面的6种方法,我相信你的查询不会有太大问题,至于复杂的查询无非就是sql语句的基本功力了。

  • 修改操作

修改一共有2种方法:
第一种:
需求:我们把上面studentinfo表中id为1的这条记录的age修改为25岁

    private void update() {try{StudentInfo student = db.findById(StudentInfo.class, 1);student.setAge(25);db.update(student, "age");}catch(Exception e){e.printStackTrace();}}

通过方法,我们知道首先要通过DBManager通过查找的方法先找到id为1的这个实体bean,如果你对里面的哪个字段需要修改,只需要重新set这个属性的值,然后调用DBManager.update方法,第一个参数是需要修改的实体,第二个参数是对应的属性。

第二种:
需求:将studentinfo表中age为25的学生的name都变成mike

    private void update() {try {List<StudentInfo> students = db.findAll(StudentInfo.class);for(StudentInfo student : students){student.setName("mike");db.update(student, WhereBuilder.b("age", "=", "25"), "name");MainActivity.showlog("修改成功");}} catch (Exception e) {e.printStackTrace();}}

修改数据一共就2种方法,基本都是需要一个实体bean对象去进行操作的,上面的第二种方法无非就是在修改数据时,多了一个限制条件,这样修改数据显得灵活一些。
上面第二种update的方法的参数简单介绍一下:
第一个参数:实体bean对象
第二个参数:一个WhereBuilder对象,主要是通过静态b方法去构造一个where条件语句
第三个参数:需要修改的字段名,如果你的需求是修改了2个或者更多个字段,只需要在后面加上相应的参数即可。例如这里每个学生还有分数score这列。而第二种方法我不止修改name还需要修改score统一为100分,参考如下

    private void update() {try {List<StudentInfo> students = db.findAll(StudentInfo.class);for(StudentInfo student : students){student.setName("mike");student.setScore(100);db.update(student, WhereBuilder.b("age", "=", "25"), "name", "score");MainActivity.showlog("修改成功");}} catch (Exception e) {e.printStackTrace();}}
  • 删除操作

1.deleteById的用法
该方法主要是根据表的主键进行单条记录的删除
需求:删除上方person表中id为5的记录

    private void delete() {try {db.deleteById(StudentInfo.class, 5);MainActivity.showlog("删除成功");} catch (DbException e) {e.printStackTrace();}}

2.delete(Object entity)的用法
该方法主要是根据实体bean进行对表里面的一条或多条数据进行删除
需求:删除表中第一条name为mike这条信息的记录

    private void delete() {try {StudentInfo student = db.selector(StudentInfo.class).where("name", "=", "mike").findFirst();db.delete(student);} catch (DbException e) {e.printStackTrace(); }}

3.delete(Class<\?> entityType)
该方法主要是用来删除表格里面的所有数据,但是注意:表还会存在,只是表里面数据没有了

    private void delete() {try {db.delete(StudentInfo.class);MainActivity.showlog("删除成功");           } catch (DbException e) {e.printStackTrace(); }}

4.delete(Class<\?> entityType, WhereBuilder whereBuilder)
该方法主要是根据where语句的条件进行删除操作
需求:将studentinfo表中age为24并且name为qianqi的信息删除

    private void delete() {try {db.delete(StudentInfo.class, WhereBuilder.b("age", "=", "24").and("name", "=", "qianqi"));} catch (DbException e) {e.printStackTrace();}}

5.dropTable(Class<\?> entityType)
该方法是用来删除指定的表

    private void delete() {try {db.dropTable(StudentInfo.class);} catch (DbException e) {e.printStackTrace();}}

6.dropDb()
该方法是用来删除数据库

   private void delete() {try {db.dropDb();} catch (DbException e) {e.printStackTrace();}}

经测试发现此方法并不能删除db文件,只是把数据库中所有的表全部删除。

  • 其他方法

    1.addColumn(Class<\?> entityType, String column)
    需求:在上方表中加入一个score字段
    StudentInfo的实体代码如下:

@Table(name = "studentinfo")
public class StudentInfo {/*** name = "id":数据库表中的一个字段* isId = true:是否是主键* autoGen = true:是否自动增长* property = "NOT NULL":添加约束*/@Column(name = "id", isId = true, autoGen = true, property = "NOT NULL")private int id;@Column(name = "name")private String name;@Column(name = "age")private int age;@Column(name = "score")private int score;public StudentInfo(String name, int age) {this.name = name;this.age = age;}//默认的构造方法必须写出,如果没有,这张表是创建不成功的public StudentInfo() {}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;}public int getScore() {return score;}public void setScore(int score) {this.score = score;}@Overridepublic String toString() {return "[id:"+id+" name:"+name+" age:"+age+" score:"+score+"]";}
}
private void addColumn() {try {db.addColumn(StudentInfo.class, "score");} catch (DbException e) {e.printStackTrace();}}

执行完addColumn方法,我们看到studentinfo表里面多了一个score字段。

总结:
上面主要介绍了XUtils3的数据库模块,包括如何创建数据库,如何创建表,如何给表进行添加一列,如何对表进行增删查改的操作。说了这么多,相信大家肯定对XUtils3的数据库模块有了一个基本的理解,至于一表对一表,多表对一表,多表对多表等等这类需求,无非就是在某个表里面加入一个字段,或者创建一个第三方表用来维护表与表之间的关系,这种类型的例子我就不举例说明了,原因是那些需求都离不开上面的增删查改的方法,我相信你只要把上面的方法完全会用,你的XUtils3的数据库模块的基本使用就不会有问题了。

Demo下载地址

Android xUtils框架解析相关推荐

  1. android xutils框架缓存图片,xUtils框架的使用

    xUtils简介 xUtils 包含了很多实用的android工具,xUtils 源于Afinal框架,对Afinal进行了大量重构,使得xUtils支持大文件上传,更全面的http请求协议支持,拥有 ...

  2. Android OkHttp框架解析

    Okhttp是由Sqare公司开发的开源网络访问库,是目前比较火的网络框架, 它处理了很多网络疑难杂症:会从很多常用的连接问题中自动恢复.如果你的服务器配置了多个IP地址,当第一个IP连接失败的时候, ...

  3. Android Watchdog框架解析、应用与改造(上)

    简言: frameworks/base/services/java/com/android/server/ 系统框架服务目录下,可以看到名为Watchdog.java文件,这是一个软件看门狗的实现,其 ...

  4. Android使用XUtils框架上传照片(一张或多张)和文本,server接收照片和文字(无乱码)...

    Android上传图片,这里我使用了如今比較流行的XUtils框架.该框架能够实现文件上传.文件下载.图片缓存等等,有待研究. 以下是Android端上传的代码: xUtils.jar下载 Strin ...

  5. Android系统(96)---Android 数据交换解析框架Gson使用详解

    Android 数据交换解析框架Gson使用详解 Json 是一种文本形式的数据交换格式,比 xml 更为轻量.Json 的解析和生成的方式很多,在 Android 平台上最常用的类库有 Gson 和 ...

  6. Github项目解析(五)--Android日志框架

    转载请标明出处:一片枫叶的专栏 本文主要讲解的是自己实现一个简单的Android日志框架,可以在Android studio控制台中输入日志信息,其简单的使用方式: L.v("...&quo ...

  7. Android开源框架PowerfulViewLibrary——PowerfulEditText的介绍和源码解析

    本篇文章已授权微信公众号 guolin_blog (郭霖)独家发布 转载请注明出处:http://blog.csdn.net/chay_chan/article/details/63685905 An ...

  8. BAT高级架构师合力熬夜15天,肝出了这份PDF版《Android百大框架源码解析》,还不快快码住。。。

    前言 为什么要阅读源码? 现在中高级Android岗位面试中,对于各种框架的源码都会刨根问底,从而来判断应试者的业务能力边际所在.但是很多开发者习惯直接搬运,对各种框架的源码都没有过深入研究,在面试时 ...

  9. android 1.0框架,KOOM V1.0.5 框架解析

    快手在2020年中旬开源了一个线上OOM监控上报的框架:KOOM,这里简单研究下. 一.官方项目介绍 1.1 描述: KOOM是快手性能优化团队在处理移动端OOM问题的过程中沉淀出的一套完整解决方案. ...

最新文章

  1. centOS怎样强制卸载PHP以及自定义安装PHP
  2. 手机内存RAM、ROM简介
  3. 揭开ASP.NET生成随机密码的面纱
  4. QT解决方案中新建动态链接库工程,且继承于QObject,解决无法生成moc_XXX.cpp文件的问题,解决工程之间的引用问题
  5. Source Insight主题推荐和显示属性设置方法
  6. 1203正规式转换为有穷自动机
  7. 容器为何物,为什么它对OpenStack很重要?
  8. mysql批量用trim限定_mybatis中批量更新sql语句,trim、foreach标签,varchar定义理解
  9. Java实现正整数数组的最小拼接数
  10. ValidateAntiForgeryToken的用途,解释和示例
  11. CSS3弹性盒模型flexbox布局基础版
  12. oeasy的html需要会java_Oeasy,会玩才会学
  13. vue有纵向和横向表头表格
  14. Excel快速入门02
  15. 更新显卡驱动后黑屏无法进入系统的解决办法
  16. html5 刮刮乐 源码,HTML5 canvas实现刮刮乐功能
  17. 【机器学习】线性分类——高斯判别分析GDA(理论+图解+公式推导)
  18. 蓝月传奇服务器例行维护,《蓝月传奇》7月14日更新维护公告
  19. 微软首席建筑师 查尔斯—西蒙尼
  20. 包裹细胞膜的磁性纳米微载体/负载斑蝥素的巨噬细胞膜包封金属有机框架纳米颗粒的研究

热门文章

  1. 弘玑RPA x 流程挖掘,构筑超自动化的坚实底座
  2. 符号执行(静态执行、动态执行、并行执行、选择执行
  3. Android 用一个监听器实现多个监听
  4. 《5G NR标准:下一代无线通信技术》读书笔记——NR概述
  5. 输入法引擎的模型设计
  6. bird quagga 静态路由配置
  7. Unity3D 入门:Unity Editor 编辑器常用快捷键
  8. 基于Android studio3.6的JNI教程之ncnn之语义分割ENet
  9. 整理分享安徽省2022年技术先进型服务企业认定申报详细范围条件材料汇总
  10. Spring Boot Starters启动器的简介说明