百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 热门文章 > 正文

数据存储与访问之文件存储读写

bigegpt 2024-08-08 12:01 2 浏览

1.Android文件的操作模式

学过Java的同学都知道,我们新建文件,然后就可以写入数据了,但是Android却不一样,因为Android是 基于Linux的,我们在读写文件的时候,还需加上文件的操作模式,Android中的操作模式如下:


2.文件的相关操作方法

3.文件读写的实现

Android中的文件读写和Java中的文件I/O相同,流程也很简单,下面我们来写个简单的示例:

实现效果图:

PS:这里用的是模拟器,因为笔者的N5并没有root,看不到文件的存储目录,下面我们打开DDMS 的File Exploer可以看到,在data/data/<包名>/file中有我们写入的文件:

我们可以点击右上角的响应图标将文件导入到电脑中,并且打开验证写入的内容:

代码实现:

首先是布局文件:main_activity.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

xmlns:tools="http://schemas.android.com/tools"

android:id="@+id/LinearLayout1"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical"

tools:context="com.jay.example.filedemo1.MainActivity">

<TextView

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="@string/nametitle" />

<EditText

android:id="@+id/editname"

android:layout_width="match_parent"

android:layout_height="wrap_content" />

<TextView

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="@string/detailtitle" />

<EditText

android:id="@+id/editdetail"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:minLines="2" />

<LinearLayout

android:layout_width="fill_parent"

android:layout_height="wrap_content"

android:orientation="horizontal">

<Button

android:id="@+id/btnsave"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="@string/btnwrite" />

<Button

android:id="@+id/btnclean"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="@string/btnclean" />

</LinearLayout>

<Button

android:id="@+id/btnread"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="@string/btnread" />

</LinearLayout>

然后我们来写一个文件协助类:FileHelper.java

/**

* Created by Jay on 2015/9/1 0001.

*/

public class FileHelper {

private Context mContext;

public FileHelper() {

}

public FileHelper(Context mContext) {

super();

this.mContext = mContext;

}

/*

* 这里定义的是一个文件保存的方法,写入到文件中,所以是输出流

* */

public void save(String filename, String filecontent) throws Exception {

//这里我们使用私有模式,创建出来的文件只能被本应用访问,还会覆盖原文件哦

FileOutputStream output = mContext.openFileOutput(filename, Context.MODE_PRIVATE);

output.write(filecontent.getBytes()); //将String字符串以字节流的形式写入到输出流中

output.close(); //关闭输出流

}

/*

* 这里定义的是文件读取的方法

* */

public String read(String filename) throws IOException {

//打开文件输入流

FileInputStream input = mContext.openFileInput(filename);

byte[] temp = new byte[1024];

StringBuilder sb = new StringBuilder("");

int len = 0;

//读取文件内容:

while ((len = input.read(temp)) > 0) {

sb.append(new String(temp, 0, len));

}

//关闭输入流

input.close();

return sb.toString();

}

}

最后是MainActivity.java,我们在这里完成相关操作:

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

private EditText editname;

private EditText editdetail;

private Button btnsave;

private Button btnclean;

private Button btnread;

private Context mContext;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

mContext = getApplicationContext();

bindViews();

}

private void bindViews() {

editdetail = (EditText) findViewById(R.id.editdetail);

editname = (EditText) findViewById(R.id.editname);

btnclean = (Button) findViewById(R.id.btnclean);

btnsave = (Button) findViewById(R.id.btnsave);

btnread = (Button) findViewById(R.id.btnread);

btnclean.setOnClickListener(this);

btnsave.setOnClickListener(this);

btnread.setOnClickListener(this);

}

@Override

