24

我尝试将大图像上传/流式传输到 REST 控制器,该控制器获取文件并将其存储到数据库中。

@Controller
@RequestMapping("/api/member/picture")
public class MemberPictureResourceController {

  @RequestMapping(value = "", method = RequestMethod.POST)
  @ResponseStatus(HttpStatus.NO_CONTENT)
  public void addMemberPictureResource(@RequestBody InputStream image) {
    // Process and Store image in database
  }
}

这是我试图实现的一个无效示例(当然,或者我猜 InputStream 不工作)。我想通过@RequestBody 流式传输/读取图像。

我到处搜索,但找不到如何实现这一目标的好例子。大多数人似乎只问如何通过表单上传图像,但不使用 REST/RestTemplate 来做到这一点。有没有人可以帮助我解决这个问题?

我很感谢任何指向正确方向的提示。

亲切的问候,克里斯

解决方案

下面我尝试在 Dirk 和 Gigadot 的输入之后发布对我有用的解决方案。目前我认为这两种解决方案都值得一看。首先,我尝试在 Dirk 的帮助下发布一个工作示例,然后我将尝试在 Gigadot 的帮助下创建一个。我将 Dirks 的答案标记为正确答案,因为我一直在明确询问如何通过 @RequestBody 上传文件。但我也很想测试 Gigadot 的解决方案,因为它可能更容易使用,也更常见。

在下面的示例中,我将文件存储在 MongoDB GridFS 中。

解决方案 1 - Dirks 推荐后的示例

控制器(在评论中使用 curl 命令进行测试):

/**
*
* @author charms
* curl -v -H "Content-Type:image/jpeg" -X PUT --data-binary @star.jpg http://localhost:8080/api/cardprovider/logo/12345
*/
@Controller
@RequestMapping("/api/cardprovider/logo/{cardprovider_id}")
public class CardproviderLogoResourceController {

  @Resource(name = "cardproviderLogoService")
  private CardproviderLogoService cardproviderLogoService;

  @RequestMapping(value = "", method = RequestMethod.PUT)
  @ResponseStatus(HttpStatus.NO_CONTENT)
  public void addCardproviderLogo(@PathVariable("cardprovider_id") String cardprovider_id,
      HttpEntity<byte[]> requestEntity) {
    byte[] payload = requestEntity.getBody();
    InputStream logo = new ByteArrayInputStream(payload);
    HttpHeaders headers = requestEntity.getHeaders();

    BasicDBObject metadata = new BasicDBObject();
    metadata.put("cardproviderId", cardprovider_id);
    metadata.put("contentType", headers.getContentType().toString());
    metadata.put("dirShortcut", "cardproviderLogo");
    metadata.put("filePath", "/resources/images/cardproviders/logos/"); 

    cardproviderLogoService.create1(logo, metadata);
  }
}

服务(未完成但作为测试工作):

@Service
public class CardproviderLogoService {

  @Autowired
  GridFsOperations gridOperation;

  public Boolean create1(InputStream content, BasicDBObject metadata) {
    Boolean save_state = false;
    try {
      gridOperation.store(content, "demo.jpg", metadata);
      save_state = true;
    } catch (Exception ex) {
      Logger.getLogger(CardproviderLogoService.class.getName())
          .log(Level.SEVERE, "Storage of Logo failed!", ex);
    }
    return save_state;    
  }
}

解决方案 2 - Gigadots 推荐后的示例

这在 Spring 手册中有描述:http: //static.springsource.org/spring/docs/3.2.1.RELEASE/spring-framework-reference/html/mvc.html#mvc-multipart

这很容易,并且默认情况下还包含所有信息。我想我至少会为二进制上传选择这个解决方案。

感谢大家发帖和回答。非常感谢。

4

4 回答 4

13

看起来好像您正在使用 spring,您可以使用 HttpEntity ( http://static.springsource.org/spring/docs/3.1.x/javadoc-api/org/springframework/http/HttpEntity.html )。

使用它,你会得到这样的东西(看看“有效载荷”的东西):

@Controller
public class ImageServerEndpoint extends AbstractEndpoint {

@Autowired private ImageMetadataFactory metaDataFactory;
@Autowired private FileService fileService;

@RequestMapping(value="/product/{spn}/image", method=RequestMethod.PUT) 
public ModelAndView handleImageUpload(
        @PathVariable("spn") String spn,
        HttpEntity<byte[]> requestEntity, 
        HttpServletResponse response) throws IOException {
    byte[] payload = requestEntity.getBody();
    HttpHeaders headers = requestEntity.getHeaders();

    try {
        ProductImageMetadata metaData = metaDataFactory.newSpnInstance(spn, headers);
        fileService.store(metaData, payload);
        response.setStatus(HttpStatus.NO_CONTENT.value());
        return null;
    } catch (IOException ex) {
        return internalServerError(response);
    } catch (IllegalArgumentException ex) {
        return badRequest(response, "Content-Type missing or unknown.");
    }
}

我们在这里使用 PUT 是因为它是一个 RESTfull “将图像放入产品”。'spn' 是产品编号,图像名称由 fileService.store() 创建。当然你也可以发布图片来创建图片资源。

于 2013-01-31T15:33:31.920 回答
6

当您发送 POST 请求时,您可以使用两种类型的编码将表单/参数/文件发送到服务器,application/x-www-form-urlencodedmultipart/form-data. 这是更多阅读

application/x-www-form-urlencoded如果您的 POST 请求中有大量内容,则使用不是一个好主意,因为它通常会使 Web 浏览器崩溃(根据我的经验)。因此, multipart/form-data推荐。

multipart/form-data如果您向调度程序添加多部分解析器,Spring 可以自动处理内容。Spring 用于处理此问题的实现是使用 apache-commons-fileupload 完成的,因此您需要将此库添加到您的项目中。

现在对于如何实际操作的主要答案已经在此处发布了博客http://viralpatel.net/blogs/spring-mvc-multiple-file-upload-example/

我希望这可以帮助您找到解决方案。您可能想了解什么是 REST。听起来你有点困惑。基本上,即使 url 很丑,几乎所有的 http 请求都是 RESTful 的。

于 2013-02-02T02:55:33.313 回答
4

我最近这样做了,请参阅此处的详细信息。

在浏览器上,我们可以使用文件 API 来加载文件,然后使用 Base64 编码对其内容进行编码,最后在发布之前将编码的内容分配给 javascript 对象。

在服务器上,我们有一个 Spring 控制器来处理请求。作为将请求正文转换为 java 对象的 json 解组的一部分,图像字节的 base64 编码值将转换为标准 Java byte[],并作为 LOB 存储在数据库中。

要检索图像,另一个 Spring Controller 方法可以通过直接流式传输字节来提供图像。

我在上面链接的博客文章假设您想将图像用作另一个对象的一部分,但如果您只想直接使用图像,原理应该是相同的。让我知道是否有任何需要澄清的地方。

于 2013-01-31T02:37:57.177 回答
0

如果您想将请求正文作为 InputStream 处理,您可以直接从请求中获取它。

@Controller
@RequestMapping("/api/member/picture")
public class MemberPictureResourceController {

    @RequestMapping(value = "", method = RequestMethod.POST)
    @ResponseStatus(HttpStatus.NO_CONTENT)
    public void addMemberPictureResource(HttpServletRequest request) {
        try {
            InputStream is = request.getInputStream();
            // Process and Store image in database
        } catch (IOException e) {
            // handle exception
        }
    }
}
于 2020-07-08T23:01:35.443 回答