前后端联调-前端返回包含数组导致后端无法接收~解决
问题:
.HttpMessageNotReadableException: JSON parse error: Cannot deserialize value of type `java.lang.String` from Array value (token `JsonToken.START_ARRAY`); nested exception is com.fasterxml.jackson.databind.exc.MismatchedInputException: Cannot deserialize value of type `java.lang.String` from Array value (token `JsonToken.START_ARRAY`)<EOL> at [Source: (org.springframework.util.StreamUtils$NonClosingInputStream); line: 1, column: 258] (through reference chain: com.xinzhi.pojo.Userinfo["address"])]
前端复选框内地址为数组,后端无法接收。
1. 前端
<template>:

<template> <div> <div class="registration-form"> <div class="register-board"> <h2>用户注册</h2> <el-form ref="registrationForm" :model="registrationForm" :rules="registerRules" label-width="100px"> <el-form-item label="用户名" prop="username"> <el-input v-model="registrationForm.username"></el-input> </el-form-item> <el-form-item label="密码" prop="password"> <el-input type="password" v-model="registrationForm.password" ></el-input> </el-form-item> <el-form-item label="姓名" prop="applicant"> <el-input v-model="registrationForm.applicant"></el-input> </el-form-item> <el-form-item label="性别" prop="gender"> <el-radio v-model="registrationForm.gender" label="男">男</el-radio> <el-radio v-model="registrationForm.gender" label="女">女</el-radio> </el-form-item> <el-form-item label="证件类型" prop="documents"> <el-select v-model="registrationForm.documents" placeholder="请选择"> <el-option label="身份证" value="documents"></el-option> <el-option label="士官证" value="officer"></el-option> <el-option label="驾驶证" value="driver"></el-option> <el-option label="护照" value="passport"></el-option> </el-select> </el-form-item> <el-form-item label="身份证号" prop="identification"> <el-input v-model="registrationForm.identification"></el-input> </el-form-item> <el-form-item label="出生日期" prop="date"> <el-date-picker v-model="registrationForm.date" type="date" ></el-date-picker> </el-form-item> <el-form-item label="证件有效开始" prop="start"> <el-date-picker v-model="registrationForm.start" type="date" @change="handleStartDateChange" ></el-date-picker> </el-form-item> <el-form-item label="证件有效结束" prop="end"> <el-date-picker v-model="registrationForm.end" type="date" ></el-date-picker> </el-form-item> <el-form-item label="证件地址" prop="address"> <el-cascader class="el-form-item-address" v-model="registrationForm.address" :options="locationOptions" placeholder="请选择留言归属地" @change="handleLocationChange" > </el-cascader> </el-form-item> <el-form-item label="居住地址" prop="full_address"> <el-input v-model="registrationForm.full_address"></el-input> </el-form-item> <el-form-item label="邮箱" prop="email"> <el-input v-model="registrationForm.email" style="width: 250px;" class="el-input-email"></el-input> <el-button type="primary" @click="sendVerificationCode" :disabled="!isEmailValid">发送验证码</el-button> </el-form-item> <el-form-item label="验证码" prop="code"> <el-input v-model="registrationForm.code"></el-input> </el-form-item> <el-form-item label="手机号" prop="phone"> <el-input v-model="registrationForm.phone"></el-input> </el-form-item> <el-form-item class="el-form-item-button"> <el-button type="primary" @click="submitForm">注册</el-button> </el-form-item> </el-form> </div> </div> </div> </template>
<script>:

<script type="text/ecmascript-6"> import "../../assets/css/register/register.css"; import axios from "axios"; export default { data() { return { // 注册板必需 registrationForm: { username: "", password: "", applicant: "", gender: "", identification: "", documents: "居民身份证", date: "", start: "", end: "", address: [], // 证件地址 full_address: "", // 居住地址 email: "", code:"", phone: "", }, // 注册表单规则遵守 registerRules: { username: [ { required: true, message: "请输入用户名", trigger: "blur" }, ], // required: true 必填,,,trigger: "blur" 指定了触发验证的时机 password: [{ required: true, message: "请输入密码", trigger: "blur" }], applicant: [ { required: true, message: "请输入真实姓名", trigger: "blur" }, ], gender: [{ required: true, message: "请选择性别", trigger: "change" }], identification: [ { required: true, message: "请输入身份证,系统正在优化,暂时仅支持居民身份证", trigger: "change" }, ], documents: [ { required: true, message: "请选择证件类型", trigger: "change" }, ], date: [ { required: true, message: "请输入出生日期", trigger: "change" }, ], start: [ { required: true, message: "请选择证件开始生效时间", trigger: "change", }, ], end: [ { required: true, message: "请选择证件结束生效时间", trigger: "change", }, { validator: this.validateEndDate, trigger: "change" } ], address: [{ required: true, message: "请输入籍贯", trigger: "change" }], // 证件地址 full_address: [ { required: true, message: "请输入现住址", trigger: "blur" }, // 居住地址 ], email: [{ required: true, message: "请输入有效邮箱", trigger: "blur" }], code: [{ required: true, message: "请输入验证码", trigger: "blur" }], phone: [ { required: true, message: "请输入能接收电话的手机号", trigger: "blur", }, ], }, isEmailValid: true, // 留言归属地表单 locationOptions: [ // 广州市 { value: "广州市", label: "广州市", children: [ // 广州市 { value: "越秀区", label: "越秀区", // 区下街道 :洪桥街道、北京街道、六榕街道、流花街道、光塔街道、人民街道、东山街道、农林街道、梅花村街道、黄花岗街道、华乐街道、建设街道、大塘街道、珠光街道、大东街道、白云街道、登峰街道、矿泉街道。 children: [ { value: "洪桥街道", label: "洪桥街道", }, { value: "北京街道", label: "北京街道", }, { value: "六榕街道", label: "六榕街道", }, { value: "流花街道", label: "流花街道", }, { value: "光塔街道", label: "光塔街道", }, ], }, }; }, methods: { // 邮箱验证码 sendVerificationCode() { // 设置按钮为不可点击状态 this.isEmailValid = false; axios.post('http://127.0.0.1:8080/emailCode', { email: this.registrationForm.email }) .then(response => { console.log(response.data); alert('验证码发送成功!'); this.isEmailValid = true; // 邮箱验证通过,允许注册 }) .catch(error => { console.error('验证码发送失败:', error); alert('验证码发送失败!'); this.isEmailValid = false; // 邮箱验证失败,禁止注册 }); }, submitForm() { // 在提交表单之前,将数组转换为以逗号分隔的字符串 this.registrationForm.address = this.registrationForm.address.join("-"); // 构造一个包含多个留言对象的数组 let allMessages = []; // 初始化一个空数组 // 构造一个留言对象 let message = { username: this.registrationForm.username, password: this.registrationForm.password, applicant: this.registrationForm.applicant, gender: this.registrationForm.gender, identification: this.registrationForm.identification, documents: this.registrationForm.documents, date: this.registrationForm.date, start: this.registrationForm.start, end: this.registrationForm.end, address: this.registrationForm.address, full_address: this.registrationForm.full_address, email: this.registrationForm.email, phone: this.registrationForm.phone }; // 将构造好的留言对象添加到数组中 allMessages.push(message); this.$refs.registrationForm.validate((valid) => { // elementUI给的表单规则,为true则继续 if (valid && this.isEmailValid) { // 提交注册信息到后端 axios .post("http://127.0.0.1:8080/enroll", allMessages) .then((res) => { console.log("传入数据:", this.allMessages); console.log(res); if (res.data.success) { alert("注册成功,请返回登录页面进行登录。"); setTimeout(() => { this.$router.push({ path: "/leadLogin" }); }, 1000); // 延时 1 秒后跳转 } else { alert("注册失败," + res.data.message); } }) .catch((err) => { }); } else { console.log("表单验证失败"); return false; } }); }, handleStartDateChange() { // 开始时间改变时,重新验证结束时间 this.$refs.registrationForm.validateField("end"); }, validateEndDate(rule, value, callback) { const startDate = new Date(this.registrationForm.start); const endDate = new Date(value); const yearDifference = endDate.getFullYear() - startDate.getFullYear(); if (![1, 5, 10, 20].includes(yearDifference)) { callback(new Error("证件有效期应为1年、5年、10年、20年")); } else { callback(); } }, handleLocationChange(value) { console.log("选中的留言归属地:", value); }, }, };
<style>:

.registration-form { box-sizing: border-box; width: 100%; height: 100%; padding-top: 10%; background-image: url(); background-repeat: no-repeat; /* background-position: center right; */ background-size: 500px; width: 400px; margin: 50px auto; } .register-board { border-radius: 20px; width: 550px; padding: 50px 100px 15px 50px; box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1); margin-top: -140px; margin-left: -160px; background-color: #ffffff; /* 设置注册板的背景颜色 */ } h2 { margin: 0px auto 40px auto; text-align: center; color: #505458; } .el-form-item-button{ margin: 0 160px; } .el-input-email{margin-right: 10px;}
解决:
先将数组转换为-分割的字符串,构造一个包含多个对象的数组,构造一个对象:将表单内所有的字段放入对象内。将构造好的留言对象添加到数组中:allMessages.push(message);
2. 后端
解决:将后端改为List接收。
Controller层:

/** * 注册接口 * @param userinfo 实体类(注册参数) * @return 状态 */ @CrossOrigin @PostMapping(value = "/enroll") public Result enroll(@RequestBody List<Userinfo> userinfo){ for (Userinfo userinfoList : userinfo) { System.out.println(userinfoList); } return userinfoLoginService.enroll(userinfo); }
ServiceImpl层:

/** * 注册 * * @param userinfo 实体类(注册参数) * @return Result方法(前端接收类型) */ @Override public Result enroll(List<Userinfo> userinfo) { for (Userinfo userinfoList : userinfo) { // 先判断用户名(账户) // 接收类型为布尔值,true判断成功 false判断失败 if (!validateAccount(userinfoList.getUsername())){ System.out.println(userinfoList.getUsername()); return Result.error("用户名格式错误,重新输入。格式为6~20位账号,必须包含数字、字母、下划线中的两种以上。"); } // 判断用户名是否已被注册 // 根据账户名查询数据库内是否已经存在该账户信息 // 进行判断,如果存在,则需让用户重新输入账户名,不存在,进行运行 if (iUserinfoMapper.selectAll(userinfoList.getUsername()) != null) return Result.error("用户名已存在,请重新输入。格式为6~20位账号,必须包含数字、字母、下划线中的两种以上。"); // 判断密码 if (!validatePassword(userinfoList.getPassword())) return Result.error("密码格式错误,重新输入。格式为8~20位字符,至少包含一个数字,大写、小写字母,特殊字符的三种及以上。"); // 使用MD5进行加密密码,并将加密的密码重新赋予实体类中进行对数据库的添加 userinfoList.setPassword(MD5Utils.encrypt(userinfoList.getPassword())); // 判断申请人 if (!validateChineseName(userinfoList.getApplicant())) return Result.error("申请人格式错误,重新输入。格式为中国姓名。"); // 判断申请人性别 if (!validateChineseGender(userinfoList.getGender())){ System.out.println(userinfoList.getGender()); return Result.error("申请人性别不是男或女,重新输入。格式为只能为男或者女。"); } // 判断身份证号码 if (!validateChineseID(userinfoList.getIdentification())){ System.out.println(userinfoList.getIdentification()); return Result.error("身份证号码格式错误,重新输入。格式为中国大陆的身份证号。"); } // 判断证件类型(暂时只支持 身份证 ) if (!validateDocuments(userinfoList.getDocuments())){ return Result.error("证件类型错误,目前仅支持居民身份证哦,请选择身份证。"); } String code = (String) redisTemplate.opsForValue().get("邮箱" + userinfoList.getEmail()); // 判断 用户输入的邮箱验证码 和 发送的验证码是否相同 /*if (!Objects.equals(userinfo.getCode(), code)) return Result.error("验证码输入错误,请重新输入。");*/ // 判断手机号 if (!validatePhoneNumber(userinfoList.getPhone())) return Result.error("手机号格式错误,重新输入。格式为中国大陆的手机号。"); // 将判断完成的数据进行添加 // 接收返回类型,如果为true则注册成功,反之失败 if (iUserinfoMapper.insertEnroll(userinfoList)) { return Result.ok("注册成功"); } else { return Result.error("注册失败"); } } return Result.ok("注册完成!");
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)