Android数据存储的五种方法汇总

Android数据存储的五种方法汇总

本文介绍Android中的5种数据存储方式。

数据存储在开发中是使用最频繁的,在这里主要介绍Android平台中实现数据存储的5种方式,分别是:

1 使用SharedPreferences存储数据

2 文件存储数据

3 SQLite数据库存储数据

4 使用ContentProvider存储数据

5 网络存储数据

第一种: 使用SharedPreferences存储数据

SharedPreferences是Android平台上一个轻量级的存储类,主要是保存一些常用的配置比如窗口状态,一般在Activity中

重载窗口状态onSaveInstanceState保存一般使用SharedPreferences完成,它提供了Android平台常规的Long长 整形、Int整形、String字符串型的保存。

它是什么样的处理方式呢? SharedPreferences类似过去Windows系统上的ini配置文件,

但是它分为多种权限,可以全局共享访问,android123提示最终是以xml方式来保存,

整体效率来看不是特别的高,对于常规的轻量级而言比SQLite要好不少,

如果真的存储量不大可以考虑自己定义文件格式。

xml 处理时Dalvik会通过自带底层的本地XML Parser解析,比如XMLpull方式,这样对于内存资源占用比较好。

它的本质是基于XML文件存储key-value键值对数据,通常用来存储一些简单的配置信息。

其存储位置在/data/data/<包名>/shared_prefs目录下。

SharedPreferences对象本身只能获取数据而不支持存储和修改,存储修改是通过Editor对象实现。

实现SharedPreferences存储的步骤如下:

一、根据Context获取SharedPreferences对象

二、利用edit()方法获取Editor对象。

三、通过Editor对象存储key-value键值对数据。

四、通过commit()方法提交数据。

下面是示例代码:

public class MainActivity extends Activity {
 @Override
 
     public void onCreate(Bundle savedInstanceState) {
 
        super.onCreate(savedInstanceState);
 
        setContentView(R.layout.main);
 
        //获取SharedPreferences对象
 
        Context ctx = MainActivity.this;       
 
        SharedPreferences sp = ctx.getSharedPreferences("SP", MODE_PRIVATE);
 
        //存入数据
 
        Editor editor = sp.edit();
 
        editor.putString("STRING_KEY", "string");
 
        editor.putInt("INT_KEY", 0);
 
        editor.putBoolean("BOOLEAN_KEY", true);
 
        editor.commit();
 
        //返回STRING_KEY的值
 
        Log.d("SP", sp.getString("STRING_KEY", "none"));
 
        //如果NOT_EXIST不存在,则返回值为"none"
 
        Log.d("SP", sp.getString("NOT_EXIST", "none"));
 
     }
 
}

这段代码执行过后,即在/data/data/com.test/shared_prefs目录下生成了一个SP.xml文件,一个应用可以创建多个这样的xml文件。

SharedPreferences对象与SQLite数据库相比,免去了创建数据库,创建表,写SQL语句等诸多操作,相对而言更加方便,简洁。

但是SharedPreferences也有其自身缺陷,

比如其职能存储boolean,int,float,long和String五种简单的数据类型,比如其无法进行条件查询等。

所以不论SharedPreferences的数据存储操作是如何简单,它也只能是存储方式的一种补充,而无法完全替代如SQLite数据库这样的其他数据存储方式。

第二种: 文件存储数据

关于文件存储,Activity提供了openFileOutput()方法可以用于把数据输出到文件中,具体的实现过程与在J2SE环境中保存数据到文件中是一样的。

文件可用来存放大量数据,如文本、图片、音频等。