public void onClick(View v) {

switch (v.getId()) {

case R.id.btnclean:

editdetail.setText("");

editname.setText("");

break;

case R.id.btnsave:

FileHelper fHelper = new FileHelper(mContext);

String filename = editname.getText().toString();

String filedetail = editdetail.getText().toString();

try {

fHelper.save(filename, filedetail);

Toast.makeText(getApplicationContext(), "数据写入成功", Toast.LENGTH_SHORT).show();

} catch (Exception e) {

e.printStackTrace();

Toast.makeText(getApplicationContext(), "数据写入失败", Toast.LENGTH_SHORT).show();

}

break;

case R.id.btnread:

String detail = "";

FileHelper fHelper2 = new FileHelper(getApplicationContext());

try {

String fname = editname.getText().toString();

detail = fHelper2.read(fname);

} catch (IOException e) {

e.printStackTrace();

}

Toast.makeText(getApplicationContext(), detail, Toast.LENGTH_SHORT).show();

break;

}

}

}

4.读取SD卡上的文件

读取流程图

代码示例

运行效果图

同样打开DDMS的File Explorer,在旧版本的系统上我们可以直接在mmt\sdcard上找到,但是新版本 的就可能需要我们自己找找了,首先我们来到这个路径下:

点开sdcard,但是没东西,我们继续找唠叨后面这个/storage/emulated/legacy下找:

好吧,他又跳到别的地方去了,我们继续找/storage/shell/emilated/0

果然找到了,我们在SD卡里生成的test.txt!导出到电脑看下里面的内容:

嘿嘿,果然读写SD卡成功~接下来我们来看下代码是怎么写的:

代码实现:

main_activity.xml:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

xmlns:tools="http://schemas.android.com/tools"

android:id="@+id/LinearLayout1"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical"

tools:context="com.jay.example.filedemo2.MainActivity">

<TextView

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="清输入文件名" />

<EditText

android:id="@+id/edittitle"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:hint="文件名" />

<TextView

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="清输入文件内容" />

<EditText

android:id="@+id/editdetail"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:hint="文件内容" />

<Button

android:id="@+id/btnsave"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="保存到SD卡" />

<Button

android:id="@+id/btnclean"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="清空" />

<Button

android:id="@+id/btnread"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:text="读取sd卡中的文件" />

</LinearLayout>

接着我们来写一个SD操作类: SDFileHelper.java

/**

* Created by Jay on 2015/9/1 0001.

*/

public class SDFileHelper {

private Context context;

public SDFileHelper() {

}

public SDFileHelper(Context context) {

super();

this.context = context;

}

//往SD卡写入文件的方法

public void savaFileToSD(String filename, String filecontent) throws Exception {

//如果手机已插入sd卡,且app具有读写sd卡的权限

if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {

filename = Environment.getExternalStorageDirectory().getCanonicalPath() + "/" + filename;

//这里就不要用openFileOutput了,那个是往手机内存中写数据的

FileOutputStream output = new FileOutputStream(filename);

output.write(filecontent.getBytes());

//将String字符串以字节流的形式写入到输出流中

output.close();

//关闭输出流

} else Toast.makeText(context, "SD卡不存在或者不可读写", Toast.LENGTH_SHORT).show();

}

//读取SD卡中文件的方法

//定义读取文件的方法:

public String readFromSD(String filename) throws IOException {

StringBuilder sb = new StringBuilder("");

if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {

filename = Environment.getExternalStorageDirectory().getCanonicalPath() + "/" + filename;

//打开文件输入流

FileInputStream input = new FileInputStream(filename);

byte[] temp = new byte[1024];

int len = 0;

//读取文件内容:

while ((len = input.read(temp)) > 0) {

sb.append(new String(temp, 0, len));

}

//关闭输入流

input.close();

}

return sb.toString();

}

}

接着MainActivity.java实现相关逻辑:

public class MainActivity extends AppCompatActivity implements View.OnClickListener{

private EditText editname;

private EditText editdetail;

private Button btnsave;

private Button btnclean;

private Button btnread;

private Context mContext;

@Override

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

mContext = getApplicationContext();

bindViews();

}

