我的编程空间,编程开发者的网络收藏夹
学习永远不晚

android数据存储之文件存储方法

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

android数据存储之文件存储方法

文件存储是 Android 中最基本的一种数据存储方式,它不对存储的内容进行任何的格式化处理,所有数据都是原封不动的保存到文件当中的。

概述

文件存取的核心就是输入流和输出流。

Android文件的操作模式
文件的相关操作方法

文件读写的实现

openFileOutput和openFileInput方法


 
  private void save2File() {
    try {
      //向文件写入内容
      FileOutputStream os = openFileOutput("file.txt", Context.MODE_PRIVATE);
      String text = "写数据到文件";
      os.write(text.getBytes("utf-8"));
      //关闭流
      os.close();
    } catch (FileNotFoundException e) {
      e.printStackTrace();
    } catch (UnsupportedEncodingException e) {
      e.printStackTrace();
    } catch (IOException e) {
      e.printStackTrace();
    }
  }
  
  private void readFile() {
    try {
      FileInputStream ins = openFileInput("file.txt");
      byte[] buffer = new byte[100];
      int byteCount = ins.read(buffer);
      String text = new String(buffer,0,byteCount,"utf-8");
      Toast.makeText(this,text,Toast.LENGTH_SHORT).show();
      ins.close();
    } catch (FileNotFoundException e) {
      e.printStackTrace();
    } catch (IOException e) {
      e.printStackTrace();
    }
  }

文件存储位置

/data/data/<package-name>/files目录下

openFileOutput和openFileInput方法可以获得操作文件的OutputStream以及InputStream对象,而且可以通过流对象处理任何文件的数据,但是这两个方法同SharedPreferences一样,只能在手机内存卡的指定目录建立文件,因此在使用上仍然有一定的局限性。

读取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


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"/>

如何判断虚拟和物理两种SDK

在默认情况下,会将一部分存储空间分给虚拟的SD卡使用(一部分用于安装Android操作系统)


android.os.Enviroment.isExternalStorageRemovalbe()

返回true:SD卡是物理的,反之SD卡是虚拟的。

用于适配不同型号手机,反射获取SD卡路径和状态


package com.turing.base.activity.dataStore.fileStore;
import android.content.Context;
import android.os.Environment;
import android.os.StatFs;
import android.os.storage.StorageManager;
import android.text.TextUtils;
import android.util.Log;
import java.io.File;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class DevMountInfo {
  private final String TAG = DevMountInfo.class.getSimpleName();
  private static final int ERROR = -1;
  // class name
  private final static String CLASS_NAME = "android.os.storage.StorageVolume";
  //remained spare memory size
  private static final int REMAINED_SPARE_IN_MB = 100;
  // method name
  private final static String METHOD_GET_VOLUME_LIST = "getVolumeList";
  private final static String METHOD_GET_VOLUME_STATE = "getVolumeState";
  private final static String METHOD_IS_REMOVABLE = "isRemovable";
  private final static String METHOD_GET_PATH = "getPath";
  private final static String MOUNTED = "mounted";
  private static DevMountInfo INSTANCE;
  private String mSDCardPath = null;
  // internal file path
  private ConcurrentLinkedQueue<String> mInternalPathList = new ConcurrentLinkedQueue<String>();
  // external file path
  private ConcurrentLinkedQueue<String> mExternalPathList = new ConcurrentLinkedQueue<String>();
  private ExecutorService mExecutor = null;
  private DevMountInfo() {
    mExecutor = Executors.newSingleThreadExecutor();
  }
  public static DevMountInfo getInstance() {
    synchronized (DevMountInfo.class) {
      if (null == INSTANCE) {
        INSTANCE = new DevMountInfo();
      }
      return INSTANCE;
    }
  }
  @Override
  protected void finalize() throws Throwable {
    super.finalize();
    synchronized (DevMountInfo.class) {
      mInternalPathList.clear();
      mExternalPathList.clear();
      mExecutor.shutdown();
      INSTANCE = null;
    }
  }
  public void init(final Context context) {
    mExecutor.execute(new Runnable() {
      @Override
      public void run() {
        executeInit(context);
      }
    });
  }
  public boolean isSDCardFull() {
    return REMAINED_SPARE_IN_MB > (getSDCardAvailSpace() * 1024);
  }
  public boolean isSDCardAvaiable() {
    return !mExternalPathList.isEmpty() || !mInternalPathList.isEmpty();
  }
  public String getSDCardPath() {
    return mSDCardPath;
  }
  public long getSDCardTotalSpace() {
    long totalSpace = 0;
    if (!TextUtils.isEmpty(mSDCardPath)) {
      StatFs sf = new StatFs(mSDCardPath);
      long blockSize = sf.getBlockSize();
      long total = sf.getBlockCount();
      totalSpace = total * blockSize / 1024;
    }
    return totalSpace;
  }
  public long getSDCardAvailSpace() {
    long availSpace = 0;
    if (!TextUtils.isEmpty(mSDCardPath)) {
      StatFs sf = new StatFs(mSDCardPath);
      long blockSize = sf.getBlockSize();
      long availCount = sf.getAvailableBlocks();
      availSpace = availCount * blockSize / 1024;
    }
    return availSpace;
  }
  public String getInternalSDCardPath() {
    return mInternalPathList.peek();
  }
  public String getExternalSDCardPath() {
    return mExternalPathList.peek();
  }
  private void executeInit(Context context) {
    StorageManager mStorageManager = (StorageManager) context.getSystemService(Context.STORAGE_SERVICE);
    if (mStorageManager != null) {
      Class<?> mStorageVolume = null;
      Method mGetVolumeListMethod = null;
      Method mGetVolumeStateMethod = null;
      Method mGetPathMethod = null;
      Method mIsRemovableMethod = null;
      Object[] mStorageVolumeList = null;
      try {
        mStorageVolume = Class.forName(CLASS_NAME);
        mGetVolumeListMethod = mStorageManager.getClass().getMethod(METHOD_GET_VOLUME_LIST, new Class[0]);
        mGetVolumeStateMethod = mStorageManager.getClass().getMethod(METHOD_GET_VOLUME_STATE, new Class[]{String.class});
        mIsRemovableMethod = mStorageVolume.getMethod(METHOD_IS_REMOVABLE, new Class[0]);
        mGetPathMethod = mStorageVolume.getMethod(METHOD_GET_PATH, new Class[0]);
        mStorageVolumeList = (Object[]) mGetVolumeListMethod.invoke(mStorageManager, new Object[0]);
        boolean mIsRemovable = false;
        if (mStorageVolumeList != null && mStorageVolumeList.length > 0) {
          int mStorageVolumeCount = mStorageVolumeList.length;
          Log.i(TAG, "init() === > StorageVolume Count = " + mStorageVolumeCount);
          mInternalPathList.clear();
          mExternalPathList.clear();
          for (int i = 0; i < mStorageVolumeCount; ++i) {
            String mStoragePath = (String) mGetPathMethod.invoke(mStorageVolumeList[i], new Object[0]);
            mIsRemovable = ((Boolean) mIsRemovableMethod.invoke(mStorageVolumeList[i], new Object[0])).booleanValue();
            if (!TextUtils.isEmpty(mStoragePath)) {
              String state = (String) mGetVolumeStateMethod.invoke(mStorageManager, new Object[]{mStoragePath});
              if ((state != null) && (state.equals(MOUNTED))) {
                if (mIsRemovable) {
                  Log.i(TAG, "init() === > external storage path = (" + mStoragePath + ")");
                  mExternalPathList.add(mStoragePath);
                } else {
                  Log.i(TAG, "init() === > internal storage path = (" + mStoragePath + ")");
                  mInternalPathList.add(mStoragePath);
                }
              }
            }
          }
        }
      } catch (ClassNotFoundException e) {
        handleInvalid();
        Log.e(TAG, "init() === > Exception:ClassNotFoundException");
      } catch (NoSuchMethodException e) {
        handleInvalid();
        Log.e(TAG, "init() === > Exception:NoSuchMethodException");
      } catch (IllegalArgumentException e) {
        handleInvalid();
        Log.e(TAG, "init() === > Exception:IllegalArgumentException");
      } catch (IllegalAccessException e) {
        handleInvalid();
        Log.e(TAG, "init() === > Exception:IllegalAccessException");
      } catch (InvocationTargetException e) {
        handleInvalid();
        Log.e(TAG, "init() === > Exception:InvocationTargetException");
      }
    } else {
      handleInvalid();
      Log.e(TAG, "init() === > can't get storage manager");
    }
    initSDCardPath();
  }
  private void handleInvalid() {
    mInternalPathList.add(Environment.getExternalStorageDirectory().getPath());
  }
  private void initSDCardPath() {
    if (!mExternalPathList.isEmpty()) {
      mSDCardPath = mExternalPathList.peek();
    } else if (!mInternalPathList.isEmpty()) {
      mSDCardPath = mInternalPathList.peek();
    } else {
      mSDCardPath = Environment.getExternalStorageDirectory().getPath();
    }
    Log.i(TAG, "initSDCardPath() === > SDCARD PATH = (" + mSDCardPath + ")");
  }
  
  public static boolean externalMemoryAvailable() {
    return android.os.Environment.getExternalStorageState().equals(
        android.os.Environment.MEDIA_MOUNTED);
  }
  
  public static long getAvailableInternalMemorySize() {
    File path = Environment.getDataDirectory();
    StatFs stat = new StatFs(path.getPath());
    long blockSize = stat.getBlockSize();
    long availableBlocks = stat.getAvailableBlocks();
    return availableBlocks * blockSize;
  }
  
  public static long getTotalInternalMemorySize() {
    File path = Environment.getDataDirectory();
    StatFs stat = new StatFs(path.getPath());
    long blockSize = stat.getBlockSize();
    long totalBlocks = stat.getBlockCount();
    return totalBlocks * blockSize;
  }
  
  public static long getAvailableInternalSystemMemorySize() {
    File path = Environment.getRootDirectory();
    StatFs stat = new StatFs(path.getPath());
    long blockSize = stat.getBlockSize();
    long availableBlocks = stat.getAvailableBlocks();
    return availableBlocks * blockSize;
  }
  
  public static long getTotalInternalSystemMemorySize() {
    File path = Environment.getRootDirectory();
    StatFs stat = new StatFs(path.getPath());
    long blockSize = stat.getBlockSize();
    long totalBlocks = stat.getBlockCount();
    return totalBlocks * blockSize;
  }
  
  public static long getAvailableExternalMemorySize() {
    if (externalMemoryAvailable()) {
      File path = Environment.getExternalStorageDirectory();
      StatFs stat = new StatFs(path.getPath());
      long blockSize = stat.getBlockSize();
      long availableBlocks = stat.getAvailableBlocks();
      return availableBlocks * blockSize;
    } else {
      return ERROR;
    }
  }
  
  public static long getTotalExternalMemorySize() {
    if (externalMemoryAvailable()) {
      File path = Environment.getExternalStorageDirectory();
      StatFs stat = new StatFs(path.getPath());
      long blockSize = stat.getBlockSize();
      long totalBlocks = stat.getBlockCount();
      return totalBlocks * blockSize;
    } else {
      return ERROR;
    }
  }
  public static long getAvailableMemorySize(String path) {
    if (null == path)
      return 0;
    StatFs stat = new StatFs(path);
    long blockSize = stat.getBlockSize();
    long availableBlocks = stat.getAvailableBlocks();
    return availableBlocks * blockSize;
  }
}

读取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");

SAX引擎读取XML文件

sax引擎读取xml文件的原理:

sax技术在处理xml文件时并不一次性把xml文件装入内存,而是一边读一般解析。

使用sax处理xml需要一个Handler对象,一般会使用org.xml.sax.helpers.DefaultHandler的子类作为Handler对象

因此,这就需要处理如下5个分析点,也可称为分析事件:

开始分析xml文件。该分析点表示sax引擎刚开始处理xml文件,还没有读取xml文件中的内容。该分析点对应于DefaultHandler类中的startDocument()事件方法,可以在该方法中做一下初始化的工作! 开始处理每一个xml元素,也就是遇到<product>,<item>这样的起始标记,sax引擎每次扫描到新的xml元素的起始标记会触发这个分析事件,对应的事件分析方法是startElement,在该方法中可以获取当前元素的名称和元素属性的相关信息 处理完一个xml元素,也就是遇到</product>,</item>这样的结束标记,该分析点对应的事件方法是endElement,在该事件中可以获得当前处理完的元素的全部信息。 处理完xml文件。如果sax引擎将整个xml文件的内容都扫描完了,就到了这个分析点,该分析点对应的事件方法endDocument(),该事件方法可能不是必需的,如果最后有以下收尾工作,如释放一下资源,可以在该方法中完成! 读取字符分析点。这是最重要的分析点。如果没有这个分析点,前4步的处理相当于白跑一遍,虽然读取了xml文件中的所有内容,但并未保存这些内容,而这个分析点所对应的characters事件方法的主要作用就是保存sax引擎读取的xml文件中的内容。更准确地说是保存xml元素的文本,也就是<product>abc</product>中的abc。