默认位置:/data/data/<包>/files/***.***。

代码示例:

public void save() {
        try {
 
            FileOutputStream outStream=this.openFileOutput("a.txt",Context.MODE_WORLD_READABLE);
 
            outStream.write(text.getText().toString().getBytes());
 
            outStream.close();
 
            Toast.makeText(MyActivity.this,"Saved",Toast.LENGTH_LONG).show();
 
        } catch (FileNotFoundException e) {
 
            return;
 
        }
 
        catch (IOException e){
 
            return ;
 
        }
 
 }

openFileOutput()方法的第一参数用于指定文件名称,不能包含路径分隔符“/” ,如果文件不存在,Android 会自动创建它。

创建的文件保存在/data/data//files目录,

如: /data/data/cn.itcast.action/files/itcast.txt ,

通过点击Eclipse菜单“Window”-“Show View”-“Other”,在对话窗口中展开android文件夹,

选择下面的File Explorer视图,然后在File Explorer视图中展开/data/data//files目录就可以看到该文件。

openFileOutput()方法的第二参数用于指定操作模式,有四种模式,分别为:

Context.MODE_PRIVATE = 0

Context.MODE_APPEND = 32768

Context.MODE_WORLD_READABLE = 1

Context.MODE_WORLD_WRITEABLE = 2

Context.MODE_PRIVATE:为默认操作模式,代表该文件是私有数据,只能被应用本身访问,

在该模式下,写入的内容会覆盖原文件的内容,如果想把新写入的内容追加到原文件中。可以使用Context.MODE_APPEND

Context.MODE_APPEND:模式会检查文件是否存在,存在就往文件追加内容,否则就创建新文件。

Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE用来控制其他应用是否有权限读写该文件。

MODE_WORLD_READABLE:表示当前文件可以被其他应用读取;

MODE_WORLD_WRITEABLE:表示当前文件可以被其他应用写入。

如果希望文件被其他应用读和写,

可以传入: openFileOutput(“itcast.txt”, Context.MODE_WORLD_READABLE + Context.MODE_WORLD_WRITEABLE);

android有一套自己的安全模型,当应用程序(.apk)在安装时系统就会分配给他一个userid,

当该应用要去访问其他资源比如文件的时候,就需要userid匹配。

默认情况下,任何应用创建的文件,sharedpreferences,数据库都应该是私有的(位于/data/data//files),其他程序无法访问。

除非在创建时指定了Context.MODE_WORLD_READABLE或者
Context.MODE_WORLD_WRITEABLE ,只有这样其他程序才能正确访问。

读取文件示例:

public void load()
 
{
 
    try {
 
        FileInputStream inStream=this.openFileInput("a.txt");
 
        ByteArrayOutputStream stream=new ByteArrayOutputStream();
 
        byte[] buffer=new byte[1024];
 
        int length=-1;
    while((length=inStream.read(buffer))!=-1)   {
 
            stream.write(buffer,0,length);
 
        }
 
        stream.close();
 
        inStream.close();
 
        text.setText(stream.toString());
 
        Toast.makeText(MyActivity.this,"Loaded",Toast.LENGTH_LONG).show();
 
    } catch (FileNotFoundException e) {
 
        e.printStackTrace();
 
    }
 
    catch (IOException e){
 
        return ;
 
    }
}

对于私有文件只能被创建该文件的应用访问,如果希望文件能被其他应用读和写,可以在创建文件时,

指定Context.MODE_WORLD_READABLE和Context.MODE_WORLD_WRITEABLE权限。

Activity还提供了getCacheDir()和getFilesDir()方法:

getCacheDir()方法用于获取/data/data//cache目录 getFilesDir()方法用于获取/data/data//files目录。

把文件存入SDCard:

使用Activity的openFileOutput()方法保存文件,文件是存放在手机空间上,一般手机的存储空间不是很大,存放些小文件还行,如果要存放像视频这样的大文件,是不可行的。

对于像视频这样的大文件,我们可以把它存放在SDCard。

SDCard是干什么的?你可以把它看作是移动硬盘或U盘。 在模拟器中使用SDCard,你需要先创建一张SDCard卡(当然不是真的SDCard,只是镜像文件)。

创建SDCard可以在Eclipse创建模拟器时随同创建,也可以使用DOS命令进行创建,

如下: 在Dos窗口中进入android SDK安装路径的tools目录,输入以下命令创建一张容量为2G的SDCard,文件后缀可以随便取,

建议使用.img: mksdcard 2048M D:\AndroidTool\sdcard.img 在程序中访问SDCard,你需要申请访问SDCard的权限。

在AndroidManifest.xml中加入访问SDCard的权限如下:

<!-- 在SDCard中创建与删除文件权限 -->     <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
  <!-- 往SDCard写入数据权限 -->     <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> 

要往SDCard存放文件,程序必须先判断手机是否装有SDCard,并且可以进行读写。

注意:访问SDCard必须在AndroidManifest.xml中加入访问SDCard的权限。

if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
    File sdCardDir = Environment.getExternalStorageDirectory();//获取SDCard目录
    File saveFile = new File(sdCardDir, “a.txt”);
 
        FileOutputStream outStream = new FileOutputStream(saveFile);
 
        outStream.write("test".getBytes());
 
        outStream.close();
 
}

Environment.getExternalStorageState()方法用于获取SDCard的状态,如果手机装有SDCard,并且可以进行读写,

那么方法返回的状态等于Environment.MEDIA_MOUNTED。

Environment.getExternalStorageDirectory()方法用于获取SDCard的目录,当然要获取SDCard的目录,你也可以这样写:

File sdCardDir = new File("/sdcard"); //获取SDCard目录

File saveFile = new File(sdCardDir, “itcast.txt”);

//上面两句代码可以合成一句:

File saveFile = new File("/sdcard/a.txt");

FileOutputStream outStream = new FileOutputStream(saveFile);

outStream.write(“test”.getBytes());

outStream.close();

第三种: SQLite数据库存储数据

SQLite是轻量级嵌入式数据库引擎,它支持 SQL 语言,并且只利用很少的内存就有很好的性能。此外它还是开源的,任何人都可以使用它。

许多开源项目((Mozilla, PHP, Python)都使用了 SQLite.SQLite 由以下几个组件组成:SQL 编译器、内核、后端以及附件。

SQLite 通过利用虚拟机和虚拟数据库引擎(VDBE),使调试、修改和扩展 SQLite 的内核变得更加方便。

特点:

面向资源有限的设备,

没有服务器进程,

所有数据存放在同一文件中跨平台,

可自由复制。

SQLite 内部结构:

SQLite 基本上符合 SQL-92 标准,和其他的主要 SQL 数据库没什么区别。它的优点就是高效,Android 运行时环境包含了完整的 SQLite。

SQLite 和其他数据库最大的不同就是对数据类型的支持,创建一个表时,可以在 CREATE TABLE 语句中指定某列的数据类型,

但是你可以把任何数据类型放入任何列中。当某个值插入数据库时,SQLite 将检查它的类型。

如果该类型与关联的列不匹配,则 SQLite 会尝试将该值转换成该列的类型。

如果不能转换,则该值将作为其本身具有的类型存储。

比如可以把一个字符串(String)放入 INTEGER 列。SQLite 称这为“弱类型”(manifest typing.)。

此外,SQLite 不支持一些标准的 SQL 功能,特别是外键约束(FOREIGN KEY constrains),

嵌套 transcaction 和 RIGHT OUTER JOIN 和 FULL OUTER JOIN, 还有一些 ALTER TABLE 功能。

除了上述功能外,SQLite 是一个完整的 SQL 系统,拥有完整的触发器,交易等等。

Android 集成了 SQLite 数据库 Android 在运行时(run-time)集成了 SQLite,所以每个 Android 应用程序都可以使用 SQLite 数据库。

对于熟悉 SQL 的开发人员来时,在 Android 开发中使用 SQLite 相当简单。但是,由于 JDBC 会消耗太多的系统资源,所以 JDBC 对于手机这种内存受限设备来说并不合适。

因此,Android 提供了一些新的 API 来使用 SQLite 数据库,Android 开发中,程序员需要学使用这些 API。

数据库存储在 data/< 项目文件夹 >/databases/ 下。 Android 开发中使用 SQLite 数据库 Activites 可以通过 Content Provider 或者 Service 访问一个数据库。

下面会详细讲解如果创建数据库,添加数据和查询数据库。 创建数据库 Android 不自动提供数据库。

在 Android 应用程序中使用 SQLite,必须自己创建数据库,然后创建表、索引,填充数据。

Android 提供了 SQLiteOpenHelper 帮助你创建一个数据库,

你只要继承 SQLiteOpenHelper 类,就可以轻松的创建数据库。

SQLiteOpenHelper 类根据开发应用程序的需要,封装了创建和更新数据库使用的逻辑。

SQLiteOpenHelper 的子类,至少需要实现三个方法:
1 构造函数,调用父类 SQLiteOpenHelper 的构造函数。

这个方法需要四个参数:上下文环境(例如,一个 Activity),数据库名字,一个可选的游标工厂(通常是 Null),一个代表你正在使用的数据库模型版本的整数。

2 onCreate()方法,它需要一个 SQLiteDatabase 对象作为参数,根据需要对这个对象填充表和初始化数据。

3 onUpgrage() 方法,它需要三个参数,一个 SQLiteDatabase 对象,一个旧的版本号和一个新的版本号,这样你就可以清楚如何把一个数据库从旧的模型转变到新的模型。

下面示例代码展示了如何继承 SQLiteOpenHelper 创建数据库:

public class DatabaseHelper extends SQLiteOpenHelper {
  DatabaseHelper(Context context, String name, CursorFactory cursorFactory, int version) 
 
  {     
 
    super(context, name, cursorFactory, version);     
 
     }     
 
     @Override    
 
     public void onCreate(SQLiteDatabase db) {     
 
         // TODO 创建数据库后,对数据库的操作     
 
     }     
 
     @Override    
 
 public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {     
 
         // TODO 更改数据库版本的操作     
 
     }     
 
 @Override    
 
 public void onOpen(SQLiteDatabase db) {     
 
         super.onOpen(db);       
 
         // TODO 每次成功打开数据库后首先被执行     
 
     }     
 
 }

public class DBHelper extends SQLiteOpenHelper {

    // 数据库名字
    private static final String DB_NAME = "person.db";
    // 数据库版本 version >= 1
    private static final int DB_VERSION = 1;

    // 指定了数据库的名称,版本信息
    public DBHelper(@Nullable Context context) {
        super(context, DB_NAME, null, DB_VERSION);
    }

    /**
     * 创建数据库中的表
     * @param db
     */
    @Override
    public void onCreate(SQLiteDatabase db) {

        // SQLite数据库里的字段一般是不区分类型的,但是主键除外,主键必须是整形
        String sql = "CREATE TABLE person (_id INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL,name CHAR(10),nickname CHAR(10))";
        db.execSQL(sql);
    }

    /**
     * 数据库升级的方法
     * @param db
     * @param oldVersion
     * @param newVersion
     */
    @Override
    public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
        if (newVersion > oldVersion){
            String sql = "DROP TABLE IF EXITS person";
            db.execSQL(sql);

            // 先删表在建表
            onCreate(db);
        }
    }
}