private void bindViews() {

editname = (EditText) findViewById(R.id.edittitle);

editdetail = (EditText) findViewById(R.id.editdetail);

btnsave = (Button) findViewById(R.id.btnsave);

btnclean = (Button) findViewById(R.id.btnclean);

btnread = (Button) findViewById(R.id.btnread);

btnsave.setOnClickListener(this);

btnclean.setOnClickListener(this);

btnread.setOnClickListener(this);

}

@Override

public void onClick(View v) {

switch (v.getId()){

case R.id.btnclean:

editdetail.setText("");

editname.setText("");

break;

case R.id.btnsave:

String filename = editname.getText().toString();

String filedetail = editdetail.getText().toString();

SDFileHelper sdHelper = new SDFileHelper(mContext);

try

{

sdHelper.savaFileToSD(filename, filedetail);

Toast.makeText(getApplicationContext(), "数据写入成功", Toast.LENGTH_SHORT).show();

}

catch(Exception e){

e.printStackTrace();

Toast.makeText(getApplicationContext(), "数据写入失败", Toast.LENGTH_SHORT).show();

}

break;

case R.id.btnread:

String detail = "";

SDFileHelper sdHelper2 = new SDFileHelper(mContext);

try

{

String filename2 = editname.getText().toString();

detail = sdHelper2.readFromSD(filename2);

}

catch(IOException e){e.printStackTrace();}

Toast.makeText(getApplicationContext(), detail, Toast.LENGTH_SHORT).show();

break;

}

}

}

最后别忘记在AndroidManifest.xml写上读写SD卡的权限哦!

<!-- 在SDCard中创建与删除文件权限 -->

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

<!-- 往SDCard写入数据权限 -->

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


5.关于原生模拟器SD卡的问题

如果是真机调试的话通常都是可以的,对于原生虚拟机的话就问题多多了,再我们前面使用 Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)可能 一直返回的是false,就是SD卡不存在,这个是主要的问题,现在新版本的SDK都会在 创建AVD的时候会同时申请一块SD卡的存储区域的

对于旧版本的sdk或者其他原因可能需要手动关联下sd卡,设置如下:

①找到创建好的avd的镜像的路径:

点击打开avd界面,点击detail,查看avd镜像的目录下

②来到avd镜像所在的路径下,复制sdcard.img的路径:

比如我的:-sdcard C:\Users\Administrator.android\avd\Jay4.2.avd\sdcard.img

③接着点击

来到以下界面:

最后apply以下,然后Run就可以了!


6.读取raw和assets文件夹下的文件

相信大家对两个文件夹并不陌生,如果我们不想自己的文件被编译成二进制文件的话, 我们可以把文件放到这两个目录下,而两者的区别如下:

  • res/raw:文件会被映射到R.java文件中,访问的时候直接通过资源ID即可访问,而且 他不能有目录结构,就是不能再创建文件夹
  • assets:不会映射到R.java文件中,通过AssetManager来访问,能有目录结构,即, 可以自行创建文件夹

读取文件资源:

res/raw

InputStream is =getResources().openRawResource(R.raw.filename);

assets

AssetManager am = getAssets();

InputStream is = am.open("filename");


代码下载:

  • FileDemo.zip:下载 FileDemo.zip
  • FileDemo2.zip:下载 FileDemo2.zip

本节小结:

好的,关于Android的数据存储与访问的第一节——文件读写就到这里,如果在学习本文中 遇到什么问题,或者觉得有些纰漏的地方,欢迎提出,万分感激,谢谢~

相关推荐

【Docker 新手入门指南】第十章:Dockerfile

Dockerfile是Docker镜像构建的核心配置文件,通过预定义的指令集实现镜像的自动化构建。以下从核心概念、指令详解、最佳实践三方面展开说明,帮助你系统掌握Dockerfile的使用逻...

