- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我正在尝试使用 Retrofit 2 下载/上传文件,但找不到任何有关如何执行此操作的教程示例。我的下载代码是:
@GET("documents/checkout")
public Call<File> checkout(@Query(value = "documentUrl") String documentUrl, @Query(value = "accessToken") String accessToken, @Query(value = "readOnly") boolean readOnly);
和
Call<File> call = RetrofitSingleton.getInstance(serverAddress)
.checkout(document.getContentUrl(), apiToken, readOnly[i]);
call.enqueue(new Callback<File>() {
@Override
public void onResponse(Response<File> response,
Retrofit retrofit) {
String fileName = document.getFileName();
try {
System.out.println(response.body());
long fileLength = response.body().length();
InputStream input = new FileInputStream(response.body());
File path = Environment.getExternalStorageDirectory();
File file = new File(path, fileName);
BufferedOutputStream output = new BufferedOutputStream(
new FileOutputStream(file));
byte data[] = new byte[1024];
long total = 0;
int count;
while ((count = input.read(data)) != -1) {
total += count;
output.write(data, 0, count);
}
output.flush();
output.close();
} catch (IOException e) {
String logTag = "TEMPTAG";
Log.e(logTag, "Error while writing file!");
Log.e(logTag, e.toString());
}
}
@Override
public void onFailure(Throwable t) {
// TODO: Error handling
System.out.println(t.toString());
}
});
我已经尝试过调用和调用,但似乎没有任何效果。
服务器端代码在正确设置了 headers 和 mime 类型后,将文件的字节写入 HttpServletResponse 的输出流中。
我做错了什么?
最后是上传代码:
@Multipart
@POST("documents/checkin")
public Call<String> checkin(@Query(value = "documentId") String documentId, @Query(value = "name") String fileName, @Query(value = "accessToken") String accessToken, @Part("file") RequestBody file);
和
RequestBody requestBody = RequestBody.create(MediaType.parse(document.getMimeType()), file);
Call<String> call = RetrofitSingleton.getInstance(serverAddress).checkin(documentId, document.getFileName(), apiToken, requestBody);
call.enqueue(new Callback<String>() {
@Override
public void onResponse(Response<String> response, Retrofit retrofit) {
System.out.println(response.body());
}
@Override
public void onFailure(Throwable t) {
System.out.println(t.toString());
}
});
谢谢!
编辑:
回答后,下载只会产生损坏的文件(没有@Streaming),上传也不会。当我使用上面的代码时,服务器返回 400 错误。改成之后
RequestBody requestBody = RequestBody.create(MediaType.parse(document.getMimeType()), file);
MultipartBuilder multipartBuilder = new MultipartBuilder();
multipartBuilder.addFormDataPart("file", document.getFileName(), requestBody);
Call<String> call = RetrofitSingleton.getInstance(serverAddress).checkin(documentId, document.getFileName(), apiToken, multipartBuilder.build());
,请求执行,但后端似乎没有收到文件。
后端代码:
@RequestMapping(value = "/documents/checkin", method = RequestMethod.POST)
public void checkInDocument(@RequestParam String documentId,
@RequestParam String name, @RequestParam MultipartFile file,
@RequestParam String accessToken, HttpServletResponse response)
我做错了什么?我能够通过 Apache HttpClient 使用纯 Java 的后端:
MultipartEntityBuilder builder = MultipartEntityBuilder.create();
builder.setMode(HttpMultipartMode.BROWSER_COMPATIBLE);
builder.addBinaryBody("file", new File("E:\\temp\\test.jpg"));
HttpEntity httpEntity = builder.build();
System.out.println("HttpEntity " + EntityUtils.toString(httpEntity.));
HttpPost httpPost = new HttpPost(uri);
httpPost.setEntity(httpEntity);
编辑 v2
对于任何有兴趣的人,现在就上传和下载工作:这些是解决方案:
服务:
@GET("documents/checkout")
public Call<ResponseBody> checkout(@Query(value = "documentUrl") String documentUrl, @Query(value = "accessToken") String accessToken, @Query(value = "readOnly") boolean readOnly);
@Multipart
@POST("documents/checkin")
public Call<String> checkin(@Query(value = "documentId") String documentId, @Query(value = "name") String fileName, @Query(value = "accessToken") String accessToken, @Part("file") RequestBody file);
下载代码:
Call<ResponseBody> call = RetrofitSingleton.getInstance(serverAddress)
.checkout(document.getContentUrl(), apiToken, readOnly[i]);
call.enqueue(new Callback<ResponseBody>() {
@Override
public void onResponse(Response<ResponseBody> response,
Retrofit retrofit) {
String fileName = document.getFileName();
try {
File path = Environment.getExternalStorageDirectory();
File file = new File(path, fileName);
FileOutputStream fileOutputStream = new FileOutputStream(file);
IOUtils.write(response.body().bytes(), fileOutputStream);
} catch (IOException e) {
Log.e(logTag, "Error while writing file!");
Log.e(logTag, e.toString());
}
}
@Override
public void onFailure(Throwable t) {
// TODO: Error handling
System.out.println(t.toString());
}
});
上传代码:
Call<String> call = RetrofitSingleton
.getInstance(serverAddress).checkin(documentId,
document.getFileName(), apiToken,
multipartBuilder.build());
call.enqueue(new Callback<String>() {
@Override
public void onResponse(Response<String> response,
Retrofit retrofit) {
// Handle response here
}
@Override
public void onFailure(Throwable t) {
// TODO: Error handling
System.out.println("Error");
System.out.println(t.toString());
}
});
最佳答案
对于下载,你可以使用 ResponseBody
作为你的返回类型 --
@GET("documents/checkout")
@Streaming
public Call<ResponseBody> checkout(@Query("documentUrl") String documentUrl, @Query("accessToken") String accessToken, @Query("readOnly") boolean readOnly);
您可以在回调中获取 ResponseBody
输入流 --
Call<ResponseBody> call = RetrofitSingleton.getInstance(serverAddress)
.checkout(document.getContentUrl(), apiToken, readOnly[i]);
call.enqueue(new Callback<ResponseBody>() {
@Override
public void onResponse(Response<ResponseBody> response,
Retrofit retrofit) {
String fileName = document.getFileName();
try {
InputStream input = response.body().byteStream();
// rest of your code
如果您的服务器正确处理多部分消息,您的上传乍一看还不错。它在工作吗?如果不是,您能解释一下故障模式吗?您也可以通过不使其多部分来简化。去掉@Multipart
注解,将@Path
转成@Body
--
@POST("documents/checkin")
public Call<String> checkin(@Query("documentId") String documentId, @Query("name") String fileName, @Query("accessToken") String accessToken, @Body RequestBody file);
关于android - 改造 2 文件下载/上传,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32856850/
我正在使用 Retrofit 和 RxJava,但似乎无法做我想做的事。 这是我对 Web 服务的声明: Observable rawRemoteDownload(@Header("Cookie")
我正在开发一个android应用程序,并使用改造发送请求和获取响应。像这样: public interface Service { @POST("/GetInfo") InfoResp
我刚刚从响应中获取代码,它说我的请求参数错误,那么我的 api 调用应该是什么样子? 这是来自文档的硬编码 API 调用 https://api.themoviedb.org/3/discover/m
实际上,当我使用具有 radio 频率的设备时,我的应用程序出现了问题。如果设备超出 radio 范围但尝试发送文件,我会收到 onFailure 消息,我对用户说没有网络,但主要问题是,一旦设备返回
我一直在关注其他答案,但缺少一个我找不到的步骤,这导致调用成功但数据未被正确解析,因为我进行的第一次调用返回了一个对象列表,但只返回了一个对象全部为空 我的模型.java public cla
对于我正在使用的服务器,我们有一个子域和一个目录,它们都绑定(bind)在一起。使用 Retrofit,您需要指定 baseURL,它似乎不允许目录。有什么方法可以实现吗? 例子: https://d
关闭。这个问题需要details or clarity .它目前不接受答案。 想改进这个问题吗? 通过 editing this post 添加细节并澄清问题. 关闭 6 年前。 Improve t
我对 Android 还很陌生.. 我正在尝试使用 Retrofit 从 MySQL 检索数据.. 我在代码中没有发现任何错误,但是当我从设备运行应用程序时,它卡在“正在获取数据的进度对话框”上...
我正在使用 Dagger2 + Retrofit + RxAndroid + OkHttp3 + 新架构组件开发一个 Android 应用程序。 最小 sdk = 16。 问题:在 API 16 上运
我需要在插入硬编码查询后设置查询。 我的 API 地址是: myapiaddress/names?q=Yoni&gender=Man&(here i need to enter dynamic
我正在创建天气应用程序,让用户可以选择按任何城市搜索天气。当用户输入有效的城市名称时,我的应用程序工作正常,但当用户输入无效的字符串 ex: asndfs,bdfbsj 然后我的应用程序终止。 如何处
我有一个 Json 字符串。我无法使用带有两个列表(类别和产品)的 Retrofit onResponse。我怎么打电话,回调?我应该用什么?通常不是列表,对吧? { "Categories": [{
我正在实现一个两级嵌套的 recyclerView 并且两个回收器 View 都使用 retrofit 进行 API 调用。这是发出同步请求的方法: public void loadSectionSt
我正在使用 Qt 编写一个应用程序并且想要一个“Metro 风格”的界面。一切都完成了,除了我不知道如何让小部件出现和消失。例如,在 WPF 中,您可以为 (UIElement.RenderTrans
我在 java 应用程序中使用 Retrofit 来访问 api-rest。 我使用简单的代码: RestAdapter.Builder().setEndpoint(uri).setLogLevel(
我需要将下面报告的数据字符串转换为以时间戳(第一个数字元素)为键的字典。我该怎么做? element=20151201091000|22844.4|22786.2|22801.6|22839.7|10
我正在尝试使用 @QueryMap 发送多个参数(就像我通常做的那样)但是这次使用改造通过 POST。 改造 API @POST("/request.php") void sendRequest(@Q
我正在使用 Retrofit 进行后端通信:如果状态码不是 200 则回调调用失败方法。但是我想在失败方法中获取状态代码以进行进一步的代码调节 @Override pu
我正在使用自定义记录器记录到 Logcat 和文件(当文件启用时)。 这在接收来自测试人员的错误报告时非常有用(因为我在应用程序中还有一个按钮可以发送错误报告并附上日志)。 问题:我正在使用 Retr
很抱歉,如果我的标题含糊不清,但我找不到更好的。 我有一个以这种方式公开服务的休息 Api:/api/{type}/{id} 4 个“类型”,因此返回 4 个类类型。 所有这些类都继承自同一个父类(s
我是一名优秀的程序员,十分优秀!