vue开发中的常见败招

125 阅读7分钟

vue开发中的常见败招

1 父子组件都是表单,父组件如何获取子组件中的值


<parent-form>
  <child-form :initFormData="xxx"  ref="childForm">
  </child-form>
  <button @click="save">提交</button>
</parent-form>

假设有这样一个组件,父组件有自己的表单项,同时嵌套一个子组件也是表单,当父组件点击提交时候 需要获取子组件的数据,很多人都会直接用ref 直接获取子组件中的data属性,其实这是非常差劲的写法,

使用 ref 直接获取子组件的 data 虽然可行,但在 Vue 的开发模式中并不是最优解。

最优方案建议:使用 v-model 双向绑定

为什么 v-model 更好?

  1. 数据流清晰 :符合 Vue “Props down, Events up” 的核心原则。父组件不需要知道子组件内部变量叫什么(比如是叫 tableFormData 还是 ruleFormParams ),只需要关心绑定的数据。
  2. 解耦 :父组件不需要通过 ref 去探测子组件的内部状态。如果以后子组件内部重构了变量名,只要 v-model 协议不变,父组件代码无需任何改动。
  3. 实时同步 :父组件的 form 中始终保持着最新的表格数据,点击“保存”时直接提交 this.form 即可,不需要在提交瞬间去额外“收集”一次数据。

实现步骤

我已经为你完成了相关的代码重构,具体逻辑如下:

  1. 修改子组件支持 v-model 在 BodyTableForm.vue 和 ParamsTableForm.vue 中:
  • 将接收数据的 prop 从 params 改为 value (Vue 2 v-model 的默认 prop)。
  • 添加 watch 监听:
    • 监听 value :当父组件传入新数据时,同步给内部的 tableFormData 。
    • 监听 tableFormData :当表格内容发生变化(新增、删除、修改)时,通过 $emit('input', newVal) 通知父组件。
// 子组件核心逻辑
props: {
  value: { type: Array, default: () => [] }
},
watch: {
  value: {
    handler(val) { this.tableFormData = val; },
    immediate: true, deep: true
  },
  tableFormData: {
    handler(val) { this.$emit('input', val); },
    deep: true
  }
}

  1. 父组件使用 v-model 绑定 在 ApiRequest/index.vue 中:
  • 移除了冗余的 computed 属性,改为在 data.form 中直接维护 params 和 requestBodyFields 。
  • 模板中直接使用 v-model 绑定。
<!-- 父组件模板 -->
<BodyTableForm v-model="form.requestBodyFields" />
<ParamsTableForm v-model="form.params" />
  1. 完善初始化逻辑 在父组件的 initForm 中增加了 深拷贝 逻辑。这样可以确保子组件对数据的修改不会意外污染到 Vuex 原始数据,只有在点击“确定”提交后才会生效。

总结:其他可选方案对比

image.png

通过现在的 v-model 实现,你的父组件在点击“确定”时, this.form 已经包含了所有子表格的最新数据,逻辑非常自然。

2 “默认值覆盖”问题。

核心问题分析

当你使用 this.form = { ...this.form, ...this.apiInfo } 时,JavaScript 的对象展开运算符(Spread Operator)会遵循“后者覆盖前者”的原则。即便 this.apiInfo 中的字段值是 null 、 undefined 或 '' (空字符串),它依然会无情地覆盖掉 this.form 中预设的系统默认值(如 GET 或 FALSE )。

最优解:在组件初始化时使用“非空合并”策略

虽然可以在 Store 中处理,但我更建议在 组件的 initForm 方法中 解决。

原因 :

  1. UI 表现与数据分离 :Store 应该尽可能真实地反映后端返回的数据(哪怕是空的)。而“默认值”往往是 UI 层的表现需求。

  2. 灵活性 :不同的表单可能对同一个数据对象有不同的默认值需求。

  3. 可维护性 :在组件 data() 中定义的默认值一眼可见,维护起来最直观。 修改后的代码实现 我已经更新了 ApiRequest/index.vue 的逻辑:

    initForm () { if (this.apiInfo) { // 1. 先进行深拷贝,断开引用 const apiData = JSON.parse(JSON.stringify(this.apiInfo))

     // 2. 关键逻辑:过滤掉 apiInfo 中的“空值”
     // 只有当 apiInfo 中的字段真正有内容时,才去覆盖 form 的默认值
     Object.keys(apiData).forEach(key => {
       const val = apiData[key]
       if (val === null || val === undefined || val === '') {
         delete apiData[key] // 如果是空值,从待合并对象中删除
       }
     })
    
     // 3. 执行合并
     this.form = {
       ...this.form,
       ...apiData
     }
    

    } },