接下来讨论具体如何创建表、插入数据、删除表等等。

调用 getReadableDatabase() 或 getWriteableDatabase() 方法,你可以得到 SQLiteDatabase 实例,具体调用那个方法,取决于你是否需要改变数据库的内容:

 db=(new DatabaseHelper(getContext())).getWritableDatabase();        return (db == null) ? false : true;  

上面这段代码会返回一个 SQLiteDatabase 类的实例,使用这个对象,你就可以查询或者修改数据库。

当你完成了对数据库的操作(例如你的 Activity 已经关闭),需要调用 SQLiteDatabase 的 Close() 方法来释放掉数据库连接。

1、添加数据的方法

  • execSQL方法进行添加数据

     /**
     * SQL插入语句:
     * INSERT INTO Book(name,author,pages,price) VALUES
    * ("The Da Vinci Code" ,"Dan Brown",454,16.96);
    */
    db.execSQL("INSERT INTO Book(name,author,pages,price) VALUES(?,?,?,?",
    new String[]{"The Lost Symbol", "Dan Brown", "510", "19.95"});
    
    
  • 使用SQLiteDatabase 中提供了一个 insert() 方法,这个方法就是专门用于添加数据的。

    insert() 方法接收三个参数,第一个参数是表名,我们希望向哪张表里添加数据,这里就传入该表的名字。第二个参数用于在未指定添加数据的情况下给某些可为空的列自动赋值NULL,一般用不到这个功能,直接传入 null 即可。第三个参数是一个 ContentValues 对象,它提供了一系列的 put() 方法重载,用于向 ContentValues 中添加数据,只需要将表中的每个列名及相对应的待添加数据传入即可。

    public void onClick(View v) {
                    //获取 SQLiteDatabase 对象
                    SQLiteDatabase db = dbHelper.getWritableDatabase();
                    //使用ContentValues 对数据进行组装
                    ContentValues values = new ContentValues();
                    //开始组装第一条数据
                    values.put("name", "The Da Vinci Code");
                    values.put("author", "Dan Brown");
                    values.put("pages", 454);
                    values.put("price", 16.96);
                    //插入第一条数据
                    db.insert("Book", null, values);
                    values.clear();
                    //开始组装第二条数据
                    values.put("name", "The Lost Symbol");
                    values.put("author", "Dan Brown");
                    values.put("pages", 510);
                    values.put("price", 19.95);
                    //插入第二条数据
                    db.insert("Book", null, values);
    }
    
    

    这里只对Book表里其中四列的数据进行了组装,id并没有赋值。因为创建表的时候就将 id 列设置为自动增长了,它的值会在入库的时候自动生成,不需要手动赋值。