Windows下最简单的ESP8266_ROTS_ESP-IDF环境搭建与腾讯云SDK编译

前言其实也没啥可说的,只是我感觉ESP-IDF对新手来说很不友好,很容易踩坑,尤其是对业余DIY爱好者搭建环境非常困难,即使有官方文档,或者网上的其他文档,但是还是很容易踩坑,多研究,记住两点就行了,...

python虚拟环境迁移(python虚拟环境conda)

主机A的虚拟环境向主机B迁移。前提条件:主机A和主机B已经安装了virtualenv1.主机A操作如下虚拟环境目录:venv进入虚拟环境:sourcevenv/bin/active(1)记录虚拟环...

Python爬虫进阶教程(二):线程、协程

简介线程线程也叫轻量级进程,它是一个基本的CPU执行单元,也是程序执行过程中的最小单元,由线程ID、程序计数器、寄存器集合和堆栈共同组成。线程的引入减小了程序并发执行时的开销,提高了操作系统的并发性能...

基于网络安全的Docker逃逸(docker)

如何判断当前机器是否为Docker容器环境Metasploit中的checkcontainer模块、(判断是否为虚拟机,checkvm模块)搭配学习教程1.检查根目录下是否存在.dockerenv文...

Python编程语言被纳入浙江高考,小学生都开始学了

今年9月份开始的新学期,浙江省三到九年级信息技术课将同步替换新教材。其中,新初二将新增Python编程课程内容。新高一信息技术编程语言由VB替换为Python,大数据、人工智能、程序设计与算法按照教材...

CentOS 7下安装Python 3.10的完整过程

1.安装相应的编译工具yum-ygroupinstall"Developmenttools"yum-yinstallzlib-develbzip2-develope...

如何在Ubuntu 20.04上部署Odoo 14

Odoo是世界上最受欢迎的多合一商务软件。它提供了一系列业务应用程序,包括CRM,网站,电子商务,计费,会计,制造,仓库,项目管理,库存等等,所有这些都无缝集成在一起。Odoo可以通过几种不同的方式进...

Ubuntu 系统安装 PyTorch 全流程指南

当前环境:Ubuntu22.04,显卡为GeForceRTX3080Ti1、下载显卡驱动驱动网站:https://www.nvidia.com/en-us/drivers/根据自己的显卡型号和...

spark+python环境搭建(python 环境搭建)

最近项目需要用到spark大数据相关技术,周末有空spark环境搭起来...目标spark,python运行环境部署在linux服务器个人通过vscode开发通过远程python解释器执行代码准备...

centos7.9安装最新python-3.11.1(centos安装python环境)

centos7.9安装最新python-3.11.1centos7.9默认安装的是python-2.7.5版本,安全扫描时会有很多漏洞,比如:Python命令注入漏洞(CVE-2015-2010...

Linux系统下,五大步骤安装Python

一、下载Python包网上教程大多是通过官方地址进行下载Python的,但由于国内网络环境问题,会导致下载很慢,所以这里建议通过国内镜像进行下载例如:淘宝镜像http://npm.taobao.or...

centos7上安装python3(centos7安装python3.7.2一键脚本)

centos7上默认安装的是python2,要使用python3则需要自行下载源码编译安装。1.安装依赖yum-ygroupinstall"Developmenttools"...

利用本地数据通过微调方式训练 本地DeepSeek-R1 蒸馏模型

网络上相应的教程基本都基于LLaMA-Factory进行,本文章主要顺着相应的教程一步步实现大模型的微调和训练。训练环境:可自行定义,mac、linux或者window之类的均可以,本文以ma...

【法器篇】天啦噜,库崩了没备份(天啦噜是什么意思?)

背景数据库没有做备份,一天突然由于断电或其他原因导致无法启动了,且设置了innodb_force_recovery=6都无法启动,里面的数据怎么才能恢复出来?本例采用解析建表语句+表空间传输的方式进行...