Code

res\raw\product.xml


<?xml version="1.0" encoding="utf-8"?>
<products>
  <product>
    <id>10</id>
    <name>电脑</name>
    <price>2067.25</price>
  </product>
  <product>
    <id>20</id>
    <name>微波炉</name>
    <price>520</price>
  </product>
  <product>
    <id>30</id>
    <name>洗衣机</name>
    <price>2400</price>
  </product>
</products>

Product.java


public class Product
{
  private int id;
  private String name;
  private float price;
  public int getId()
  {
    return id;
  }
  public void setId(int id)
  {
    this.id = id;
  }
  public String getName()
  {
    return name;
  }
  public void setName(String name)
  {
    this.name = name;
  }
  public float getPrice()
  {
    return price;
  }
  public void setPrice(float price)
  {
    this.price = price;
  }
}

XML2Product.java(DefaultHandler子类)

DefaultHandler子类 ,核心类,负责处理分析点事件。


import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.helpers.DefaultHandler;
import java.util.ArrayList;
import java.util.List;
public class XML2Product extends DefaultHandler {
  private List<Product> products;
  private Product product;
  private StringBuffer buffer = new StringBuffer();
  public List<Product> getProducts() {
    return products;
  }
  @Override
  public void characters(char[] ch, int start, int length)
      throws SAXException {
    buffer.append(ch, start, length);
    super.characters(ch, start, length);
  }
  @Override
  public void startDocument() throws SAXException {
    // 开始分析xml文件,创建List对象用于保存分析完的Product对象
    products = new ArrayList<Product>();
  }
  @Override
  public void startElement(String uri, String localName, String qName,
               Attributes attributes) throws SAXException {
    if (localName.equals("product")) {
      // 如果分析的是<product>标签,则创建一个Product对象
      product = new Product();
    }
    super.startElement(uri, localName, qName, attributes);
  }
  @Override
  public void endElement(String uri, String localName, String qName)
      throws SAXException {
    if (localName.equals("product")) {
      // 处理完 <product>标签后 将product对象添加到products中
      products.add(product);
    } else if (localName.equals("id")) {
      // 设置id属性的值
      product.setId(Integer.parseInt(buffer.toString().trim()));
      // 将标签内容的缓存区清空
      buffer.setLength(0);
    } else if (localName.equals("name")) {
      product.setName(buffer.toString().trim());
      buffer.setLength(0);
    } else if (localName.equals("price")) {
      product.setPrice(Float.parseFloat(buffer.toString().trim()));
      buffer.setLength(0);
    }
    super.endElement(uri, localName, qName);
  }
}

Xml2JavaObjectAct


import android.app.AlertDialog;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Xml;
import android.view.View;
import com.turing.base.R;
import java.io.InputStream;
import java.util.List;
public class Xml2JavaObjectAct extends AppCompatActivity {
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_xml2_java_object);
  }
  public void onClick_XMLToObject(View view) {
    try {
      // 打开资源文件
      InputStream is = getResources().openRawResource(R.raw.products);
      XML2Product xml2Product = new XML2Product();
      // 开始分析priducts.xml文件
      android.util.Xml.parse(is, Xml.Encoding.UTF_8, xml2Product);
      // 输出转换后的java对象
      List<Product> products = xml2Product.getProducts();
      String msg = "共" + products.size() + "个产品\n";
      for (Product product : products) {
        msg += "id:" + product.getId() + " 产品名:" + product.getName()
            + " 价格:" + product.getPrice() + "\n";
      }
      // 弹出对话框
      new AlertDialog.Builder(this).setTitle("产品信息").setMessage(msg)
          .setPositiveButton("关闭", null).show();
    } catch (Exception e) {
    }
  }
}

效果图

Code

activity_jar_zip.xml


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="fill_parent"
  android:layout_height="fill_parent"
  android:orientation="vertical" >
  <Button
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:onClick="onClick_Jar_Compress"
    android:text="用jar格式压缩文件" />
  <Button
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:onClick="onClick_Jar_Uncompress"
    android:text="解压jar格式文件" />
  <Button
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:onClick="onClick_Zip_Compress"
    android:text="用zip格式压缩文件" />
  <Button
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:onClick="onClick_Zip_Uncompress"
    android:text="解压zip格式文件" />
</LinearLayout>

JarZipAct


import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;
import com.turing.base.R;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.util.jar.JarEntry;
import java.util.jar.JarInputStream;
import java.util.jar.JarOutputStream;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
import java.util.zip.ZipOutputStream;
public class JarZipAct extends AppCompatActivity {
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_jar_zip);
  }
  public void onClick_Jar_Compress(View view) {
    try {
      // 使用FileOutputStream对象指定一个要输出的压缩文件(file.jar)
      FileOutputStream fos = new FileOutputStream(
          android.os.Environment.getExternalStorageDirectory()
              + "/file.jar");
      // 第一步 创建JarOutputStream对象
      JarOutputStream jos = new JarOutputStream(fos);
      // 第二步 创建一个JarEntry对象,并指定待压缩文件在压缩包中的文件名
      JarEntry jarEntry = new JarEntry("strings.xml");
      jos.putNextEntry(jarEntry);
      InputStream is = getResources().getAssets().open("strings.xml");
      byte[] buffer = new byte[8192];
      int count = 0;
      // 第四步 写入数据
      while ((count = is.read(buffer)) >= 0) {
        jos.write(buffer, 0, count);
      }
      // 第五步 关闭当前的JarEntry等对象
      is.close();
      jos.closeEntry();
      jos.close();
      Toast.makeText(this, "成功将strings.xml文件以jar格式压缩.", Toast.LENGTH_LONG)
          .show();
    } catch (Exception e) {
      Toast.makeText(this, e.getMessage(), Toast.LENGTH_LONG).show();
    }
  }
  public void onClick_Jar_Uncompress(View view) {
    try {
      // 定义要解压的文件
      String filename = android.os.Environment
          .getExternalStorageDirectory() + "/file.jar";
      if (!new File(filename).exists()) {
        Toast.makeText(this, "压缩文件不存在.", Toast.LENGTH_LONG).show();
        return;
      }
      // 使用FileInputStream对象指定要解压的对象
      FileInputStream fis = new FileInputStream(filename);
      // 1 创建JarInputStream对象来读取压缩文件(file.jar)
      JarInputStream jis = new JarInputStream(fis);
      // 2 调用getNextJarEntry方法打开压缩包中的第一个文件 ,如果有多个,多次调用该方法
      JarEntry jarEntry = jis.getNextJarEntry();
      // 3 输出已解压的文件
      FileOutputStream fos = new FileOutputStream(
          android.os.Environment.getExternalStorageDirectory()
              + "/" + jarEntry.getName());
      byte[] buffer = new byte[8192];
      int count = 0;
      // 4 输出已解压的字节流
      while ((count = jis.read(buffer)) >= 0) {
        fos.write(buffer, 0, count);
      }
      // 5 关闭
      jis.closeEntry();
      jis.close();
      fos.close();
      Toast.makeText(this, "成功解压jar格式的文件.", Toast.LENGTH_LONG).show();
    } catch (Exception e) {
      Toast.makeText(this, e.getMessage(), Toast.LENGTH_LONG).show();
    }
  }
  public void onClick_Zip_Compress(View view) {
    try {
      // 指定了2个待压缩的w文件,都在assets目录中
      String[] filenames = new String[]
          {"main.xml", "strings.xml"};
      FileOutputStream fos = new FileOutputStream(
          android.os.Environment.getExternalStorageDirectory()
              + "/file.zip");
      ZipOutputStream zos = new ZipOutputStream(fos);
      int i = 1;
      //枚举filenames中的所有待压缩文件
      while (i <= filenames.length) {
        // 从filenames数组中取出当前待压缩的温佳明,作为压缩后的文件名,以保持要说前后文件名称一致
        ZipEntry zipEntry = new ZipEntry(filenames[i - 1]);
        // 打开当前的ZipEntry对象
        zos.putNextEntry(zipEntry);
        InputStream is = getResources().getAssets().open(
            filenames[i - 1]);
        byte[] buffer = new byte[8192];
        int count = 0;
        // 写入数据
        while ((count = is.read(buffer)) >= 0) {
          zos.write(buffer, 0, count);
        }
        zos.flush();
        // 关闭当前的ZipEntry对象
        zos.closeEntry();
        is.close();
        i++;
      }
      zos.finish();
      zos.close();
      Toast.makeText(this, "成功将main.xml、strings.xml文件以zip格式压缩.",
          Toast.LENGTH_LONG).show();
    } catch (Exception e) {
      Toast.makeText(this, e.getMessage(), Toast.LENGTH_LONG).show();
    }
  }
  public void onClick_Zip_Uncompress(View view) {
    try {
      // 指定待解压的文件
      String filename = android.os.Environment
          .getExternalStorageDirectory() + "/file.zip";
      if (!new File(filename).exists()) {
        Toast.makeText(this, "压缩文件不存在.", Toast.LENGTH_LONG).show();
        return;
      }
      FileInputStream fis = new FileInputStream(filename);
      ZipInputStream zis = new ZipInputStream(fis);
      ZipEntry zipEntry = null;
      // 通过不断调用getNextEntry方法来解压file.zip中所有的文件
      while ((zipEntry = zis.getNextEntry()) != null) {
        FileOutputStream fos = new FileOutputStream(
            android.os.Environment.getExternalStorageDirectory()
                + "/" + zipEntry.getName());
        byte[] buffer = new byte[8192];
        int count = 0;
        while ((count = zis.read(buffer)) >= 0) {
          fos.write(buffer, 0, count);
        }
        zis.closeEntry();
        fos.close();
      }
      zis.close();
      Toast.makeText(this, "成功解压jar格式的文件.", Toast.LENGTH_LONG).show();
    } catch (Exception e) {
      Toast.makeText(this, e.getMessage(), Toast.LENGTH_LONG).show();
    }
  }
}

原文链接:http://blog.csdn.net/yangshangwei/article/details/50831269

您可能感兴趣的文章:Android获取SD卡路径及SDCard内存的方法android 获取手机内存及 内存可用空间的方法Android获取内外置存储卡的方法Android编程获取sdcard卡信息的方法Android获取内置sdcard跟外置sdcard路径Android之获取手机内部及sdcard存储空间的方法


免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

android数据存储之文件存储方法

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

android数据存储之文件存储方法

文件存储是 Android 中最基本的一种数据存储方式,它不对存储的内容进行任何的格式化处理,所有数据都是原封不动的保存到文件当中的。概述 文件存取的核心就是输入流和输出流。 Android文件的操作模式 文件的相关操作方法文件读写的实现
2022-06-06

