【Android】第4章 数据存储(上)
来源:互联网 发布:淘宝网男装鞋 编辑:程序博客网 时间:2024/06/07 10:17
4.0debug调试
1、debug调试的作用
1)跟踪程序的代理,找出问题出现的地方,更快的解决问题
2)梳理已有功能代码的运行逻辑流程
2、如何进行debug调试
1). 开启Debug的一般步骤:
a. 设置断点 : 在某个方法体的特定行打断点
b. Debug运行
c. 进入Debug视图模式, 运行至断点处悬停
2).Debug视图的组成
1 : 常用debug操作工具栏
2 : 线程堆栈视图
3 : 变量视图、断点视图
4 : 代码视图
3). debug常用操作工具栏
1 :代表执行完当前断点区域进入下一个断点或结束
2 : 中断程序
3 : 进入当前行的方法(单步跳入)
4 : 执行完当前行, 进入下一行(单步跳过)
5 : 跳出当前方法执行(单步跳出)
4.1理论概述
1、Android数据存储方式
2.测试用例
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <TextView android:id="@+id/textView1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="数据存储方式" android:textSize="20sp" android:textColor="#ff0000" android:layout_margin="10dp"/> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:onClick="onClickSP" android:text="SharedPreference存储" /> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:onClick="onClickIF" android:text="内部文件存储" /> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:onClick="onClickOF" android:text="外部文件存储" /> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:onClick="onClickDB" android:text="数据库存储" /> <Button android:layout_width="match_parent" android:layout_height="wrap_content" android:onClick="onClickNW" android:text="网络存储存储" /></LinearLayout>
import android.app.Activity;import android.content.Intent;import android.os.Bundle;import android.view.View;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}// 测试sp存储public void onClickSP(View v) {startActivity(new Intent(this, SpActivity.class));}// 测试手机内部文件存储public void onClickIF(View v) {startActivity(new Intent(this, IFActivity.class));}// 测试手机外部文件存储public void onClickOF(View v) {startActivity(new Intent(this, OFActivity.class));}// 测试Sqlite数据库存储public void onClickDB(View v) {startActivity(new Intent(this, DBActivity.class));}public void onClickNW(View v) {startActivity(new Intent(this, NetworkActivity.class));}}
2、测试用例
4.2sp存储说明
SP存储专门用来存储一些单一的小数据
存储数据的类型: boolean, float, int, long,String
数据保存的路径: /data/data/packageName/shared_prefs/yyy.xml
可以设置数据只能是当前应用读取, 而别的应用不可以
应用卸载时会删除此数据
1) SharedPreferences: 对应sp文件的接口
a) context. getSharedPreferences (String name, int mode): 得到SP对象
i. name: 文件名(不带.xml)
ii. mode: 生成的文件模式(是否是私有的,即其它应用是否可以访问)
b) Editor sp.edit() : 得到Editor对象
c) Xxx sp.getXxx(name,defaultValue): 根据name得到对应的数据
2) Editor : 能更新sp文件的接口
a) Editor put(name, value) : 保存一个键值对, 没有真正保存到文件中
b) Editor remove(name)
c) commit(): 提交, 数据真正保存到文件中了
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <EditText android:id="@+id/et_sp_key" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="存储的key" /> <EditText android:id="@+id/et_sp_value" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="存储的value" /> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <Button android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="save" android:text="保 存" /> <Button android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="read" android:text="读 取" /> </LinearLayout></LinearLayout>
4.3sp存储_保存数据
1、分析思路
1. 得到sp对象
2. 得到editor对象
3. 得到输入的key/value
4. 使用editor保存key-value
5. 提示
2、源码
package com.xiongpan.l04_datastorage;import android.app.Activity;import android.content.Context;import android.content.SharedPreferences;import android.content.SharedPreferences.Editor;import android.os.Bundle;import android.view.View;import android.widget.EditText;import android.widget.Toast;/** * 测试sp存储的界面 */public class SpActivity extends Activity {private EditText et_sp_key;private EditText et_sp_value;private SharedPreferences sp;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_sp);et_sp_key = (EditText) findViewById(R.id.et_sp_key);et_sp_value = (EditText) findViewById(R.id.et_sp_value);//1. 得到sp对象sp = getSharedPreferences("xiongpan", Context.MODE_PRIVATE);}public void save(View v) {//2. 得到editor对象Editor edit = sp.edit();//3. 得到输入的key/valueString key = et_sp_key.getText().toString();String value = et_sp_value.getText().toString();//4. 使用editor保存key-valueedit.putString(key, value).commit();//5. 提示Toast.makeText(this, "保存完成!", 0).show();}
4.4sp存储_读取数据
##测试sp存储_读取
1. 得到输入的key
2. 根据key读取对应的value
3. 显示
public void read(View v) {//1. 得到输入的keyString key = et_sp_key.getText().toString();//2. 根据key读取对应的valueString value = sp.getString(key, null);//3. 显示if(value==null) {Toast.makeText(this, "没有找到对应的value", 0).show();} else {et_sp_value.setText(value);}}}
4.5手机内部文件存储_说明
1、说明
应用运行需要的一些较大的数据或图片可以用文件保存的手机内部
文件类型: 任意
数据保存的路径: /data/data/projectPackage/files/
可以设置数据只能是当前应用读取, 而别的应用不可以
应用卸载时会删除此数据
4.6手机内部文件存储_保存文件
1、思路分析
测试手机内部文件存储_保存
1. 得到InputStream-->读取assets下的logo.png
2. 得到OutputStream-->/data/data/packageName/files/logo.png
3. 边读边写
4. 提示
2、布局源码
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <TextView android:id="@+id/textView1" android:layout_width="match_parent" android:layout_height="wrap_content" android:padding="5dp" android:text="1. 将asserts下的logo.png保存到手机内部\n2. 读取手机内部图片文件显示" android:textColor="#ff0000" android:textSize="15sp" /> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <Button android:id="@+id/btn_if_save" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="save" android:text="保 存" /> <Button android:id="@+id/btn_if_read" android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="read" android:text="读 取" /> </LinearLayout> <ImageView android:id="@+id/iv_if" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@drawable/ic_launcher" /></LinearLayout>package com.xiongpan.l04_datastorage;import java.io.FileOutputStream;import java.io.IOException;import java.io.InputStream;import android.app.Activity;import android.content.Context;import android.content.res.AssetManager;import android.graphics.Bitmap;import android.graphics.BitmapFactory;import android.os.Bundle;import android.view.View;import android.widget.ImageView;import android.widget.Toast;/** * 测试手机内部文件存储 * * @author 张晓飞 * */public class IFActivity extends Activity {private ImageView iv_if;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_if);iv_if = (ImageView) findViewById(R.id.iv_if);}public void save(View v) throws IOException {//1. 得到InputStream-->读取assets下的logo.png//得到AssetManager AssetManager manager = getAssets();//读取文件InputStream is = manager.open("logo.png");//2. 得到OutputStream-->/data/data/packageName/files/logo.pngFileOutputStream fos = openFileOutput("logo.png", Context.MODE_PRIVATE);//3. 边读边写byte[] buffer = new byte[1024];int len = -1;while((len=is.read(buffer))!=-1) {fos.write(buffer, 0, len);}fos.close();is.close();//4. 提示Toast.makeText(this, "保存完成", 0).show();}
4.7手机内部文件存储_读取文件
1、思路分析
##测试手机内部文件存储_读取
1. 得到图片文件的路径
2. 读取加载图片文件得到bitmap对象
3. 将其设置到imageView中显示
2、相关API
读取文件
FileInputStream fis = openFileInput("logo.png");
保存文件
FileOutputStream fos = openFileOutput("logo.png",MODE_PRIVATE)
得到files文件夹对象
File filesDir = getFilesDir();
操作asserts下的文件
得到AssetManager : context.getAssets();
读取文件: InputStream open(filename);
加载图片文件
Bitmap BitmapFactory.decodeFile(StringpathName) // .bmp/.png/.jpg
public void read(View v) {// /data/data/packageName/files/logo.png//1. 得到图片文件的路径// /data/data/packageName/filesString filesPath = getFilesDir().getAbsolutePath();String imagePath = filesPath+"/logo.png";//2. 读取加载图片文件得到bitmap对象Bitmap bitmap = BitmapFactory.decodeFile(imagePath);//3. 将其设置到imageView中显示iv_if.setImageBitmap(bitmap);}}
4.8_手机外部文件存储_说明
1、说明
• 应用运行用到的数据文件(如图片)可以保存到sd卡中
• 文件类型: 任意
• 数据保存的路径:
– 路径1: /storage/sdcard/Android/data/packageName/files/
– 路径2: /storage/sdcard/xxx/
• 路径1 :其它应用可以访问,应用卸载时删除
• 路径2 : 其它应用可以访问, 应用卸载时不会删除
• 必须保证sd卡挂载在手机上才能读写, 否则不能操作
2、相关API
• Environment : 操作SD卡的工具类
– 得到SD卡的状态:Environment.getExternalStorageState()
– 得到SD卡的路径:Environment.getExternalStorageDirectory()
– SD卡可读写的挂载状态值:Environment.MEDIA_MOUNTED
– context. getExternalFilesDir():
– 得到/mnt/sdcard/Android/data/pageckage_name/files/xxx.txt
– 操作SD卡的权限:
– android.permission.WRITE_EXTERNAL_STORAGE
• /mnt/sdcard/Android/data/pageckage_name/files/
• /mnt/sdcard/Android/data/pageckage_name/files/music/
• /mnt/sdcard/Android/data/pageckage_name/files/images/
4.9_手机外部文件存储_保存文件
1、思路分析
1. 判断sd卡状态, 如果是挂载的状态才继续, 否则提示
2. 读取输入的文件名/内容
3. 得到指定文件的OutputStream
1).得到sd卡下的files路径
2).组成完整路径
3). 创建FileOutputStream
4. 写数据
5. 提示
2、布局
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <EditText android:id="@+id/et_of_name" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="存储的文件名" /> <EditText android:id="@+id/et_of_content" android:layout_width="match_parent" android:layout_height="wrap_content" android:hint="存储的文件内容" /> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <Button android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="save" android:text="保 存" /> <Button android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="read" android:text="读 取" /> </LinearLayout> <LinearLayout android:layout_width="match_parent" android:layout_height="wrap_content" > <Button android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="save2" android:text="保 存2" /> <Button android:layout_width="0dp" android:layout_height="wrap_content" android:layout_weight="1" android:onClick="read2" android:text="读 取2" /> </LinearLayout></LinearLayout>
3、操作SD卡需要注意的问题
<!--操作sd卡 -->
<uses-permissionandroid:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
4、源码
package com.xiongpan.l04_datastorage;import java.io.ByteArrayOutputStream;import java.io.File;import java.io.FileInputStream;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import android.app.Activity;import android.os.Bundle;import android.os.Environment;import android.view.View;import android.widget.EditText;import android.widget.Toast;/** * 测试手机外部文件存储 * * @author 张晓飞 * */public class OFActivity extends Activity {private EditText et_of_name;private EditText et_of_content;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_of);et_of_name = (EditText) findViewById(R.id.et_of_name);et_of_content = (EditText) findViewById(R.id.et_of_content);}public void save(View v) throws IOException {//1. 判断sd卡状态, 如果是挂载的状态才继续, 否则提示if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {//2. 读取输入的文件名/内容String fileName = et_of_name.getText().toString();String content = et_of_content.getText().toString();//3. 得到指定文件的OutputStream//1).得到sd卡下的files路径String filesPath = getExternalFilesDir(null).getAbsolutePath();//2).组成完整路径String filePath = filesPath+"/"+fileName;//3). 创建FileOutputStreamFileOutputStream fos = new FileOutputStream(filePath);//4. 写数据 fos.write(content.getBytes("utf-8"));fos.close();//5. 提示Toast.makeText(this, "保存完成", 0).show();} else {Toast.makeText(this, "sd卡没有挂载", 0).show();}}
4.10手机外部文件是存储_读取文件
public void read(View v) throws Exception {// 1. 判断sd卡状态, 如果是挂载的状态才继续, 否则提示if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {// 2. 读取输入的文件名String fileName = et_of_name.getText().toString();// 3. 得到指定文件的InputStream// 1).得到sd卡下的files路径String filesPath = getExternalFilesDir(null).getAbsolutePath();// 2).组成完整路径String filePath = filesPath + "/" + fileName;// 3). 创建FileInputStreamFileInputStream fis = new FileInputStream(filePath);// 4. 读取数据, 成StringByteArrayOutputStream baos = new ByteArrayOutputStream();byte[] buffer = new byte[1024];int len = -1;while((len=fis.read(buffer))!=-1) {baos.write(buffer, 0, len);}String content = baos.toString();// 5. 显示et_of_content.setText(content);} else {Toast.makeText(this, "sd卡没有挂载", 0).show();}}
4.11手机外部文件存储_保存读取
// /storage/sdcard/xiongpan/xxx.txtpublic void save2(View v) throws IOException {//1. 判断sd卡状态, 如果是挂载的状态才继续, 否则提示if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {//2. 读取输入的文件名/内容String fileName = et_of_name.getText().toString();String content = et_of_content.getText().toString();//3. 得到指定文件的OutputStream//1). /storage/sdcard/String sdPath = Environment.getExternalStorageDirectory().getAbsolutePath();//2). /storage/sdcard/xiongpan/(创建文件夹)File file = new File(sdPath+"/xiongpan");if(!file.exists()) {file.mkdirs();//创建文件夹}//3). /storage/sdcard/xiongpan/xxx.txtString filePath = sdPath+"/xiongpan/"+fileName;//4). 创建输出流FileOutputStream fos = new FileOutputStream(filePath);//4. 写数据 fos.write(content.getBytes("utf-8"));fos.close();//5. 提示Toast.makeText(this, "保存完成", 0).show();} else {Toast.makeText(this, "sd卡没有挂载", 0).show();}}public void read2(View v) throws Exception {// 1. 判断sd卡状态, 如果是挂载的状态才继续, 否则提示if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {// 2. 读取输入的文件名String fileName = et_of_name.getText().toString();// 3. 得到指定文件的InputStreamString sdPath = Environment.getExternalStorageDirectory().getAbsolutePath();String filePath = sdPath+"/xiongpan/"+fileName;FileInputStream fis = new FileInputStream(filePath);// 4. 读取数据, 成StringByteArrayOutputStream baos = new ByteArrayOutputStream();byte[] buffer = new byte[1024];int len = -1;while((len=fis.read(buffer))!=-1) {baos.write(buffer, 0, len);}String content = baos.toString();fis.close();// 5. 显示et_of_content.setText(content);} else {Toast.makeText(this, "sd卡没有挂载", 0).show();}}}
4.12_应用练习1_界面布局与流程分析
修改防盗的名称
• 功能描述:
1. 长按手机防盗, 显示修改的Dialog
2. 通过dialog修改手机防盗名称
• 关键技术点:
1. SharedPreferences的使用
2. AlertDialog的使用
3.GridView+BaseAdapter的使用
1. 给gridView的item添加长按监听(只能第一个有响应)
2. 在回调方法中显示AlertDialog
3. 处理点击修改
1). 界面更新
2). 将名称保存到sp中
4. 如何保证退出后再进入显示修改过的名称
4.13_应用练习1_修改防盗名称
1、布局
<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="80dp" android:layout_height="80dp" android:orientation="vertical" android:gravity="center"> <ImageView android:id="@+id/iv_item_icon" android:layout_width="60dp" android:layout_height="60dp" android:src="@drawable/ic_launcher" /> <TextView android:id="@+id/tv_item_name" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="名称" android:textSize="16sp"/></LinearLayout>
2、源码
import android.app.Activity;import android.app.AlertDialog;import android.content.Context;import android.content.DialogInterface;import android.content.SharedPreferences;import android.os.Bundle;import android.view.View;import android.widget.AdapterView;import android.widget.EditText;import android.widget.AdapterView.OnItemLongClickListener;import android.widget.TextView;import android.widget.Toast;import android.widget.AdapterView.OnItemClickListener;import android.widget.GridView;public class MainActivity extends Activity implements OnItemLongClickListener {private GridView gv_main;private MainAdapter adapter;String[] names = new String[] { "手机防盗", "通讯卫士", "软件管理", "流量管理", "进程管理","手机杀毒", "缓存清理", "高级工具", "设置中心" };int[] icons = new int[] { R.drawable.widget01, R.drawable.widget02,R.drawable.widget03, R.drawable.widget04, R.drawable.widget05,R.drawable.widget06, R.drawable.widget07, R.drawable.widget08,R.drawable.widget09 };private SharedPreferences sp;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);gv_main = (GridView) findViewById(R.id.gv_main);adapter = new MainAdapter(this, names, icons);gv_main.setAdapter(adapter);//给gridView的Item设置点击监听gv_main.setOnItemClickListener(new OnItemClickListener() {@Overridepublic void onItemClick(AdapterView<?> parent, View view,int position, long id) {//得到当前点击的名称String name = names[position];//提示Toast.makeText(MainActivity.this, name, 1).show();}});//给gridView的item添加长按监听(只能第一个有响应)gv_main.setOnItemLongClickListener(this);sp = getSharedPreferences("xfzhang", Context.MODE_PRIVATE);}@Overridepublic boolean onItemLongClick(AdapterView<?> parent, View view,int position, long id) {if(position==0) {//得到当前显示的名称final TextView textView = (TextView) view.findViewById(R.id.tv_item_name);String name = textView.getText().toString();//为dialog准备输入框对象final EditText editText = new EditText(this);editText.setHint(name);//显示AlertDialognew AlertDialog.Builder(this).setTitle("修改名称").setView(editText).setPositiveButton("修改", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {String newName = editText.getText().toString();//1). 界面更新textView.setText(newName);//2). 将名称保存到sp中sp.edit().putString("NAME", newName).commit();}}).setNegativeButton("取消", null).show();}return true;}}package com.xiongpan.app04_gridview;import android.content.Context;import android.content.SharedPreferences;import android.view.View;import android.view.ViewGroup;import android.widget.BaseAdapter;import android.widget.ImageView;import android.widget.TextView;public class MainAdapter extends BaseAdapter {private String[] names;private int[] icons;private Context context;private SharedPreferences sp;public MainAdapter(Context context, String[] names, int[] icons) {super();this.context = context;this.names = names;this.icons = icons;sp = context.getSharedPreferences("xfzhang", Context.MODE_PRIVATE);}@Overridepublic int getCount() {return names.length;}@Overridepublic Object getItem(int position) {return names[position];}@Overridepublic long getItemId(int position) {return 0;}@Overridepublic View getView(int position, View convertView, ViewGroup parent) {if(convertView==null) {convertView = View.inflate(context, R.layout.item_main, null);}ImageView imageView = (ImageView) convertView.findViewById(R.id.iv_item_icon);TextView textView = (TextView) convertView.findViewById(R.id.tv_item_name);imageView.setImageResource(icons[position]);textView.setText(names[position]);if(position==0) {//从sp中读取保存的名称, 如果存在显示String savedName = sp.getString("NAME", null);if(savedName!=null) {textView.setText(savedName);}}return convertView;}}
- 【Android】第4章 数据存储(上)
- [Android][第一行代码][第 6 章 数据存储]
- Android学习第九天---Android数据存储
- 第8章 Android数据存储与IO
- 第 6 章 数据存储
- Android 数据存储 《第一行代码》
- 【第一行代码】Android数据存储
- Android攻城狮的第二门课(第3季)第3章 数据存储之文件存储
- Android系统中的数据存储方式(上)
- Android基础数据存储(上)
- Android攻城狮的第二门课(第3季)第1章 数据存储之SharedPreferences
- Android攻城狮的第二门课(第3季)第2章 数据存储之SQLite
- Android学习9Android上的数据存储01
- Android学习10Android上的数据存储02
- 第2章 变量和数据存储
- 第6章 数据存储全方案
- 数据存储上
- Android学习之 第 6 章 数据存储全方案,详解 持久化技术
- SpringBoot入门(一)
- 0024_Swap Nodes in Pairs
- 32,64位编译环境下的sizeof问题经常出现。
- eval解析及反引号与$()区别
- 机器学习--神经网络算法系列--梯度下降与随机梯度下降算法
- 【Android】第4章 数据存储(上)
- POJ 2421 Constructing Roads 笔记
- 简单导出excel
- 小背包
- 面试题40—数组只出现一次的数字
- 【转载 Hadoop&Spark 动手实践 2】Hadoop2.7.3 HDFS理论与动手实践
- java中jvm之我理解
- 面试题41—和为s的两个数
- 如何解决LSTM循环神经网络中的超长序列问题