2、删除数据的方法

  • execSQL方法进行添加数据

            /**
             *SQL删除语句:
             * DELETE FROM Book WHERE pages > 500;
             */
            db.execSQL("DELETE FROM Book WHERE pages > ?",new String[]{"500"});
    
    
  • SQLiteDatabase 中提供了一个 delete()方法专门用于删除数据,这个方法接收三个参数,第一个参数仍然是表名,第二、第三个参数用于约束删除某一行或某几行的数据,不指定的话默认就是删除所有行。

                public void onClick(View v) {
                    SQLiteDatabase db = dbHelper.getWritableDatabase();
                    db.delete("Book", "pages > ?", new String[]{"500"});
                }
    
    

3、修改(更新)数据的方法

  • execSQL方法进行添加数据

            /**
             * SQL更新语句:
             * UPDATE Book SET price = 10.99 WHERE name = "The Da Vinci Code" ;
             */
            db.execSQL("UPDATE Book SET price = ? WHERE name = ?",
                    new String[]{"10.99", "The Da Vinci Code"});
    
    
  • SQLiteDatabase 中提供了一个非常好用的 update() 方法用于对数据进行更新,

    这个方法接收四个参数,第一个参数和 insert()方法一样,也是表名,在这里指定去更新哪张表里的数据。第二个参数是 ContentValues 对象,把要更新的数据在这里组装进去。第三、第四个参数用于去约束更新某一行的数据,不指定的话默认就是更新所有行。

                public void onClick(View v) {
                    SQLiteDatabase db = dbHelper.getWritableDatabase();
                    ContentValues values = new ContentValues();
                    values.put("price", 10.99);
                    //?是一个占位符,通过字符串数组为每个占位符指定相应的内容
                    db.update("Book", values, "name = ?", new String[]{"The Da Vinci Code"});
                }
    
    

