前后端联调-前端返回包含数组导致后端无法接收~解决

问题:

.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>
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);
    },
  },
  
};
Script
复制代码

 

<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;}
css
复制代码

解决:

  先将数组转换为-分割的字符串,构造一个包含多个对象的数组,构造一个对象:将表单内所有的字段放入对象内。将构造好的留言对象添加到数组中: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);
    }
controlle
复制代码

 

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("注册完成!");
impl
复制代码

 

posted @   学Java的`Bei  阅读(628)  评论(0编辑  收藏  举报
相关博文:
阅读排行:
· winform 绘制太阳,地球,月球 运作规律
· AI与.NET技术实操系列(五):向量存储与相似性搜索在 .NET 中的实现
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
点击右上角即可分享
微信分享提示