Android 下载文件并将其打开到外部存储

Android 下载文件并将其打开到外部存储,android,download,android-external-storage,Android,Download,Android External Storage,目前,我正在尝试将文件从某个URL下载到设备的外部存储器或microsd,然后在下载完成后打开它。我按照他的回答 以下是我的oncreate中的内容: // execute this when the downloader must be fired final DownloadTask downloadTask = new DownloadTask(MainActivity.this); if (ActivityCompat.checkSelfPermission(MainAc

目前,我正在尝试将文件从某个URL下载到设备的外部存储器或microsd,然后在下载完成后打开它。我按照他的回答

以下是我的oncreate中的内容:

// execute this when the downloader must be fired
    final DownloadTask downloadTask = new DownloadTask(MainActivity.this);
    if (ActivityCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
        ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, permission_external_memory);
        return;
    } else {
        downloadTask.execute("http://www.sample-videos.com/doc/Sample-doc-file-100kb.doc");
    }
这是我的下载任务代码:

private class DownloadTask extends AsyncTask<String, Integer, String> {

    private Context context;
    private PowerManager.WakeLock mWakeLock;

    public DownloadTask(Context context) {
        this.context = context;
    }

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
        // take CPU lock to prevent CPU from going off if the user
        // presses the power button during download
        PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
        mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,
                getClass().getName());
        mWakeLock.acquire();
        mProgressDialog.show();
    }

    @Override
    protected void onProgressUpdate(Integer... progress) {
        super.onProgressUpdate(progress);
        // if we get here, length is known, now set indeterminate to false
        mProgressDialog.setIndeterminate(false);
        mProgressDialog.setMax(100);
        mProgressDialog.setProgress(progress[0]);
    }

    @Override
    protected String doInBackground(String... sUrl) {
        InputStream input = null;
        OutputStream output = null;
        HttpURLConnection connection = null;
        try {
            URL url = new URL(sUrl[0]);
            connection = (HttpURLConnection) url.openConnection();
            connection.connect();

            // expect HTTP 200 OK, so we don't mistakenly save error report
            // instead of the file
            if (connection.getResponseCode() != HttpURLConnection.HTTP_OK) {
                return "Server returned HTTP " + connection.getResponseCode()
                        + " " + connection.getResponseMessage();
            }

            // this will be useful to display download percentage
            // might be -1: server did not report the length
            int fileLength = connection.getContentLength();

            // download the file
            input = connection.getInputStream();
            File mypath = new File(Environment.getExternalStorageDirectory(), "myfiletest.doc");
            output = new FileOutputStream(mypath);

            byte data[] = new byte[4096];
            long total = 0;
            int count;
            while ((count = input.read(data)) != -1) {
                // allow canceling with back button
                if (isCancelled()) {
                    input.close();
                    return null;
                }
                total += count;
                // publishing the progress....
                if (fileLength > 0) // only if total length is known
                    publishProgress((int) (total * 100 / fileLength));
                output.write(data, 0, count);
            }
        } catch (Exception e) {
            return e.toString();
        } finally {
            try {
                if (output != null)
                    output.close();
                if (input != null)
                    input.close();
            } catch (IOException ignored) {
            }

            if (connection != null)
                connection.disconnect();
        }
        return null;
    }

    @Override
    protected void onPostExecute(String result) {
        mWakeLock.release();
        mProgressDialog.dismiss();
        if (result != null)
            Toast.makeText(context, "Download error: " + result, Toast.LENGTH_LONG).show();
        else {
            Toast.makeText(context, "File downloaded", Toast.LENGTH_SHORT).show();
            Intent intent = new Intent();
            intent.setAction(Intent.ACTION_VIEW);
            intent.setDataAndType(Uri.parse(Environment.getExternalStorageDirectory(), "myfiletest.doc"), "text/*");
            context.startActivity(intent);
        }
    }

}
私有类下载任务扩展了异步任务{
私人语境;
private PowerManager.WakeLock mWakeLock;
公共下载任务(上下文){
this.context=上下文;
}
@凌驾
受保护的void onPreExecute(){
super.onPreExecute();
//如果用户
//下载时按下电源按钮
PowerManager pm=(PowerManager)context.getSystemService(context.POWER\u服务);
mWakeLock=pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,
getClass().getName());
mWakeLock.acquire();
mProgressDialog.show();
}
@凌驾
受保护的void onProgressUpdate(整数…进度){
super.onProgressUpdate(进度);
//如果我们到了这里,长度是已知的,现在将undeterminate设置为false
mProgressDialog.setUndeterminate(false);
mProgressDialog.setMax(100);
mProgressDialog.setProgress(progress[0]);
}
@凌驾
受保护的字符串背景(字符串…sUrl){
InputStream输入=null;
OutputStream输出=null;
HttpURLConnection=null;
试一试{
URL=新URL(sUrl[0]);
connection=(HttpURLConnection)url.openConnection();
connection.connect();
//希望HTTP 200正常,这样我们就不会错误地保存错误报告
//而不是文件
if(connection.getResponseCode()!=HttpURLConnection.HTTP\u确定){
return“Server returned HTTP”+connection.getResponseCode()
+“”+连接。getResponseMessage();
}
//这将有助于显示下载百分比
//可能是-1:服务器未报告长度
int fileLength=connection.getContentLength();
//下载该文件
输入=连接。getInputStream();
File mypath=新文件(Environment.getExternalStorageDirectory(),“myfiletest.doc”);
输出=新文件输出流(mypath);
字节数据[]=新字节[4096];
长总计=0;
整数计数;
而((计数=输入。读取(数据))!=-1){
//允许使用“后退”按钮取消
如果(isCancelled()){
input.close();
返回null;
}
总数+=计数;
//发布进度。。。。
if(fileLength>0)//仅当总长度已知时
出版进度((整数)(总计*100/文件长度));
输出.写入(数据,0,计数);
}
}捕获(例外e){
返回e.toString();
}最后{
试一试{
if(输出!=null)
output.close();
如果(输入!=null)
input.close();
}捕获(忽略IOException){
}
if(连接!=null)
连接断开();
}
返回null;
}
@凌驾
受保护的void onPostExecute(字符串结果){
mWakeLock.release();
mProgressDialog.disclose();
如果(结果!=null)
Toast.makeText(上下文,“下载错误:+result,Toast.LENGTH_LONG).show();
否则{
Toast.makeText(上下文,“下载文件”,Toast.LENGTH_SHORT).show();
意图=新意图();
intent.setAction(intent.ACTION\u视图);
setDataAndType(Uri.parse(Environment.getExternalStorageDirectory(),“myfiletest.doc”),“text/*”;
背景。开始触觉(意图);
}
}
}
但下载完成后,我无法打开文件,也无法使用文件管理器找到下载的文件。这是下载完成并尝试打开文件后显示的对话框:


我在这里做错了什么?

我已经试过你的代码了。它工作得很好,而不是这条线:

intent.setDataAndType(Uri.parse(Environment.getExternalStorageDirectory(), "myfiletest.doc"), "text/*");
所以我把它改成:

 intent.setDataAndType(Uri.parse(Environment.getExternalStorageDirectory().getAbsolutePath()+"/myfiletest.doc"), "text/*");
并且明确地从设置中指定所有权限,ans还包括所有必需的权限。下载后,它会询问您要使用哪个应用程序打开下载的文件?选择适当的选项将打开一个文件

因此,您只缺少显式权限。干杯

编辑:
它保存到
/storage/emulated/0/

您能提供您正在使用的url吗?您是否授予了写入权限?是的,先生,我已将INTERNET、访问网络、写入外部存储,甚至读取外部存储
long total=0总共下载了多少字节?对不起,先生,我不太明白。是的,下载后会询问使用哪个应用程序打开下载的文件。我也试着改变这一行。但是,当我从设备的文件管理器或使用USB电缆通过电脑浏览时,似乎找不到下载的文件。我想我已经检查了所需的权限。在我的情况下,在这些更改之后,您的代码工作得非常好。查看编辑。如果仍然找不到文件,我将删除答案。没关系,先生。我真的很感谢你的帮助。这只是因为我对存储的实现还不熟悉。实际上,我也刚刚找到了该文件,但不是直接通过我的设备的内置文件管理器或通过我的计算机找到的,而是通过fx文件管理器找到的