Android xUtils框架解析

来源:互联网 发布:2017年网络谣言的案例 编辑:程序博客网 时间:2024/05/05 02:07

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

    @Override    public 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;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        x.view().inject(this);        ...    }}

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

@ContentView(R.layout.fragment_http)public class HttpFragment extends Fragment {    @Override    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {        return x.view().inject(this, inflater, container);    }    @Override    public 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();        @Override        public int getCount() {            return listUrl.size();        }        @Override        public String getItem(int position) {            return listUrl.get(position);        }        @Override        public long getItemId(int position) {            return position;        }        @Override        public 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;...@Overrideprotected 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>() {        @Override        public void onSuccess(String result) {            Toast.makeText(HttpUtilsActivity.this, result, Toast.LENGTH_SHORT).show();            progressDialog.cancel();        }        //请求异常后的回调方法        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        //主动调用取消请求的回调方法        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public 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>() {        @Override        public void onSuccess(String result) {             MainActivity.showlog("onSuccess-->result="+result);             Toast.makeText(HttpUtilsActivity.this, result, Toast.LENGTH_SHORT).show();                 }        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public 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>() {            @Override            public void onSuccess(String result) {                MainActivity.showlog("onSuccess-->result="+result);                Toast.makeText(HttpUtilsActivity.this, result, Toast.LENGTH_SHORT).show();            }            @Override            public void onError(Throwable ex, boolean isOnCallback) {                MainActivity.showlog("onError-->Throwable="+ex.getMessage());            }            @Override            public void onCancelled(CancelledException cex) {                MainActivity.showlog("onCancelled-->CancelledException="+cex.getMessage());            }            @Override            public 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>() {        @Override        public void onSuccess(File result) {             try {                    Bitmap bm = BitmapFactory.decodeStream(new FileInputStream(file));                    image.setImageBitmap(bm);                } catch (FileNotFoundException e) {                    e.printStackTrace();                }        }        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public void onFinished() {        }        //网络请求之前回调        @Override        public void onWaiting() {        }        //网络请求开始的时候回调        @Override        public void onStarted() {        }        //下载的时候不断回调的方法        @Override        public 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 {    //检查服务器返回的响应头信息    @Override    public void checkResponse(UriRequest request) throws Throwable {    }    /**     * 转换result为resultType类型的对象     *     * @param resultType  返回值类型(可能带有泛型信息)     * @param resultClass 返回值类型     * @param result      字符串数据     * @return     * @throws Throwable     */    @Override    public 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());                }                @Override                public void onError(Throwable ex, boolean isOnCallback) {}                @Override                public void onCancelled(CancelledException cex) {}                @Override                public 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>(){                @Override                public void onSuccess(Weather result) {                    super.onSuccess(result);                    Toast.makeText(HttpUtilsActivity.this, result.weatherinfo.toString(), Toast.LENGTH_SHORT).show();                    MainActivity.showlog(result.weatherinfo.toString());                }                @Override                public 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>() {        @Override        public void onSuccess(Drawable result) {        }        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public void onFinished() {        }    });    x.image().bind(image04, urls[3], options, new Callback.CommonCallback<Drawable>() {        @Override        public void onSuccess(Drawable result) {        }        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public void onFinished() {        }    });    /**     * loadDrawable()方法加载图片     */    Callback.Cancelable cancelable = x.image().loadDrawable(urls[0], options, new Callback.CommonCallback<Drawable>() {        @Override        public void onSuccess(Drawable result) {            image03.setImageDrawable(result);        }        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public void onFinished() {        }    });    //主动取消loadDrawable()方法    //cancelable.cancel();    /**     * loadFile()方法     * 应用场景:当我们通过bind()或者loadDrawable()方法加载了一张图片后,     * 它会保存到本地文件中,那当我需要这张图片时,就可以通过loadFile()方法进行查找。     * urls[0]:网络地址     */    x.image().loadFile(urls[0],options,new Callback.CacheCallback<File>(){        @Override        public boolean onCache(File result) {            //在这里可以做图片另存为等操作            Log.i("JAVA","file:"+result.getPath()+result.getName());            return true; //相信本地缓存返回true        }        @Override        public void onSuccess(File result) {            Log.i("JAVA","file");        }        @Override        public void onError(Throwable ex, boolean isOnCallback) {        }        @Override        public void onCancelled(CancelledException cex) {        }        @Override        public 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() {                    @Override                    public 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;    }    @Override    public 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;    }    @Override    public 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下载地址

2 0
原创粉丝点击