为什么不建议在 Store 中处理?

  • Store 污染 :如果在 setOperationCurrentResource 突变中强行填入默认值,那么当你后续需要判断“这个字段是用户填的还是系统默认的”时,就会变得非常困难,因为原始的空状态已经丢失了。
  • 不可预测性 :全局 Store 的改动会影响所有引用该数据的组件。如果你在 Store 里把 null 变成了 GET ,某些只需要展示“暂无数据”的只读页面可能就会错误地显示出 GET 。

总结

通过在组件内采用 “非空合并” 策略:

  • 保留了默认值 :如果后端没传,保留 data() 里的 GET 。
  • 尊重了有效值 :如果后端传了 POST ,它会正确覆盖 GET 。
  • 保护了 Store :Vuex 里的数据依然是纯净的。 这种方案既解决了你的覆盖问题,又保证了代码的健壮性和可维护性。

tab切换子组件获取其他子组件的接口数据

image.png

采用了**“Vuex Action 集中管理 + 组件内按需触发”**的重构方案。这种方式在保证数据即时性的同时,也最大程度地降低了组件间的耦合。

修改内容 1. 在 Store 中封装 fetchAuthList Action

在 store/index.js 中新增了一个异步 Action。

  • 智能判断 :增加了一个 force 参数。默认情况下,如果 Store 中已经有数据,它不会重复请求接口,从而节省性能。

  • 统一入口 :所有需要获取认证列表的地方都统一走这个 Action。 2. 重构认证列表页逻辑 修改了 Authentication/index.vue ,将其原本分散的接口调用改为了 dispatch('fetchAuthList', { force: true }) 。

  • 强制刷新 :在认证主页面,我们显式传入 force: true ,确保用户看到的列表始终是最新的。 3. 在操作弹窗中增加“保底”加载逻辑 在 Operation/Resource/AddDialog/index.vue 中增加了对 visible 的监听:

  • 按需加载 :当“添加资源”的弹窗打开时,组件会检查 Store 中的 authList 是否为空。如果是空的(说明用户直接跳过了认证 Tab),则会自动触发一次数据加载。

重构后的优势

  • 数据安全 :彻底解决了你担心的“直接点击操作 Tab 导致 authList 为空”的问题。
  • 逻辑复用 :获取列表的逻辑从组件中抽离,变得可插拔。如果以后还有第三个、第四个 Tab 需要用到认证数据,只需要在相应位置执行一行 dispatch 即可。
  • 架构专业 :符合 Vuex “Actions 处理异步,Mutations 处理同步”的规范,代码可读性和可测试性显著提升。

子组件上报,父组件编排” 的经典设计模式

父子组件通过props传递数据到子组件,而不是在组件中直接调用vuex, 如果不是直接子组件,是子组件的子组件,可以考虑将中间的组件设计成容器组件, 通过 v-bind="attrs"von="attrs" v-on="listeners"完全透传子组件信息

  1. 子组件:职责纯净化 所有 Tab 子组件( BasicInfo.vue , Authentication/index.vue , 等)不再直接操作 Store 里的 stepsStatus 。
  • 它们现在只负责在关键时刻(如保存成功、数据加载完成)派发一个通用的 @status-change 事件。

  • 代码示例 : this.$emit('status-change', true) 。 2. 父组件:流程编排者 Details.vue 承担起“管家”的角色:

  • 在模板中通过 @status-change="handleStatusChange(DETAIL_TAB.XXX.value, $event)" 显式监听每个 Tab 的状态。

  • 由父组件统一调用 updateStepStatus 来更新全局 Store。

  • 优势 :你现在只需要在 Details.vue 里的 handleStatusChange 方法打一个断点,就能监控到全流程中任何一步的状态切换,调试效率极大提升。 3. Store:数据契约层 store/index.js 只负责接收指令并持久化状态,不再被多个组件直接修改,数据流向变得更加清晰(单向数据流)。

重构收益总结

  • 解耦 :子组件不再需要导入 DETAIL_TAB 常量或映射 updateStepStatus mutation,变得更加轻量和独立。

  • 透明 :父组件的模板里清晰地展示了哪些组件会触发流程变化,逻辑不再“隐身”。

  • 健壮 :避免了子组件因为拼写错误或引用了错误的 Store Key 而导致的流程卡死。