Android 应用开发-实现将公共存储空间内的文件复制到应用的私用存储空间中

2024-05-16 01:52

本文主要是介绍Android 应用开发-实现将公共存储空间内的文件复制到应用的私用存储空间中,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、前言

几个月前,我用Android Studio给公司销售部门的同事开发了一款手机app,让同事们用自己的手机就能进行商品的扫码盘点操作,帮他们提高了工作效率,他们用了一段时间,反映还不错。不过前几天,销售部门的同事找到我,说近期公司新增了一些商品,用我的这款软件无法正常扫码这些新商品,希望我能解决问题。

  

这个问题的产生原因是,因为我的能力和资源有限,开发的这款手机app只是一个单机版的辅助工具。在开发时,商品信息是以Sqlit3数据库文件的形式保存在raw文件夹下,随代码一并打包在apk里,软件安装后第一次运行时,会将raw文件夹内的数据库文件导入到应用的私有存储空间内,等于是内嵌在软件中的。这个应用提供了一个手工添加商品信息的功能,但只能一个个手工添加,遇到大量新增的商品信息会很麻烦。这次新增了400多个商品信息,因此同事赶紧找我们解决这个问题。

这篇日志记录了解决问题的过程以备忘。

二、实现过程

要解决这个问题,最简单的方法就是将raw文件夹里的数据库文件更新,再重新打包成apk后,让同事重装一下。但使用这个软件的同事有好些个,有些同事觉得这样的话,以后要是更新商品就要重装一次应用,觉得不妥。希望我还是提供批量添加新商品信息的功能。

鉴于此,我决定为这个应用开发一个导入商品信息的功能。该功能要达到的目标是,在商品更新后,由与我对接的同事按我要求的模板将商品信息做成一个xls文件,然后通过通信软件(如钉钉、微信)将这个xls文件发送给其他的同事。其他同事在手机上下载这个xls文件,然后进入本应用的导入商品模块,选择这个xls文件,将文件内的数据导入的应用内数据库文件中。这样的好处是,以后的商品更新操作都可以由同事们自己完成,不用我插手了。我按这个思路在网上搜索了相关的知识,我希望使用第三方的poi库读取xls文件的数据内容(因为之前开发的将db文件中将数据导出为xls文件就是用的poi库),然后将获取到的数据更新到应用的私有空间内。网上这方面的内容还是比较容易找到,将搜索到的几篇文章中的知识点了解清楚后,依瓢画葫芦就完成了这个功能。通过真机调试,实现了从选定的数据功能。

(Android Studio连真机调试,成功将xls文件中的数据添加到数据库中)

但是在我将应用打包成apk发给同事进行测试时,在所有测试的同事手机上使用该功能导入数据时都出现了闪退现象,这让我很疑惑,之后我将apk文件安装到我的手机上,也出现了闪退,我将手机连接电脑,通过Android Studio检查错误信息,看到了如下的错误提示:

(执行导入商品操作时出现的错误提示)

通过对错误提示的分析,发现是执行语句“InputStream input = contentResolver.openInputStream(fileUri);”时出的错,获取到的对象为空。但是,我之后又通过Android Studio连真机执行“run”命令,用调试模式覆盖掉apk安装文件后,再测试又是正常的。反复进行安装apk和Android Studio连真机调试,都是安装apk后执行导入功能就闪退,调试模式下正常,这就很郁闷了。网上又找不要引起这个问题的原因,导致开发受阻。

但问题总要尽快解决,不然会让这款应用的使用感受严重下降。不得已我要另寻他法。

之后我转换一下思路,开发一个替换数据库文件的功能来解决问题。这个功能的要达到的目标是,在商品信息发生变化后,由与我对接的同事将新的商品信息提供给我,我做好db数据库文件,返回给他,由他通过通信软件(如钉钉、微信)将这个db文件发送给其他的同事。其他同事在手机上下载db文件,然后进入本应用的更新数据模块,选择这个db文件,将文件内容读取出来覆盖应用内原来的数据库文件,达到更新商品信息的目的。读取db文件不会用到poi库,有可能避开上面的问题。有了之前的开发经验,只需要对原来的代码做适当的修改就得到了新的功能,比之前的开发快了不少。在完成了新模块的开发后,再次进行真机调试和打包apk测试,均正常实现了从公共存储空间将db文件的内容复制到应用私有空间中,没有出现闪退问题。

 

