• Android应用开发基础之三:数据存储和界面展现(三)


    生成XML文件备份短信

    • 创建几个虚拟的短信对象,存在list中
    • 备份数据通常都是备份至sd卡

    使用StringBuffer拼接字符串

    • 把整个xml文件所有节点append到sb对象里

      sb.append("<?xml version='1.0' encoding='utf-8' standalone='yes' ?>");
      //添加smss的开始节点
      sb.append("<smss>");
      .......
      
    • 把sb写到输出流中

      fos.write(sb.toString().getBytes());
      

    使用XMl序列化器生成xml文件

    • 得到xml序列化器对象

      XmlSerializer xs = Xml.newSerializer();
      
    • 给序列化器设置输出流

      File file = new File(Environment.getExternalStorageDirectory(), "backupsms.xml");
      FileOutputStream fos = new FileOutputStream(file);
      //给序列化器指定好输出流
      xs.setOutput(fos, "utf-8");
      
    • 开始生成xml文件

      xs.startDocument("utf-8", true);
      xs.startTag(null, "smss");
      ......
      

    pull解析xml文件

    • 先自己写一个xml文件,存一些天气信息

    拿到xml文件

        InputStream is = getClassLoader().getResourceAsStream("weather.xml");
    

    拿到pull解析器

        XmlPullParser xp = Xml.newPullParser();
    

    开始解析

    • 拿到指针所在当前节点的事件类型

      int type = xp.getEventType();
      
    • 事件类型主要有五种

      • START_DOCUMENT:xml头的事件类型
      • END_DOCUMENT:xml尾的事件类型
      • START_TAG:开始节点的事件类型
      • END_TAG:结束节点的事件类型
      • TEXT:文本节点的事件类型
    • 如果获取到的事件类型不是END_DOCUMENT,就说明解析还没有完成,如果是,解析完成,while循环结束

      while(type != XmlPullParser.END_DOCUMENT)
      
    • 当我们解析到不同节点时,需要进行不同的操作,所以判断一下当前节点的name

      • 当解析到weather的开始节点时,new出list
      • 当解析到city的开始节点时,创建city对象,创建对象是为了更方便的保存即将解析到的文本
      • 当解析到name开始节点时,获取下一个节点的文本内容,temp、pm也是一样

        case XmlPullParser.START_TAG:
        //获取当前节点的名字
            if("weather".equals(xp.getName())){
                citys = new ArrayList<City>();
            }
            else if("city".equals(xp.getName())){
                city = new City();
            }
            else if("name".equals(xp.getName())){
                //获取当前节点的下一个节点的文本
                String name = xp.nextText();
                city.setName(name);
            }
            else if("temp".equals(xp.getName())){
                String temp = xp.nextText();
                city.setTemp(temp);
            }
            else if("pm".equals(xp.getName())){
                String pm = xp.nextText();
                city.setPm(pm);
            }
            break;
        
    • 当解析到city的结束节点时,说明city的三个子节点已经全部解析完了,把city对象添加至list

      case XmlPullParser.END_TAG:
          if("city".equals(xp.getName())){
                  citys.add(city);
          }
      

    测试

    • 黑盒测试
      • 测试逻辑业务
    • 白盒测试

      • 测试逻辑方法
    • 根据测试粒度

      • 方法测试:function test
      • 单元测试:unit test
      • 集成测试:integration test
      • 系统测试:system test
    • 根据测试暴力程度

      • 冒烟测试:smoke test
      • 压力测试:pressure test

    单元测试junit

    • 定义一个类继承AndroidTestCase,在类中定义方法,即可测试该方法

    • 在指定指令集时,targetPackage指定你要测试的应用的包名

      <instrumentation 
      android:name="android.test.InstrumentationTestRunner"
      android:targetPackage="com.ibky.junit"
      ></instrumentation>
      
    • 定义使用的类库

      <uses-library android:name="android.test.runner"></uses-library>
      
    • 断言的作用,检测运行结果和预期是否一致

    • 如果应用出现异常,会抛给测试框架

    SQLite数据库

    • 轻量级关系型数据库
    • 创建数据库需要使用的api:SQLiteOpenHelper

      • 必须定义一个构造方法:

        //arg1:数据库文件的名字
        //arg2:游标工厂
        //arg3:数据库版本
        public MyOpenHelper(Context context, String name, CursorFactory factory, int version){}
        
      • 数据库被创建时会调用:onCreate方法
      • 数据库升级时会调用:onUpgrade方法

    创建数据库

    //创建OpenHelper对象
    MyOpenHelper oh = new MyOpenHelper(getContext(), "person.db", null, 1);
    //获得数据库对象,如果数据库不存在,先创建数据库,后获得,如果存在,则直接获得
    SQLiteDatabase db = oh.getWritableDatabase();
    
    • getWritableDatabase():打开可读写的数据库
    • getReadableDatabase():在磁盘空间不足时打开只读数据库,否则打开可读写数据库
    • 在创建数据库时创建表

      public void onCreate(SQLiteDatabase db) {
          // TODO Auto-generated method stub
          db.execSQL("create table person (_id integer primary key autoincrement, name char(10), phone char(20), money integer(20))");
      }
      

    数据库的增删改查

    SQL语句

    • insert into person (name, phone, money) values ('张三', '159874611', 2000);
    • delete from person where name = '李四' and _id = 4;
    • update person set money = 6000 where name = '李四';
    • select name, phone from person where name = '张三';

    执行SQL语句实现增删改查

        //插入
        db.execSQL("insert into person (name, phone, money) values (?, ?, ?);", new Object[]{"张三", 15987461, 75000});
        //查找
        Cursor cs = db.rawQuery("select _id, name, money from person where name = ?;", new String[]{"张三"});
    
    • 测试方法执行前会调用此方法

      protected void setUp() throws Exception {
          super.setUp();
          //                  获取虚拟上下文对象
          oh = new MyOpenHelper(getContext(), "people.db", null, 1);
      }
      

    使用api实现增删改查

    • 插入

      //以键值对的形式保存要存入数据库的数据
      ContentValues cv = new ContentValues();
      cv.put("name", "刘能");
      cv.put("phone", 1651646);
      cv.put("money", 3500);
      //返回值是改行的主键,如果出错返回-1
      long i = db.insert("person", null, cv);
      
    • 删除

      //返回值是删除的行数
      int i = db.delete("person", "_id = ? and name = ?", new String[]{"1", "张三"});
      
    • 修改

      ContentValues cv = new ContentValues();
      cv.put("money", 25000);
      int i = db.update("person", cv, "name = ?", new String[]{"赵四"});
      
    • 查询

      //arg1:要查询的字段
      //arg2:查询条件
      //arg3:填充查询条件的占位符
      Cursor cs = db.query("person", new String[]{"name", "money"}, "name = ?", new String[]{"张三"}, null, null, null);
      while(cs.moveToNext()){
          //                          获取指定列的索引值
          String name = cs.getString(cs.getColumnIndex("name"));
          String money = cs.getString(cs.getColumnIndex("money"));
          System.out.println(name + ";" + money);
      }
      

    事务

    • 保证多条SQL语句要么同时成功,要么同时失败
    • 最常见案例:银行转账
    • 事务api

      try {
          //开启事务
          db.beginTransaction();
          ...........
          //设置事务执行成功
          db.setTransactionSuccessful();
      } finally{
          //关闭事务
          //如果此时已经设置事务执行成功,则sql语句生效,否则不生效
          db.endTransaction();
      }
      

    把数据库的数据显示至屏幕

    1. 任意插入一些数据
    2. 定义业务bean:Person.java
    3. 读取数据库的所有数据

      Cursor cs = db.query("person", null, null, null, null, null, null);
      while(cs.moveToNext()){
          String name = cs.getString(cs.getColumnIndex("name"));
          String phone = cs.getString(cs.getColumnIndex("phone"));
          String money = cs.getString(cs.getColumnIndex("money"));
          //把读到的数据封装至Person对象
          Person p = new Person(name, phone, money);
          //把person对象保存至集合中
          people.add(p);
      }
      
    4. 把集合中的数据显示至屏幕

       LinearLayout ll = (LinearLayout) findViewById(R.id.ll);
       for(Person p : people){
           //创建TextView,每条数据用一个文本框显示
           TextView tv = new TextView(this);
           tv.setText(p.toString());
           //把文本框设置为ll的子节点
           ll.addView(tv);
       }
      
    5. 分页查询

      Cursor cs = db.query("person", null, null, null, null, null, null, "0, 10");
      

    ListView

    • 就是用来显示一行一行的条目的
    • MVC结构
      • M:model模型层,要显示的数据 ————people集合
      • V:view视图层,用户看到的界面 ————ListView
      • c:control控制层,操作数据如何显示 ————adapter对象
    • 每一个条目都是一个View对象

    BaseAdapter

    • 必须实现的两个方法

      • 第一个

        //系统调用此方法,用来获知模型层有多少条数据
        @Override
        public int getCount() {
            return people.size();
        }
        
      • 第二个

        //系统调用此方法,获取要显示至ListView的View对象
        //position:是return的View对象所对应的数据在集合中的位置
        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            System.out.println("getView方法调用" + position);
            TextView tv = new TextView(MainActivity.this);
            //拿到集合中的元素
            Person p = people.get(position);
            tv.setText(p.toString());
        
            //把TextView的对象返回出去,它会变成ListView的条目
            return tv;
        }
        
    • 屏幕上能显示多少个条目,getView方法就会被调用多少次,屏幕向下滑动时,getView会继续被调用,创建更多的View对象显示至屏幕
    • 布局填充器的三种写法
     1 public class MainActivity extends Activity {
     2 
     3     List<Person> personList;
     4     @Override
     5     protected void onCreate(Bundle savedInstanceState) {
     6         super.onCreate(savedInstanceState);
     7         setContentView(R.layout.activity_main);
     8         
     9         personList = new ArrayList<Person>();
    10         
    11         //查询数据库,获取50个哥们的数据
    12         MyOpenHelper oh = new MyOpenHelper(this);
    13         SQLiteDatabase db = oh.getWritableDatabase();
    14         Cursor cursor = db.query("person", null, null, null, null, null, null, null);
    15         while(cursor.moveToNext()){
    16             String name = cursor.getString(1);
    17             String phone = cursor.getString(2);
    18             String money = cursor.getString(3);
    19             
    20             Person p = new Person(name, phone, money);
    21             personList.add(p);
    22         }
    23         
    24         ListView lv = (ListView) findViewById(R.id.lv);
    25         //给ListView设置一个适配器,适配器的作用,就是把集合中的数据显示至ListView
    26         lv.setAdapter(new MyAdapter());
    27     }
    28     class MyAdapter extends BaseAdapter{
    29 
    30         //返回模型层中数据的数量
    31         @Override
    32         public int getCount() {
    33             return personList.size();
    34         }
    35         
    36         //返回一个View对象,被返回的View对象会作为ListView的一个条目显示至屏幕
    37         //position:返回的View对象,在整个ListView当中,属于第几个条目
    38         //convertView:当条目被划出屏幕时,该条目会被缓存起来,当getView方法再次调用时,缓存条目就会作为convertView传入getView方法中
    39         @Override
    40         public View getView(int position, View convertView, ViewGroup parent) {
    41 //            System.out.println("getview调用:"+position);
    42 //            TextView tv = new TextView(MainActivity.this);
    43 //            tv.setText(personList.get(position).toString());
    44 //            tv.setTextSize(18);
    45             
    46             
    47             //1.拿到布局填充器对象
    48 //            LayoutInflater li = LayoutInflater.from(MainActivity.this);
    49 //            View v = li.inflate(R.layout.listview_item, null);
    50             
    51             //2.通过获取系统服务拿到布局填充器对象
    52 //            LayoutInflater li = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);
    53 //            View v = li.inflate(R.layout.listview_item, null);
    54             
    55             View v = null;
    56             if(convertView == null){
    57             
    58                 //3.通过资源id,把指定的布局文件填充成View对象
    59                 v = View.inflate(MainActivity.this, R.layout.listview_item, null);
    60                 
    61             }
    62             else{
    63                 v = convertView;
    64             }
    65             
    66             //给view对象三个textview赋值
    67             TextView tv_name = (TextView) v.findViewById(R.id.tv_name);
    68             TextView tv_phone = (TextView) v.findViewById(R.id.tv_phone);
    69             TextView tv_salary = (TextView) v.findViewById(R.id.tv_salary);
    70             
    71             Person p = personList.get(position);
    72             tv_name.setText(p.getName());
    73             tv_phone.setText(p.getPhone());
    74             tv_salary.setText(p.getMoney());
    75             return v;
    76         }
    77         
    78         @Override
    79         public Object getItem(int position) {
    80             return null;
    81         }
    82 
    83         @Override
    84         public long getItemId(int position) {
    85             return 0;
    86         }
    87 
    88         
    89         
    90     }

    条目的缓存

    • 当条目划出屏幕时,系统会把该条目缓存至内存,当该条目再次进入屏幕,系统在重新调用getView时会把缓存的条目作为convertView参数传入,但是传入的条目不一定是之前被缓存的该条目,即系统有可能在调用getView方法获取第一个条目时,传入任意一个条目的缓存  

  • 相关阅读:
    Echarts使用笔记
    vue导出pdf
    mvn上传pom/jar至Nexus私服
    Linux下安装Redis
    国内首发,这款 Serverless 云原生一体化部署工具正式开源!
    使用云托管快速部署CMS内容管理系统
    云开发走进高校,腾讯云联合新工科联盟探索新生工程教育模式
    从0到3000万,TA们用云开发这样做
    基于 ThinkJS 的云开发体验
    使用小程序·云开发静态托管实现免鉴权h5跳转小程序
  • 原文地址:https://www.cnblogs.com/McCa/p/4204952.html
Copyright © 2020-2023  润新知