4、查询数据的方法

  • execSQL方法进行添加数据

            /**
             * SQL查询语句:
             * SELECT * FROM BOOK ;
             */
            db.rawQuery("SELECT * FROM BOOK", null);
    
    
  • SQLiteDatabase 中还提供了一个 query() 方法用于对数据进行查询。这个方法的参数非常复杂,最短的一个方法重载也需要传入七个参数。

    query()方法参数及对应SQL:

    table:指定查询的表名,对应 from table_name

    columns:指定查询的列名,对应 select column1,column2 …

    selection:指定 where 的约束条件,where column = value

    selectionArgs:为 where 中的占位符提供具体的值

    groupBy:指定需要分组的列,group by column

    having:对分组后的结果进一步约束,having column = value

    orderBy:指定查询结果的排序方式,order by column

    虽然 query()方法的参数非常多,但是不必每条查询语句都指定上所有的参数,多数情况下只需传入少数几个参数就可以完成查询操作了。

    调用 query()方法后会返回一个 Cursor 对象,查询到的所有数据都将从这个对象中取出

                public void onClick(View v) {
                    SQLiteDatabase db = dbHelper.getWritableDatabase();
                    //查询Book表中的所有数据
                    Cursor cursor = db.query("Book", null, null, null, null, null, null, null);
                    //遍历Curosr对象,取出数据并打印
                    while (cursor.moveToNext()) {
                        String name = cursor.getString(cursor.getColumnIndex("name"));
                        String author = cursor.getString(cursor.getColumnIndex("author"));
                        int pages = cursor.getInt(cursor.getColumnIndex("pages"));
                        double price = cursor.getDouble(cursor.getColumnIndex("price"));
                        Log.d("woider", "Book Name:" + name + " Author:" 
                                + author + " Pages:" + pages + " Price:" + price);
                    }
                    //关闭Cursor
                    cursor.close();
                }
    
    

第四种: 使用ContentProvider存储数据

Android这个系统和其他的操作系统还不太一样,我们需要记住的是,数据在Android当中是私有的,当然这些数据包括文件数据和数据库数据以及一些其他类型的数据。

那这个时候有读者就会提出问题,难道两个程序之间就没有办法对于数据进行交换?Android这么优秀的系统不会让这种情况发生的。

解决这个问题主要靠ContentProvider。一个Content Provider类实现了一组标准的方法接口,从而能够让其他的应用保存或读取此Content Provider的各种数据类型。

也就是说,一个程序可以通过实现一个Content Provider的抽象接口将自己的数据暴露出去。外界根本看不到,也不用看到这个应用暴露的数据在应用当中是如何存储的,

或者是用数据库存储还是用文件存储,还是通过网上获得,这些一切都不重要,

重要的是外界可以通过这一套标准及统一的接口和程序里的数据打交道,可以读取程序的数据,也可以删除程序的数据,当然,中间也会涉及一些权限的问题。

一个程序可以通过实现一个ContentProvider的抽象接口将自己的数据完全暴露出去,而且ContentProviders是以类似数据库中表的方式将数据暴露,

也就是说ContentProvider就像一个“数据库”。那么外界获取其提供的数据,也就应该与从数据库中获取数据的操作基本一样,只不过是采用URI来表示外界需要访问的“数据库”。

Content Provider提供了一种多应用间数据共享的方式,比如:联系人信息可以被多个应用程序访问。

Content Provider是个实现了一组用于提供其他应用程序存取数据的标准方法的类。

应用程序可以在Content Provider中执行如下操作: 查询数据 修改数据 添加数据 删除数据

标准的Content Provider: Android提供了一些已经在系统中实现的标准Content Provider,

比如联系人信息,图片库等等,你可以用这些Content Provider来访问设备上存储的联系人信息,图片等等。

查询记录:

在Content Provider中使用的查询字符串有别于标准的SQL查询。

很多诸如select, add, delete, modify等操作我们都使用一种特殊的URI来进行,这种URI由3个部分组成, “content://”, 代表数据的路径,和一个可选的标识数据的ID。