(将db文件内容覆盖了原数据库)    (更新后可以查询到新增的商品信息)

这样做,就绕开了上面读取xls文件出错的问题,但我的同事不会做db数据库文件,还需要我插手,相比前面的方案要略逊一筹。但至少同事提出的批量更新商品信息到手机内嵌的数据库中的需求得到了解决。以后如果找到了前述问题的解决办法,再做处理。有时候遇到些问题并不一定是坏事,在解决问题的过程中,也能学到很多的知识

三、部分代码展示

更新数据功能模块xml文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:app="http://schemas.android.com/apk/res-auto"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"android:background="@color/gray_245"android:orientation="vertical"tools:context=".UpdateDbActivity"><androidx.appcompat.widget.Toolbarandroid:id="@+id/toolbar"android:layout_width="match_parent"android:layout_height="wrap_content"android:background="@color/yellow_455"app:navigationIcon="@drawable/ic_back_b" ><TextViewandroid:id="@+id/tv_title"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center"android:textColor="@color/black"android:textSize="24sp"android:textStyle="bold"android:text="@string/GoodsInfoActivity_update" /></androidx.appcompat.widget.Toolbar><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="vertical" ><Buttonandroid:id="@+id/btn_select"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center"android:layout_marginTop="20dp"android:text="@string/select" /><TextViewandroid:layout_width="match_parent"android:layout_height="wrap_content"android:layout_margin="10dp"android:text="选择DB文件,才能更新商品信息数据。"android:textSize="16sp"  /><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:layout_margin="10dp"android:padding="0dp"android:background="@drawable/shape_round_bg_white"android:orientation="vertical"><TextViewandroid:id="@+id/tv_fileInfo"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_marginTop="10dp"android:paddingStart="10dp"android:paddingEnd="10dp"android:layout_marginBottom="20dp"android:text="@string/fileInfo"android:textSize="16sp" /><TextViewandroid:id="@+id/tv_fileName"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_marginTop="10dp"android:paddingStart="10dp"android:paddingEnd="10dp"android:layout_marginBottom="20dp"android:text="@string/fileName"android:textSize="16sp" /><TextViewandroid:id="@+id/tv_fileSize"android:layout_width="match_parent"android:layout_height="wrap_content"android:layout_marginTop="10dp"android:paddingStart="10dp"android:paddingEnd="10dp"android:layout_marginBottom="20dp"android:text="@string/fileSize"android:textSize="16sp" /></LinearLayout><Buttonandroid:id="@+id/btn_updateDB"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center"android:layout_marginBottom="20dp"android:text="@string/updateDB" /><TextViewandroid:id="@+id/tv_tips"android:layout_width="wrap_content"android:layout_height="wrap_content"android:layout_gravity="center"android:layout_marginBottom="20dp"android:text="" /></LinearLayout></LinearLayout>

功能模块java文件

