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

数据存储与访问之文件存储读写 文件读写位置的作用

yuyutoo 2024-10-19 11:09 4 浏览 0 评论

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

相关推荐

史上最全的浏览器兼容性问题和解决方案

微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...

平面设计基础知识_平面设计基础知识实验收获与总结
平面设计基础知识_平面设计基础知识实验收获与总结

CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...

2025-02-21 16:01 yuyutoo

写作排版简单三步就行-工具篇_作文排版模板

和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...

写一个2048的游戏_2048小游戏功能实现

1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...

今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……

  最近的天气暖和得让很多小伙伴们喊“热”!!!  昨天的气温到底升得有多高呢?你家有没有榜上有名?...

CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式

之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...

柠檬科技肖勃飞:大数据风控助力信用社会建设

...

你的自我界限够强大吗?_你的自我界限够强大吗英文

我的结果:A、该设立新的界限...

行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?

行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...

让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华

去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...

2025-02-21 16:00 yuyutoo

今年国家综合性消防救援队伍计划招录消防员15000名

记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...

一起盘点最新 Chrome v133 的5大主流特性 ?

1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...

竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使

style="text-align:center;"data-mce-style="text-align:...

学物理能做什么?_学物理能做什么 卢昌海

作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...

你不知道的关于这只眯眼兔的6个小秘密
你不知道的关于这只眯眼兔的6个小秘密

在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...

2025-02-21 16:00 yuyutoo

取消回复欢迎 发表评论: