从Java应用程序下载文件

3

您好,我需要帮助从我的Java应用程序中下载文件。

URL为"http://my.site.com/UICFEWebroot/QueryOneDateAllCur?lang=ita&rate=0&initDay=11&initMonth=10&initYear=2010&refCur=euro&R1=csv"

我尝试使用以下代码,但结果是一个空文件。

URL urlAgg = new URL(address);  

int lf = urlAgg.openConnection().getContentLength();  
FileOutputStream fos = new FileOutputStream("agg" + File.separator + "cambio" + gg + mm + aaaa + ".csv");   
InputStream in = urlAgg.openStream();  
for (int i = 0; i < lf; i++)
  {
   byte[] b = new byte[1];   
   in.read(b);  
   fos.write(b);   
  }

fos.close();  
in.close();
3个回答

3

这对我有用:

package download;

import java.io.*;
import java.net.URL;

/**
 * DownloadDemo
 * User: Michael
 * Date: Oct 11, 2010
 * Time: 10:19:34 AM
 */
public class DownloadDemo
{
    public static void main(String[] args)
    {
        StringBuilder contents = new StringBuilder(4096);
        BufferedReader br = null;

        try
        {
            String downloadSite = ((args.length > 0) ? args[0] : "http://www.google.com");
            String outputFile = ((args.length > 1) ? args[1] : "currencies.csv");
            URL url = new URL(downloadSite);
            InputStream is = url.openConnection().getInputStream();
            br = new BufferedReader(new InputStreamReader(is));
            PrintStream ps = new PrintStream(new FileOutputStream(outputFile));
            String line;
            String newline = System.getProperty("line.separator");
            while ((line = br.readLine()) != null)
            {
                contents.append(line).append(newline);
            }
            ps.println(contents.toString());
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        finally
        {
            try { if (br != null) br.close(); } catch(IOException e) { e.printStackTrace(); }
        }
    }
}

以下是部分结果(太大无法显示全部):

C:\JDKs\jdk1.6.0_13\bin\java -Didea.launcher.port=7533  com.intellij.rt.execution.application.AppMain download.DownloadDemo http://uif.bancaditalia.it/UICFEWebroot/QueryOneDateAllCur?lang=ita&rate=0&initDay=11&initMonth=10&initYear=2010&refCur=euro&R1=csv
Quotazioni in euro riferite al 11/10/2010""Paese,Valuta,Codice ISO,Codice UIC,Quotazione,Convenzione di cambio,Nota""AFGHANISTAN,Afghani,AFN,115,62.8792,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOALBANIA,Lek,ALL,047,138.163,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOALGERIA,Dinaro Algerino,DZD,106,103.035,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOANGOLA,Readjustado Kwanza,AOA,087,128.395,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOANTIGUA E BARBUDA,Dollaro Caraibi Est,XCD,137,3.76272,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOANTILLE OLANDESI,Fiorino Antille Olandesi,ANG,132,2.48061,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOARABIA SAUDITA,Riyal Saudita,SAR,075,5.22619,Foreign currency amount for 1 Euro,CAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATOARGENTINA,Peso Argentina,ARS,216,5.51578,Foreign currency amount for 1 EuroCAMBI INDICATIVI CALCOLATI GIORNALMENTE DA BI SULLA BASE DELLE RILEVAZIONI DI MERCATO

Process finished with exit code 0

好的,但如果我想将文件保存为 CVS 格式,你能帮我吗? - Giovanni
看起来它已经是CSV格式了,除了换行符。我已经添加了它们并将结果写入修改后的代码中的文件。只需清除前导内容,你就可以了。 - duffymo

2
您可以更改“for”子句一段时间。只是为了确保下载内容长度不正确:
 String urlTemp = "the URL";
 File saveFile = new File("File to save path");  
 URL url = new URL(urlTemp);
 URLConnection connection = url.openConnection();
 InputStream is = connection.getInputStream();
 FileOutputStream fos = new FileOutputStream(saveFile);


 byte[] buffer = new byte[1024];
 int read = 0;
 while ((read = is.read(buffer, 0, buffer.length)) >= 0) {
     fos.write(buffer, 0, read);
 }

 fos.flush();
 fos.close();
 is.close();

还需要尝试使用try/catch语句。如果要下载的文件很大,您可能需要在连接对象上设置更长的超时时间:

connection .setConnectTimeout(timeoutonnect);
connection .setReadTimeout(timeoutRead );

希望这段代码能够帮到你!

这个网址不是问题,这段代码会将网址的内容作为流下载。连接超时也很重要,如果文件不大但服务器或连接速度慢的话。 - Fgblanch
在你的代码中,你是逐字节下载文件,这并不是很高效的方式 ;) - Fgblanch
1
内容长度不一定是错误的。这是一个可选的头部信息。当它不存在时,响应通常会使用分块编码进行传输。 - BalusC
没错,如果你在firebug中检查那个URL请求的响应,你会发现content-length是不存在的。 - Fgblanch

2

我总是会使用一个库来跳过这些样板代码。这里有一个使用commons / io的例子:

final String url = "http://www.etc.etc";
final String fileName = "/foo/bar/baz.txt";

InputStream in = null;
OutputStream out = null;
try{
    in = new URL(url).openStream();
    final File f = new File(fileName);
    final File par = f.getParentFile();
    if(!par.exists() && !par.mkdirs()){
        throw new IllegalStateException(
            "Couldn't create folder " + par);
    }
    out = FileUtils.openOutputStream(f);
    IOUtils.copy(in, out);
} catch(final IOException e){
    // handle exception
} finally{
    IOUtils.closeQuietly(in);
    IOUtils.closeQuietly(out);
}

编辑(duffymo): 这是一个可以实际运行的代码形式:

编辑(seanizer):将其移动到pastebin


如果我添加实际需要使您的代码运行的内容(例如导入、类、主方法等),您的版本将有49行Java代码;而我的只有44行。两者风格相同。我认为我的更短,而您的则具有更多依赖性。如果您愿意,我可以发布它。 - duffymo
@duffymo 真的吗?在我的机器上,我的代码(http://pastebin.com/d7wMEfG4)是35行,而你的代码(http://pastebin.com/rjecPgFC)是45行,都包括一些空行。但这不是我的重点:我更喜欢使用描述性的高级方法,让每个人都能立即看到代码的作用,而你使用高效但不太具有沟通性的代码。是的:你的代码可能表现得更好,但我的代码更易读,对于经验不足的开发人员来说更易维护。我认为两种方法都是有效的。 - Sean Patrick Floyd
是的,就像我说的,我可以发布它。我可以编辑你的答案,这样你就可以看到了。里面还有其他东西,而且我倾向于使用不会最小化行数的风格,但两者很相似。我怀疑没有性能差异:网络下载淹没了我们可能进行的任何微观优化。我猜你的库代码更具性能。你提到的可读性很有道理。我只是对你的“更少的代码行”说法感到好奇。不是真的,所以这不是决定因素。 - duffymo
@duffymo的代码行显然是一个复杂的度量标准,因为你可以用许多方式编写Java代码。我的做法是将我们每个人的代码粘贴到Eclipse中,并让格式化程序使用我的标准设置进行操作,而不改变结果(我将你的主方法粘贴到了我的现有类中)。所以,是的,结果受到我的格式化程序设置的影响。如果你有时间的话,我会很感兴趣看看你的版本。 - Sean Patrick Floyd
@duffymo,实际上我在我的答案中从未做出过这种主张。我说我正在跳过样板文件。对我来说,这意味着我正在编写的是种类,而不是数量的代码。 - Sean Patrick Floyd
1
测量的表现 - 两者都是1210-1240毫秒。 - duffymo

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接