import androidx.appcompat.app.AppCompatActivity;
import androidx.appcompat.widget.Toolbar;import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.provider.DocumentsContract;
import android.provider.OpenableColumns;
import android.util.Log;
import android.view.View;
import android.webkit.MimeTypeMap;
import android.widget.TextView;
import android.widget.Toast;import com.bahamutj.easyinventory.database.GoodsDbHelper;import java.util.Locale;public class UpdateDbActivity extends AppCompatActivity implements View.OnClickListener {private int state = 0;  // 状态,0-初始状态,1-已选择了文件,2-已完成导入,-1-异常状态private final static int DB_CODE = 1;private Uri fileUri;private TextView tv_fileInfo, tv_fileName,tv_fileSize, tv_tips;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_update_db);Toolbar toolbar = findViewById(R.id.toolbar);  // 工具栏toolbar.setTitle("");setSupportActionBar(toolbar);  // 要导入androidx.appcompat.widget.Toolbar 否则报错toolbar.setNavigationOnClickListener(view -> {finish(); // 结束当前页面});tv_fileInfo = findViewById(R.id.tv_fileInfo);tv_fileName = findViewById(R.id.tv_fileName);tv_fileSize = findViewById(R.id.tv_fileSize);tv_tips = findViewById(R.id.tv_tips);findViewById(R.id.btn_select).setOnClickListener(this);  // 选择文件按钮设置监听findViewById(R.id.btn_updateDB).setOnClickListener(this);  // 导入商品按钮设置监听}@Overridepublic void onClick(View v) {int id = v.getId();if ( id == R.id.btn_select) {  // 选择DB文件this.tv_tips.setText("");// 打开Download目录UriUri uri = Uri.parse("content://com.android.externalstorage.documents/document/primary:Download");Intent intent = new Intent(Intent.ACTION_GET_CONTENT);intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, false); // 是否允许多选intent.addCategory(Intent.CATEGORY_OPENABLE);// 设置文件类型String[] mimetypes = {"application/octet-stream", "application/x-sqlite3","application/vnd.sqlite3", "application/x-trash"};intent.putExtra(Intent.EXTRA_MIME_TYPES, mimetypes);  // 设置文件格式intent.setType("*/*");intent.putExtra(DocumentsContract.EXTRA_INITIAL_URI, uri);startActivityForResult(intent, DB_CODE);} else if ( id == R.id.btn_updateDB ) {  // 更新数据if (this.state == 1) {this.tv_tips.setText("更新数据中...");// 更新数据库GoodsDbHelper dbHelper = new GoodsDbHelper(this);int result = dbHelper.copyDatabase(this, this.fileUri);if (result ==1) {this.state = 2;  // 设置状态this.tv_tips.setText("数据已完成更新。");} else {this.tv_tips.setText("数据更新失败。");}} else if (this.state == 0) {Toast.makeText(this, "未选择文件", Toast.LENGTH_SHORT).show();this.tv_tips.setText("");} else if (this.state == 2) {Toast.makeText(this, "数据已更新,不能重复更新", Toast.LENGTH_SHORT).show();this.tv_tips.setText("");}}}@Overrideprotected void onActivityResult(int requestCode, int resultCode, Intent data) {super.onActivityResult(requestCode, resultCode, data);if (resultCode == RESULT_OK && data != null) {this.tv_tips.setText("");Uri uri = data.getData();  // 获取文件uriString uriString = uri.toString();// String type = getContentResolver().getType(uri);  // 获取到文件的类型// 检查文件的扩展名String extension = MimeTypeMap.getFileExtensionFromUrl(uriString);if (!extension.equals("db")) {Toast.makeText(this, "选择的不是db格式数据库文件", Toast.LENGTH_SHORT).show();Toast.makeText(this, "未获取到文件", Toast.LENGTH_SHORT).show();this.tv_fileInfo.setText("文件信息:");this.tv_fileName.setText("文件名称:");this.tv_fileSize.setText("文件大小:");this.state = -1;} else {String fileName;// 获取文件路径、文件名称和文件大小String fileInfo = uri.getPath();fileName = this.getFileName(this, uri);long fileSize = this.getFileSize(this, uri) / 1024;this.tv_fileInfo.setText(String.format(Locale.CHINESE, "%s%s", "文件信息:\n" , fileInfo));this.tv_fileName.setText(String.format(Locale.CHINESE, "%s%s", "文件名称:" , fileName));this.tv_fileSize.setText(String.format(Locale.CHINESE, "%s%d%s", "文件大小:", fileSize, "KB"));this.state = 1;  // 设置状态this.fileUri = uri;}} else {tv_fileInfo.setText("文件信息:");this.tv_fileName.setText("文件名称:");this.tv_fileSize.setText("文件大小:");this.state = 0;}}// 从uri获取文件的名称private String getFileName(Context context, Uri uri) {Cursor returnCursor = context.getContentResolver().query(uri, null, null, null, null);/** Get the column indexes of the data in the Cursor,* move to the first row in the Cursor, get the data,* and close the Cursor.*/int nameIndex = returnCursor.getColumnIndex(OpenableColumns.DISPLAY_NAME);returnCursor.moveToFirst();String name = (returnCursor.getString(nameIndex));returnCursor.close();return name;}// 从uri获取文件的大小private long getFileSize(Context context, Uri uri) {Cursor returnCursor = context.getContentResolver().query(uri, null, null, null, null);/** Get the column indexes of the data in the Cursor,* move to the first row in the Cursor, get the data,* and close the Cursor.*/int sizeIndex = returnCursor.getColumnIndex(OpenableColumns.SIZE);returnCursor.moveToFirst();long size = (returnCursor.getLong(sizeIndex));returnCursor.close();return size;}}
GoodsDbHelper.java部分代码
public class GoodsDbHelper extends SQLiteOpenHelper {private static final String TAG = "GoodsDbHelper";private static final String DB_NAME = "easy_joy.db";public static final String TABLE_NAME = "tb_goods"; // 表的名称public static final String PACKAGE_NAME = "com.bahamutj.easyinventory";// 下面的设置,数据库的位置为(/data/data/com.bahamutj.easyinventory/databases/easy_joy.db)public static final String DB_PATH =  "/data"+ Environment.getDataDirectory().getAbsolutePath() + "/" + PACKAGE_NAME+ "/databases/";public static final String DB_FILE = DB_PATH + DB_NAME;private static final int DB_VERSION = 1;private SQLiteDatabase mDB = null;  // 数据库的实例public static GoodsDbHelper mHelper = null;  // 数据库帮助器的实例//private final Context mContext;public GoodsDbHelper(Context context) {super(context, DB_NAME, null, DB_VERSION);//mContext = context;}@Overridepublic void onCreate(SQLiteDatabase sqLiteDatabase) {// 在这里可以创建数据库表格,如果不需要可留空}@Overridepublic void onUpgrade(SQLiteDatabase sqLiteDatabase, int oldVersion, int newVersion) {// 在这里可以处理数据库升级的逻辑,如果不需要可留空}// 利用单例模式获取数据库帮助器的唯一实例,防止重复获取public static GoodsDbHelper getInstance(Context context) {if (mHelper == null) {mHelper = new GoodsDbHelper(context);}return mHelper;}// 省略......// 复制用户选择的数据库文件到应用的数据库路径中public int copyDatabase(Context mContext, Uri fileUri){int result = 0;ContentResolver contentResolver = mContext.getContentResolver();try {InputStream inputStream = contentResolver.openInputStream(fileUri);OutputStream outputStream = new FileOutputStream(mContext.getDatabasePath(DB_NAME));byte[] buffer = new byte[2048];  // 设置缓存大小int length;while ((length = inputStream.read(buffer)) > 0) {outputStream.write(buffer, 0, length);}outputStream.flush();outputStream.close();inputStream.close();result = 1;} catch (IOException e) {e.printStackTrace(); }return result;}}    

这篇关于Android 应用开发-实现将公共存储空间内的文件复制到应用的私用存储空间中的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/993539

相关文章

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

中文分词jieba库的使用与实景应用(一)

知识星球:https://articles.zsxq.com/id_fxvgc803qmr2.html 目录 一.定义: 精确模式(默认模式): 全模式: 搜索引擎模式: paddle 模式(基于深度学习的分词模式): 二 自定义词典 三.文本解析   调整词出现的频率 四. 关键词提取 A. 基于TF-IDF算法的关键词提取 B. 基于TextRank算法的关键词提取

水位雨量在线监测系统概述及应用介绍

在当今社会,随着科技的飞速发展,各种智能监测系统已成为保障公共安全、促进资源管理和环境保护的重要工具。其中,水位雨量在线监测系统作为自然灾害预警、水资源管理及水利工程运行的关键技术,其重要性不言而喻。 一、水位雨量在线监测系统的基本原理 水位雨量在线监测系统主要由数据采集单元、数据传输网络、数据处理中心及用户终端四大部分构成,形成了一个完整的闭环系统。 数据采集单元:这是系统的“眼睛”,

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

hdu1394(线段树点更新的应用)

题意:求一个序列经过一定的操作得到的序列的最小逆序数 这题会用到逆序数的一个性质,在0到n-1这些数字组成的乱序排列,将第一个数字A移到最后一位,得到的逆序数为res-a+(n-a-1) 知道上面的知识点后,可以用暴力来解 代码如下: #include<iostream>#include<algorithm>#include<cstring>#include<stack>#in

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象