先上正确的示例:
主要是设置我们的request的content-type为multipart/form-data
NSDictionary *param = @{@"assignee" :self.userId,
@"projectName" :itemName.text,
@"proceedingName":Name.text,
@"content" :content.text,
@"urgency" :string
};
BaseNetwork *net = [BaseNetwork new];
[net.httpSessionManager.requestSerializer setValue:@"multipart/form-data" forHTTPHeaderField:@"Content-Type"];
[net uploadImageRequestURL:[NSString stringWithFormat:@"%@act/proceeding/start", GW_Domain] imageArray:self.imageArray parameters:param progress:^(float progress) {
其次是,将我们获取的imageArray(文件集合)按照web端(后台)的key约定,组装。【我们Java 后台的文件集合声明的key名称是files】
------至此,ios这边的代码就结束了。
Java后台:
controller层
/**
* 发起申请(app)
*
* @param proceeding
* @return
*/
@ResponseBody
@RequestMapping("/start")
// @RequiresPermissions("act:proceeding:apply")
public R start(Proceeding proceeding) {
try {
// 启动事项审批工作流程
proceedingService.appStartProcess(proceeding, getUser());
return R.ok();
} catch (BDException e) {
LOGGER.error("app端发起事项审批流程异常", e);
return R.error(e.getMessage());
} catch (Exception e) {
LOGGER.error("app端发起事项审批流程异常", e);
return R.error();
}
}
Proceeding 载体bean
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import org.springframework.web.multipart.MultipartFile;
import com.shengshi.common.domain.UploadFileDO;
import com.shengshi.common.exception.FileException;
import com.shengshi.common.utils.Base64Utils; public class Proceeding implements Serializable { private static final long serialVersionUID = 1L; /**
* 主键UUID
*/
private String id; /**
* 事项审批申请人
*/
private Long userId; /**
* 项目名称
*/
private String projectName; /**
* 事项名称
*/
private String proceedingName; /**
* 事项内容
*/
private String content; /**
* 缓急程度 0:常规 1:紧急 2:特级
*/
private String urgency; /**
* 创建时间
*/
private Date createTime; /**
* 事项审批状态 0:录入 1:审核中 2:审核通过 3:审核不通过
*/
private String status; /**
* 事项审批流程实例ID
*/
private String procInstId; /**
* 任务指定处理人
*/
private String assignee; /**
* 任务发起时间
*/
private Date applyTime; /**
* 上传文件列表,之所以没有规范为multipartfile是在get方法做了另外封装【要兼容移动端和app】
*/
private List<Object> files; /**
* 关联文件,多个用逗号分隔
*/
private String delIds; /**
* 上传文件记录列表
*/
private List<UploadFileDO> uploadFileList; public String getId() {
return id;
} public void setId(String id) {
this.id = id;
} public Long getUserId() {
return userId;
} public void setUserId(Long userId) {
this.userId = userId;
} public String getProjectName() {
return projectName;
} public void setProjectName(String projectName) {
this.projectName = projectName;
} public String getProceedingName() {
return proceedingName;
} public void setProceedingName(String proceedingName) {
this.proceedingName = proceedingName;
} public String getContent() {
return content;
} public void setContent(String content) {
this.content = content;
} public String getUrgency() {
return urgency;
} public void setUrgency(String urgency) {
this.urgency = urgency;
} public Date getCreateTime() {
return createTime;
} public void setCreateTime(Date createTime) {
this.createTime = createTime;
} public String getStatus() {
return status;
} public void setStatus(String status) {
this.status = status;
} public String getProcInstId() {
return procInstId;
} public void setProcInstId(String procInstId) {
this.procInstId = procInstId;
} public String getAssignee() {
return assignee;
} public void setAssignee(String assignee) {
this.assignee = assignee;
} public Date getApplyTime() {
return applyTime;
} public void setApplyTime(Date applyTime) {
this.applyTime = applyTime;
} public List<Object> getFiles() {
return files;
} public void setFiles(List<Object> files) {
this.files = files;
} //之所以封装这个方法,是因为我们需要兼容app(直接form表单提交,文件类型)和h5移动端(将文件用base64 encode一下,以字符串的形式传输)的文件上传
public List<MultipartFile> getFileList() throws FileException {
List<MultipartFile> list = new ArrayList<MultipartFile>();
if (files == null) {
return list;
}
for (Object obj : files) {
if (obj instanceof MultipartFile) {
list.add((MultipartFile) obj);
} else {// 移动app的from提交,文件为base64字符串
// 截取文件字符串 获取文件的类型和内容 base64文件的形式
// “data:image/jpeg;base64,kjkjfkj4k5j43kj34j34kj534k5j43j34kj5k34j5534kj5534j4”
// 这里再说明一下,如果将来对接android图片上传也采取base64方式的时候,
// 有可能解析文件不全,因为Java后台默认会把“+”(加号)替换成“ ”空格,接收的时候先把空格全部替换成“+”
String base64Img = obj.toString();
list.add(Base64Utils.base64ConvertFile(base64Img));
}
}
return list;
} public String getDelIds() {
return delIds;
} public void setDelIds(String delIds) {
this.delIds = delIds;
} public List<UploadFileDO> getUploadFileList() {
return uploadFileList;
} public void setUploadFileList(List<UploadFileDO> uploadFileList) {
this.uploadFileList = uploadFileList;
} }
------正确示例到此结束
我们出现的问题就是,ios端不小心将文件参数多封装了一层,如下:
NSDictionary *param = @{@"assignee" :self.userId,
@"projectName" :itemName.text,
@"proceedingName":Name.text,
@"content" :content.text,
@"urgency" :string,
@"files" :data
};
BaseNetwork *net = [BaseNetwork new];
[net.httpSessionManager.requestSerializer setValue:@"multipart/form-data" forHTTPHeaderField:@"Content-Type"];
[net uploadImageRequestURL:[NSString stringWithFormat:@"%@act/proceeding/start", GW_Domain] imageArray:self.imageArray parameters:param progress:^(float progress) {
就是这个多给的参数,【files】. 后台得到的等于是key:files value:formdata。 本来要的是文件的格式,因为疏忽,导致后台一直按照字符串解析,匹配不成文件类型。记录一下。【本文图一已经对imageArray便利处理时加了相应的key,所以无需在form表单再去嵌套】