前言

最近到了一家公司,跟一个同事做项目,比如常规的一些操作用SharedPreferences就很好搞定,他跟我说SharedPreferences 会影响性能说了一堆… 难道别的存储方式就不耗费性能吗?不消耗内存吗?

有关Android存储

Android中的数据存储方案主要有:共享首选项(SharedPreferences)、内部存储(Internal Storage)、外部存储(External Storage)、SQLite数据库、 网络存储 静态变量等

SharedPreferences 共享首选项

先说一下SharedPreferences 简称SP , SharedPreferences类提供了一个通用框架,使开发者能够以键值对的方式,永久性的保存一些原始数据类型的数据,包括:布尔值,浮点值,整型值,长整型和字符串,被保存的数据可以跨多个用户会话永久保留(即使应用已经终止)。

为什么使用 SP 它的优点是什么? 使用SP 为Android开发者提供了更为便利的存储方式,但是相对的他也是有一些缺点的。

当我们首次创建 SharedPreferences 对象时,会根据文件名将文件下内容一次性加载到 mMap(SharedPreferencesImpl 成员) 容器中,每当我们 edit 都会创建一个新的 EditorImpl 对象,当修改或者添加数据时会将数据添加到 mModifiled (EditorImpl 成员)容器中,然后 commit 或 apply 操作比较 mMap 与 mModifiled 数据修正 mMap 中最后一次提交数据,然后写入到文件中。而 get 直接从 mMap 中读取。试想如果此时你存储了一些大型 key 或 value 它们会一直存储在内存中得不到释放。

(1) 不要存放大的 key 和 value 在 SharedPreferences 中,否则会一直存储在内存中得不到释放,内存使用过高会频发引发GC,导致界面丢帧甚至ANR
(2) 不相关的配置选项最好不要放在一起,单个文件越大读取速度则越慢。
(3) 读取频繁的 key 和不频繁的 key 尽量不要放在一起(如果整个文件本身就较小则忽略,为了这点性能添加维护得不偿失)。
(4) 尽量不要存放 JSON 和 HTML,这种可以直接文件缓存。Json/HTML 文件较小忽略不计!
(5) 不要指望这货能够跨进程通信 Context.PROCESS

静态变量存储方式

因为这种方式我与我一个同事有了一些意见不一的地方,他喜欢使用静态变量。现在的开发是针对于设备开发,而并非手机app的开发,如果app开发他的那种搞法非得出现问题不可。
因为Activity初始化的静态变量也会被置空,因此它的生命周期是不稳定的。也就是说,在Android中静态变量可能随时被系统置空,变为null,其它地方在使用的时候就会产生空指针现象

在Android开发中不提倡过多使用static类型的变量(除了 static final)因为静态变量生命周期较长,而且不易被系统回收,因此如果不能合理地使用静态变量,就会适得其反,造成大量的内存浪费,所谓过犹不及。
经过我的多方查证 建议在具备下列全部条件的情况下,尽量使用静态变量:
(1) 变量所包含的对象体积较大,占用内存较多。
(2) 变量所包含的对象生命周期较长。
(3) 变量所包含的对象数据稳定。
(4) 该类的对象实例有对该变量所包含的对象的共享需求。
如果变量不具备上述特点建议你不要轻易地使用静态变量,以免弄巧成拙。
对于全局变量使用Android提供了 Application 其生命周期与应用程序共存亡。所以对于全局变量可以写在 Application。

到底使用哪一种存储方式?

(1) 如果是简单的数据结构,首选 SharedPreferences
(2) 如果是数据量大处理复杂的话,首选是sql数据库 对于结构化的数据,一定要使用数据库,虽然会显得比较麻烦,但是后续的使用中会获益无穷
(3) 普通文件就是指文本文件,二进制文件,多媒体文件,还有频繁的读取要是用二进制文件。
(4) 使用网络存储时要避免传输大数据量数据,应注意对网络传输方式的优化(减少请求次数等)。

