MIME中的Multipart/mixed

总体来说,MIME消息由消息头和消息体两大部分组成。现在我们关注的是MIME邮件,因此在以下的讨论中姑且称“消息”为“邮件”。在上面的例子中,例1的1-6行,例2的1—8行,例3的1-18行,是邮件头;例1的8—9行,例2的10—82行,例3的20—3128行,是邮件体。邮件头与邮件体之间以空行进行分隔,如例1的第7行,例2的第9行,例3的第19行。邮件头中不允许出现空行。有一些邮件不能被邮件客户端软件识别,显示的是原始码,就是因为首行是空行。

邮件头包含了发件人、收件人、主题、时间、MIME版本、邮件内容的类型等重要信息。每条信息称为一个域,由域名后加“: ”和信息内容构成,可以是一行,较长的也可以占用多行。域的首行必须“顶头”写,即左边不能有空白字符(空格和制表符);续行则必须以空白字符打头,且第一个空白字符不是信息本身固有的,解码时要过滤掉。如例2的7-8行,例3的4-5行,13-14行,分别属于一个域。

邮件体包含邮件的内容,它的类型由邮件头的“Content-Type”域指出。常见的简单类型有text/plain(纯文本)和text/html(超文本)。

例2和例3中出现的multipart类型,是MIME邮件的精髓。邮件体被分为多个段,每个段又包含段头和段体两部分,这两部分之间也以空行分隔。常见的multipart类型有三种:multipart/mixed, multipart/related和multipart/alternative。从它们的名称,不难推知这些类型各自的含义和用处。它们之间的层次关系可归纳为下图所示:

+------------------------- multipart/mixed ----------------------------+
|                                                                      |
|  +----------------- multipart/related ------------------+            |
|  |                                                      |            |
|  |  +----- multipart/alternative ------+  +----------+  |  +------+  |
|  |  |                                  |  | 内嵌资源 |  |  | 附件 |  |
|  |  |  +------------+  +------------+  |  +----------+  |  +------+  |
|  |  |  | 纯文本正文 |  | 超文本正文 |  |                |            |
|  |  |  +------------+  +------------+  |  +----------+  |  +------+  |
|  |  |                                  |  | 内嵌资源 |  |  | 附件 |  |
|  |  +----------------------------------+  +----------+  |  +------+  |
|  |                                                      |            |
|  +------------------------------------------------------+            |
|                                                                      |
+----------------------------------------------------------------------+

可以看出,如果在邮件中要添加附件,必须定义multipart/mixed段;如果存在内嵌资源,至少要定义multipart/related段;如果纯文本与超文本共存,至少要定义multipart/alternative段。什么是“至少”?举个例子说,如果只有纯文本与超文本正文,那么在邮件头中将类型扩大化,定义为multipart/related,甚至multipart/mixed,都是允许的。

multipart诸类型的共同特征是,在段头指定“boundary”参数字符串,段体内的每个子段以此串定界。所有的子段都以“--”+boundary行开始,父段则以“--”+boundary+“--”行结束。段与段之间也以空行分隔。在邮件体是multipart类型的情况下,邮件体的开始部分(第一个“--”+boundary行之前)可以有一些附加的文本行,相当于注释,解码时应忽略。段间也可以有一些附加的文本行,不会显示出来,如果有兴趣,不妨验证一下。

结合boundary定界和multipart层次关系图,我们分析一下例2和例3的邮件体层次与段嵌套关系。

在例2中,10-12行是附加文本行,13-82行是multipart/alternative型的段,包含两个子段:13-30行是纯文本正文,32-79行是超文本正文。

在例3中,20-21行是附加文本行,22-3127行是multipart/mixed型的段,包含3个子段:22-171行是multipart/related段,173-1688行与1690-3125行是两个附件。multipart/related段又包含两个子段:27-61行是multipart/alternative段,63-169行是一个内嵌资源(图片)。multipart/alternative段又包含两个子段:31-48行是纯文本正文,40-59行是超文本正文。

例1只有纯文本正文,实际上属于multipart层次关系图中的一个特殊情况。如果非要避简就繁,写成下面的形式,也是完全符合MIME精神的。

Date: Thu, 18 Apr 2002 09:32:45 +0800
From: <bhw98@sina.com>
To: <bhwang@jlonline.com>
Subject: Test
Mime-Version: 1.0
Content-Type: multipart/alternative; boundary="{[(^_^)]}"
  
--{[(^_^)]}
Content-Type: text/plain; charset="iso-8859-1"
Content-Transfer-Encoding: 7bit
 
This is a simple mail.
 
--{[(^_^)]}--
  • 4
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
ResolvableType 可以正确解析 multipart/mixed 类型的数据。multipart/mixed 是一种 MIME 类型,它允许在一个 HTTP 请求或响应传输多个部分数据,每个部分数据都有自己的 MIME 类型和头部信息。在处理 multipart/mixed 数据时,可以使用 ResolvableType 来获取每个部分数据的类型信息和泛型信息。 下面是一个使用 ResolvableType 解析 multipart/mixed 数据的示例代码: ```java public void handleMixedData(ServerWebExchange exchange) { exchange.getRequest().getBody() .flatMapMany(dataBuffer -> { // 将请求体转换为 MultiValueMap MultiValueMap<String, HttpEntity> parts = new LinkedMultiValueMap<>(); HttpHeaders headers = new HttpHeaders(); MediaType contentType = MediaType.parseMediaType(exchange.getRequest().getHeaders().getContentType().toString()); long contentLength = exchange.getRequest().getHeaders().getContentLength(); MultipartResolver resolver = new DefaultMultipartResolver(); MultipartHttpServletRequest multipartRequest = resolver.resolveMultipart(exchange.getRequest()); Map<String, MultipartFile> multipartFiles = multipartRequest.getFileMap(); for (Map.Entry<String, MultipartFile> entry : multipartFiles.entrySet()) { String name = entry.getKey(); MultipartFile file = entry.getValue(); if (file.isEmpty()) { continue; } HttpEntity<byte[]> part = new HttpEntity<>(file.getBytes(), headers); parts.add(name, part); } return Flux.fromIterable(parts.entrySet()); }) .flatMap(entry -> { // 使用 ResolvableType 解析每个部分数据的类型信息和泛型信息 ResolvableType type = ResolvableType.forClassWithGenerics(ResponseEntity.class, MyModel.class); ResponseEntity<MyModel> responseEntity = new ResponseEntity<>((MyModel) entry.getValue(), HttpStatus.OK); return ServerResponse.ok().body(responseEntity); }); } ``` 在上面的示例,我们使用 MultiValueMap 来存储每个部分数据,然后使用 ResolvableType 来解析每个部分数据的类型信息和泛型信息。我们指定了 ResponseEntity<MyModel> 类型,并将每个部分数据转换为 ResponseEntity<MyModel> 类型的对象。最后,我们使用 ServerResponse 返回响应结果。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值