Android中的数据储存之文件存储

当我们在使用各种程序时,其实际上是在和各种数据打交道,当我们聊QQ,刷微博,看新闻,其实都是在和里面的数据交互例如在聊天时发出的消息,以及在登录时输入的账号密码,其实都是瞬时数据,那什么是瞬时数据呢?就是指储存在内存中,有可能因为程序关闭或
2023-06-04

Android基础教程数据存储之文件存储

Android基础教程数据存储之文件存储将数据存储到文件中并读取数据1、新建FilePersistenceTest项目,并修改activity_main.xml中的代码,如下:(只加入了EditText,用于输入文本内容,不管输入什么按下b
2023-05-30

android文件存储和SharedPreferences存储的方法

这篇文章主要讲解了“android文件存储和SharedPreferences存储的方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“android文件存储和SharedPreference
2023-06-30

Android用文件存储数据的方法

本文实例为大家分享了Android用文件存储数据的具体代码,供大家参考,具体内容如下存储数据示例: private void saveFileData() { BufferedWriter writer = null; try
2023-05-30

Android 数据库文件存取至储存卡的方法

废话不多说了,直接给大家贴代码了,具体代码如下 2022-06-06

Android(数据存储:文件存储、SharedPreferences)验证QQ密码

前情提要1 :静态页面-简易QQ登录页面(纯activity_main.xml页面) 【https://blog.csdn.net/weixin_44949135/article/details/104504728】 前情提要2 :动态页面
2022-06-06

Android使用文件进行数据存储的方法

本文实例讲述了Android使用文件进行数据存储的方法。分享给大家供大家参考。具体如下: 很多时候我们开发的软件需要对处理后的数据进行存储,以供再次访问。Android为数据存储提供了如下几种方式: 文件 SharedPreferences
2022-06-06

详解Android数据存储之Android 6.0运行时权限下文件存储的思考

前言: 在我们做App开发的过程中基本上都会用到文件存储,所以文件存储对于我们来说是相当熟悉了,不过自从Android 6.0发布之后,基于运行时权限机制访问外置sdcard是需要动态申请权限,所以以往直接sdcard根目录上直接新建了一个
2022-06-06

实例详解Android文件存储数据方式

总体的来讲,数据存储方式有三种:一个是文件,一个是数据库,另一个则是网络。下面通过本文给大家介绍Android文件存储数据方式。 1.文件存储数据使用了Java中的IO操作来进行文件的保存和读取,只不过Android在Context类中封装
2022-06-06

Android-存储路径与管理应用文件存储方法汇总

一、获取设备存储路径Android手机存储分为两个部分:内部存储和外部存储,内部存储一般是手机自带的存储空间,外部存储指外插SD卡提供的存储空间;随着手机发展,这两个存储的定义又有了一些些变化,新的手机不再有外插SD卡的概念,采取了内置闪存
2022-06-06

【Android】数据存储

Content1. 保存数据到文件1.1. 内部、外部存储区1.2. 在内部存储区 (internal) 存放文件1.3. 在外部存储区 (external) 存放文件2. 使用共享首选项(Shared Preferences)3. 序列化
2022-06-06

【Android-数据存储】外部存储(存储在SD卡上)

android的数据存储主要分为三类: 内部存储:link. 外部存储 sharedpreferences:link. 本文主要介绍外部存储的相关知识。 存储路径:/storage/emulated/0 ——————————————————
2022-06-06

Android文件存储:内部存储,外部存储及其路径

Android系统数据存储分为内部存储和外部存储,内部存储是手机系统自带的存储,一般空间都不大,外部存储一般是SD卡的存储,空间一般都比较大。一般我们会把存储内容放在外部存储空间里。在使用过程需要注意事项: 先判断SD卡是否可用,可用时优先
2022-06-06

Android 文件数据存储实例详解

Android之文件数据存储 一、文件保存数据介绍 Activity提供了openFileOutput()方法可以用于把数据输出到文件中,具体的实现过程与在J2SE环境中保存数据到文件中是一样的。文件可用来存放大量数据,如文本、图书、音频等
2022-06-06

Android数据存储之SQLite使用

SQLite是一款开源的、嵌入式关系型数据库,第一个版本Alpha发布于2000年。SQLite在便携性、易用性、紧凑性、高效性和可靠性方面有着突出的表现。 在Android中创建的SQLite数据库存储在:/data/data/<包名>/
2022-06-06

Android学习之文件存储读取

前言 相信大家都知道知道,在AndroidOS中,提供了五中数据存储方式,分别是:ContentProvider存储、文件存储、SharedPreference存储、SQLite数据库存储、网络存储。那么这一篇,我们介绍文件存储。 1.An
2022-06-06

android存储数据的方法有哪些

在Android中,存储数据的方法主要有以下几种:SharedPreferences:用于存储简单的键值对数据,适合存储一些小型的配置信息。文件存储:可以通过FileOutputStream和FileInputStream类来读写文件,适
android存储数据的方法有哪些
2024-04-09

编程热搜

  • Android:VolumeShaper
    VolumeShaper(支持版本改一下,minsdkversion:26,android8.0(api26)进一步学习对声音的编辑,可以让音频的声音有变化的播放 VolumeShaper.Configuration的三个参数 durati
    Android:VolumeShaper
  • Android崩溃异常捕获方法
    开发中最让人头疼的是应用突然爆炸,然后跳回到桌面。而且我们常常不知道这种状况会何时出现,在应用调试阶段还好,还可以通过调试工具的日志查看错误出现在哪里。但平时使用的时候给你闹崩溃,那你就欲哭无泪了。 那么今天主要讲一下如何去捕捉系统出现的U
    Android崩溃异常捕获方法
  • android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
    系统的设置–>电池–>使用情况中,统计的能耗的使用情况也是以power_profile.xml的value作为基础参数的1、我的手机中power_profile.xml的内容: HTC t328w代码如下:
    android开发教程之获取power_profile.xml文件的方法(android运行时能耗值)
  • Android SQLite数据库基本操作方法
    程序的最主要的功能在于对数据进行操作,通过对数据进行操作来实现某个功能。而数据库就是很重要的一个方面的,Android中内置了小巧轻便,功能却很强的一个数据库–SQLite数据库。那么就来看一下在Android程序中怎么去操作SQLite数
    Android SQLite数据库基本操作方法
  • ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
    工作的时候为了方便直接打开编辑文件,一些常用的软件或者文件我们会放在桌面,但是在ubuntu20.04下直接直接拖拽文件到桌面根本没有效果,在进入桌面后发现软件列表中的软件只能收藏到面板,无法复制到桌面使用,不知道为什么会这样,似乎并不是很
    ubuntu21.04怎么创建桌面快捷图标?ubuntu软件放到桌面的技巧
  • android获取当前手机号示例程序
    代码如下: public String getLocalNumber() { TelephonyManager tManager =
    android获取当前手机号示例程序
  • Android音视频开发(三)TextureView
    简介 TextureView与SurfaceView类似,可用于显示视频或OpenGL场景。 与SurfaceView的区别 SurfaceView不能使用变换和缩放等操作,不能叠加(Overlay)两个SurfaceView。 Textu
    Android音视频开发(三)TextureView
  • android获取屏幕高度和宽度的实现方法
    本文实例讲述了android获取屏幕高度和宽度的实现方法。分享给大家供大家参考。具体分析如下: 我们需要获取Android手机或Pad的屏幕的物理尺寸,以便于界面的设计或是其他功能的实现。下面就介绍讲一讲如何获取屏幕的物理尺寸 下面的代码即
    android获取屏幕高度和宽度的实现方法
  • Android自定义popupwindow实例代码
    先来看看效果图:一、布局
  • Android第一次实验
    一、实验原理 1.1实验目标 编程实现用户名与密码的存储与调用。 1.2实验要求 设计用户登录界面、登录成功界面、用户注册界面,用户注册时,将其用户名、密码保存到SharedPreference中,登录时输入用户名、密码,读取SharedP
    Android第一次实验

目录