文件存储方式(实战工具类)

文件存储方式工具类

数据库存储方式(实战工具类)

数据库最近学会的是一个DAO 简直爱死这个了,处理数据库数据是真的轻松。

示例:

public class DbController {

    /**
     * Helper
     */
    private DaoMaster.DevOpenHelper mHelper;
    /**
     * 数据库
     */
    private SQLiteDatabase db;
    /**
     * DaoMaster
     */
    private DaoMaster mDaoMaster;
    /**
     * DaoSession
     */
    private DaoSession mDaoSession;
    /**
     * 上下文
     */
    private Context context;

    private static DbController mDbController;

    /**
     * dao
     */
    private StudentsDao studentsDao;


    /**
     * 获取单例
     */
    public static DbController getInstance(Context context) {
        if (mDbController == null) {
            synchronized (DbController.class) {
                if (mDbController == null) {
                    mDbController = new DbController(context);
                }
            }
        }
        return mDbController;
    }


    /**
     * 初始化
     *
     * @param context
     */
    public DbController(Context context) {
        this.context = context;
        mHelper = new DaoMaster.DevOpenHelper(context, "person.db", null);
        mDaoMaster = new DaoMaster(getWritableDatabase());
        mDaoSession = mDaoMaster.newSession();
        studentsDao = mDaoSession.getStudentsDao();
        //  diningRoomDao = mDaoSession.getDiningRoomDao();


    }


    /**
     * 获取可写数据库
     *
     * @return
     */
    private SQLiteDatabase getWritableDatabase() {
        if (mHelper == null) {
            mHelper = new DaoMaster.DevOpenHelper(context, "person.db", null);
        }
        SQLiteDatabase db = mHelper.getWritableDatabase();
        return db;
    }

    public boolean execSQL(String sql) {
        if (db == null) {
            db = getWritableDatabase();
        }
        try {
            db.execSQL(sql);
            return true;
        } catch (SQLException s) {
            Log.e("SQLException", s.getMessage());
            return false;
        }
    }

    /**
     * 插入学生数据
     *
     * @param
     */
    public void insertStudents(Students studentsData) {
        studentsDao.insertOrReplace(studentsData);
    }


    /**
     * 批量插入学生数据
     */
    public void saveStudentLists(final List<StudentBean> list) {
        if (list == null || list.isEmpty()) {
            return;
        }
        studentsDao.getSession().runInTx(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < list.size(); i++) {
                    Students studentsData = new Students();
                    studentsData.setLast_change_id(list.get(i).getLast_change_id());
                    studentsData.setClass_name(list.get(i).getClass_name());
                    studentsDao.insertOrReplace(studentsData);

                }
            }
        });

    }


    /**
     * 修改学生信息
     *
     * @param studentsData 学生信息
     */
    public void upDateStudents(Students studentsData) {
        studentsDao.update(studentsData);
    }

    /**
     * 查询所有学生数据
     */
    public List<Students> searchAllStudents() {
        List<Students> list = studentsDao.queryBuilder().list();
        return list;
    }

    /**
     * 按条件查询数据学生信息
     */
    public Students searchStudentsByID(int user_id) {
        Students studentsData = studentsDao.queryBuilder().where(StudentsDao.Properties.Id.eq(user_id)).build().unique();
        return studentsData;
    }

    /**
     * 按条件查询卡号
     */
    public Students searchStudentsByCard(String card) {
        Students studentsData = studentsDao.queryBuilder().where(StudentsDao.Properties.Real_number.eq(card)).build().unique();
        return studentsData;
    }

    /**
     * 删除学生数据
     */
    public void delete(String userId) {
        studentsDao.queryBuilder().where(StudentsDao.Properties.Id.eq(userId)).buildDelete().executeDeleteWithoutDetachingEntities();
    }

    /**
     * 清理所有学生数据
     */
    public void deleteStudentData() {
        studentsDao.deleteAll();
    }

}

每天进步一点点。。。