以下是一些示例URI:

content://media/internal/images 这个URI将返回设备上存储的所有图片

content://contacts/people/ 这个URI将返回设备上的所有联系人信息

content://contacts/people/45 这个URI返回单个结果(联系人信息中ID为45的联系人记录)

尽管这种查询字符串格式很常见,但是它看起来还是有点令人迷惑。

为此,Android提供一系列的帮助类(在android.provider包下),里面包含了很多以类变量形式给出的查询字符串,这种方式更容易让我们理解一点,参见下例:

MediaStore.Images.Media.INTERNAL_CONTENT_URI Contacts.People.CONTENT_URI

因此,如上面content://contacts/people/45这个URI就可以写成如下形式:

Uri person = ContentUris.withAppendedId(People.CONTENT_URI, 45);

然后执行数据查询: Cursor cur = managedQuery(person, null, null, null);

这个查询返回一个包含所有数据字段的游标,我们可以通过迭代这个游标来获取所有的数据:

package com.wissen.testApp;
public class ContentProviderDemo extends Activity {
 
    @Override
 
    public void onCreate(Bundle savedInstanceState) {
 
        super.onCreate(savedInstanceState);
 
        setContentView(R.layout.main);
 
       displayRecords();
 
    }
 
    private void displayRecords() {
 
        //该数组中包含了所有要返回的字段
 
     String columns[] = new String[] { People.NAME, People.NUMBER };
 
       Uri mContacts = People.CONTENT_URI;
 
       Cursor cur = managedQuery(
 
           mContacts,
 
          columns,  // 要返回的数据字段
 
       null,          // WHERE子句
 
       null,         // WHERE 子句的参数
 
       null         // Order-by子句
 
     );
 
       if (cur.moveToFirst()) {
 
           String name = null;
 
           String phoneNo = null;
 
           do {
 
              // 获取字段的值
 
         name = cur.getString(cur.getColumnIndex(People.NAME));
 
             phoneNo = cur.getString(cur.getColumnIndex(People.NUMBER));
 
             Toast.makeText(this, name + ” ” + phoneNo, Toast.LENGTH_LONG).show();
 
          } while (cur.moveToNext());
 
       }
 
    }
}

上例示范了一个如何依次读取联系人信息表中的指定数据列name和number。

修改记录:

我们可以使用ContentResolver.update()方法来修改数据,我们来写一个修改数据的方法:

private void updateRecord(int recNo, String name) {
         Uri uri = ContentUris.withAppendedId(People.CONTENT_URI, recNo);
         ContentValues values = new ContentValues();
         values.put(People.NAME, name);
         getContentResolver().update(uri, values, null, null);
 
    }

现在你可以调用上面的方法来更新指定记录: updateRecord(10, ”XYZ”); //更改第10条记录的name字段值为“XYZ”

添加记录:

要增加记录,我们可以调用ContentResolver.insert()方法,该方法接受一个要增加的记录的目标URI,以及一个包含了新记录值的Map对象,

调用后的返回值是新记录的URI,包含记录号。

上面的例子中我们都是基于联系人信息簿这个标准的Content Provider,现在我们继续来创建一个insertRecord() 方法以对联系人信息簿中进行数据的添加:

private void insertRecords(String name, String phoneNo) {
    ContentValues values = new ContentValues();
 
    values.put(People.NAME, name);
 
    Uri uri = getContentResolver().insert(People.CONTENT_URI, values);
 
    Log.d(”ANDROID”, uri.toString());
 
    Uri numberUri = Uri.withAppendedPath(uri, People.Phones.CONTENT_DIRECTORY);
 
    values.clear();
 
    values.put(Contacts.Phones.TYPE, People.Phones.TYPE_MOBILE);
 
    values.put(People.NUMBER, phoneNo);
 
    getContentResolver().insert(numberUri, values);
 
}

这样我们就可以调用insertRecords(name, phoneNo)的方式来向联系人信息簿中添加联系人姓名和电话号码。

删除记录:

Content Provider中的getContextResolver.delete()方法可以用来删除记录。

rivate void deleteRecords() {

Uri uri = People.CONTENT_URI;

getContentResolver().delete(uri, null, null);

}

你也可以指定WHERE条件语句来删除特定的记录:

getContentResolver().delete(uri, “NAME=” + “‘XYZ XYZ’”, null);

这将会删除name为‘XYZ XYZ’的记录。

创建Content Provider:

至此我们已经知道如何使用Content Provider了,现在让我们来看下如何自己创建一个Content Provider。

要创建我们自己的Content Provider的话,我们需要遵循以下几步:

  1. 创建一个继承了ContentProvider父类的类

  2. 定义一个名为CONTENT_URI,并且是public static final的Uri类型的类变量,你必须为其指定一个唯一的字符串值,最好的方案是以类的全名称,

如: public static final Uri CONTENT_URI = Uri.parse( “content://com.google.android.MyContentProvider”);

  1. 创建你的数据存储系统。大多数Content Provider使用Android文件系统或SQLite数据库来保持数据,但是你也可以以任何你想要的方式来存储。

  2. 定义你要返回给客户端的数据列名。如果你正在使用Android数据库,则数据列的使用方式就和你以往所熟悉的其他数据库一样。

但是,你必须为其定义一个叫_id的列,它用来表示每条记录的唯一性。

  1. 如果你要存储字节型数据,比如位图文件等,那保存该数据的数据列其实是一个表示实际保存文件的URI字符串,

客户端通过它来读取对应的文件数据,处理这种数据类型的Content Provider需要实现一个名为_data的字段,_data字段列出了该文件在Android文件系统上的精确路径。

这个字段不仅是供客户端使用,而且也可以供ContentResolver使用。客户端可以调用ContentResolver.openOutputStream()方法来处理该URI指向的文件资源,

如果是ContentResolver本身的话,由于其持有的权限比客户端要高,所以它能直接访问该数据文件。

  1. 声明public static String型的变量,用于指定要从游标处返回的数据列。

  2. 查询返回一个Cursor类型的对象。所有执行写操作的方法如insert(), update() 以及delete()都将被监听。

我们可以通过使用ContentResover().notifyChange()方法来通知监听器关于数据更新的信息。

  1. 在AndroidMenifest.xml中使用标签来设置Content Provider。

  2. 如果你要处理的数据类型是一种比较新的类型,你就必须先定义一个新的MIME类型,以供ContentProvider.geType(url)来返回。

MIME类型有两种形式:

一种是为指定的单个记录的,还有一种是为多条记录的。

这里给出一种常用的格式: vnd.android.cursor.item/vnd.yourcompanyname.contenttype (单个记录的MIME类型)

比如, 一个请求列车信息的URI如content://com.example.transportationprovider/trains/122

可能就会返回typevnd.android.cursor.item/vnd.example.rail这样一个MIME类型。

vnd.android.cursor.dir/vnd.yourcompanyname.contenttype (多个记录的MIME类型)

比如, 一个请求所有列车信息的URI如content://com.example.transportationprovider/trains

可能就会返回vnd.android.cursor.dir/vnd.example.rail这样一个MIME 类型。

下列代码将创建一个Content Provider,它仅仅是存储用户名称并显示所有的用户名称(使用 SQLLite数据库存储这些数据):

package com.wissen.testApp;
public class MyUsers {
 
    public static final String AUTHORITY  = “com.wissen.MyContentProvider”;
 
    // BaseColumn类中已经包含了 _id字段
 
   public static final class User implements BaseColumns {
 
        public static final Uri CONTENT_URI  = Uri.parse(”content://com.wissen.MyContentProvider”);
 
        // 表数据列
 
     public static final String  USER_NAME  = “USER_NAME”;
 
    }
}

上面的类中定义了Content Provider的CONTENT_URI,以及数据列。下面我们将定义基于上面的类来定义实际的Content Provider类:

package com.wissen.testApp.android;
public class MyContentProvider extends ContentProvider {
 
    private SQLiteDatabase     sqlDB;
 
    private DatabaseHelper    dbHelper;
 
    private static final String  DATABASE_NAME     = “Users.db”;
 
    private static final int        DATABASE_VERSION         = 1;
 
    private static final String TABLE_NAME   = “User”;
 
    private static final String TAG = “MyContentProvider”;
 
    private static class DatabaseHelper extends SQLiteOpenHelper {
 
        DatabaseHelper(Context context) {
 
            super(context, DATABASE_NAME, null, DATABASE_VERSION);
 
        }
 
        @Override
 
        public void onCreate(SQLiteDatabase db) {
 
            //创建用于存储数据的表
 
        db.execSQL(”Create table ” + TABLE_NAME + “( _id INTEGER PRIMARY KEY AUTOINCREMENT, USER_NAME TEXT);”);
 
        }
 
        @Override
 
        public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
 
            db.execSQL(”DROP TABLE IF EXISTS ” + TABLE_NAME);
 
            onCreate(db);
 
        }
 
    }
 
    @Override
 
    public int delete(Uri uri, String s, String[] as) {
 
        return 0;
 
    }
 
    @Override
 
    public String getType(Uri uri) {
 
        return null;
 
    }
 
    @Override
 
    public Uri insert(Uri uri, ContentValues contentvalues) {
 
        sqlDB = dbHelper.getWritableDatabase();
 
        long rowId = sqlDB.insert(TABLE_NAME, “”, contentvalues);
 
        if (rowId > 0) {
 
            Uri rowUri = ContentUris.appendId(MyUsers.User.CONTENT_URI.buildUpon(), rowId).build();
 
            getContext().getContentResolver().notifyChange(rowUri, null);
 
            return rowUri;
 
        }
 
        throw new SQLException(”Failed to insert row into ” + uri);
 
    }
 
    @Override
 
    public boolean onCreate() {
 
        dbHelper = new DatabaseHelper(getContext());
 
        return (dbHelper == null) ? false : true;
 
    }
 
    @Override
 
    public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs, String sortOrder) {
 
        SQLiteQueryBuilder qb = new SQLiteQueryBuilder();
 
        SQLiteDatabase db = dbHelper.getReadableDatabase();
 
        qb.setTables(TABLE_NAME);
 
        Cursor c = qb.query(db, projection, selection, null, null, null, sortOrder);
 
        c.setNotificationUri(getContext().getContentResolver(), uri);
 
        return c;
 
    }
 
    @Override
 
    public int update(Uri uri, ContentValues contentvalues, String s, String[] as) {
 
        return 0;
 
    }
}

一个名为MyContentProvider的Content Provider创建完成了,它用于从Sqlite数据库中添加和读取记录。

Content Provider的入口需要在AndroidManifest.xml中配置:

之后,让我们来使用这个定义好的Content Provider:

public class MyContentDemo extends Activity {
 
    @Override
 
    protected void onCreate(Bundle savedInstanceState) {
 
        super.onCreate(savedInstanceState);
 
        insertRecord(”MyUser”);
 
        displayRecords();
 
    }
 
    private void insertRecord(String userName) {
 
        ContentValues values = new ContentValues();
 
        values.put(MyUsers.User.USER_NAME, userName);
 
        getContentResolver().insert(MyUsers.User.CONTENT_URI, values);
 
    }
 
    private void displayRecords() {
 
        String columns[] = new String[] { MyUsers.User._ID, MyUsers.User.USER_NAME };
 
        Uri myUri = MyUsers.User.CONTENT_URI;
 
        Cursor cur = managedQuery(myUri, columns,null, null, null );
 
        if (cur.moveToFirst()) {
 
            String id = null;
 
            String userName = null;
 
            do {
 
                id = cur.getString(cur.getColumnIndex(MyUsers.User._ID));
 
                userName = cur.getString(cur.getColumnIndex(MyUsers.User.USER_NAME));
 
                Toast.makeText(this, id + ” ” + userName, Toast.LENGTH_LONG).show();
 
           } while (cur.moveToNext());
 
       }
 
    }
}

上面的类将先向数据库中添加一条用户数据,然后显示数据库中所有的用户数据。

第五种: 网络存储数据

前面介绍的几种存储都是将数据存储在本地设备上,除此之外,还有一种存储(获取)数据的方式,通过网络来实现数据的存储和获取。

我们可以调用WebService返回的数据或是解析HTTP协议实现网络数据交互。

具体需要熟悉java.net.*,Android.net.*这两个包的内容,在这就不赘述了,请大家参阅相关文档。

下面是一个通过地区名称查询该地区的天气预报,以POST发送的方式发送请求到webservicex.net站点,访问WebService.webservicex.net站点上提供查询天气预报的服务。

代码如下:

import java.util.ArrayList;
import java.util.List;
 
import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.protocol.HTTP;
import org.apache.http.util.EntityUtils;
 
import android.app.Activity;
import android.os.Bundle;
 
public class MyAndroidWeatherActivity extends Activity {
    //定义需要获取的内容来源地址
    private static final String SERVER_URL =
        "http://www.webservicex.net/WeatherForecast.asmx/GetWeatherByPlaceName"; 
 
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
 
        HttpPost request = new HttpPost(SERVER_URL); //根据内容来源地址创建一个Http请求
        // 添加一个变量
        List<NameValuePair> params = new ArrayList<NameValuePair>();
        // 设置一个地区名称
        params.add(new BasicNameValuePair("PlaceName", "NewYork"));  //添加必须的参数
 
        try {
            //设置参数的编码
            request.setEntity(new UrlEncodedFormEntity(params, HTTP.UTF_8));
            //发送请求并获取反馈
            HttpResponse httpResponse = new DefaultHttpClient().execute(request);
 
            // 解析返回的内容
            if(httpResponse.getStatusLine().getStatusCode() != 404){
               String result = EntityUtils.toString(httpResponse.getEntity());
               System.out.println(result);
            }
        } catch (Exception e) {
            e.printStackTrace();
       }
    }
}

别忘记了在配置文件中设置访问网络权限:

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容