当前位置: 首页 > ai >正文

Vue3 + TypeScript,使用provide提供只读的响应式数据的详细分析与解决方法

原始无类型写法(不报错)

typescript

const applySampleTableData = ref<ApplySample[]>([]);
const applySampleListSymbol = Symbol("applySampleList");
provide(applySampleListSymbol, readonly(applySampleTableData));
  • 类型推断:此时 applySampleListSymbol 的类型是 symbol

  • 无约束检查:TypeScript 不会严格校验 provide 的值类型

添加类型后的写法(报错)

typescript

const applySampleListSymbol = Symbol("applySampleList");
const applySampleListSymbol = Symbol("applySampleList") as InjectionKey<Readonly<Ref<ApplySample[]>>>;
provide(applySampleListSymbol, readonly(applySampleTableData)); // 报错
  • 深层类型不匹配readonly() 会将嵌套结构转换为深层只读

  • 类型收缩:实际提供的类型是 Readonly<Ref<**readonly** ApplySample[]>>
    (注意数组元素自动变为只读)


类型对比表格

位置你声明的类型实际提供的类型
外层RefReadonly<Ref<ApplySample[]>>✅ 匹配 Readonly<Ref<...>>
数组元素ApplySample[](可变数组)readonly ApplySample[](只读数组)
元素属性假设 ApplySample 属性可变自动变为 readonly 属性

修正方案(3种可选)

方案一:放宽注入键类型(推荐)

typescript

// 修改注入键定义(允许只读数组)
const applySampleListSymbol = Symbol("applySampleList"
) as InjectionKey<Readonly<Ref<readonly ApplySample[]>>>;
// 或更精确的写法
type ApplySampleListType = Readonly<Ref<Readonly<ApplySample>[]>>;
const applySampleListSymbol = Symbol("applySampleList"
) as InjectionKey<ApplySampleListType>;
方案二:保持数据可变性(不推荐)

typescript

// 移除readonly包装(失去保护)
provide(applySampleListSymbol, applySampleTableData); // 直接传递可变Ref
方案三:完全类型一致(最佳实践)

typescript

// 步骤1:定义只读接口
interface ApplySample {readonly id: number; // 所有属性显式声明为readonlyreadonly sampleName: string;// ...
}// 步骤2:定义注入键
type ApplySampleListType = Readonly<Ref<ApplySample[]>>;
const applySampleListSymbol = Symbol("applySampleList"
) as InjectionKey<ApplySampleListType>;// 步骤3:提供数据
provide(applySampleListSymbol, readonly(applySampleTableData));
方案四:根据方案三摸索出来的方法,将数组设置为只读 readonly  
Readonly<Ref<readonly ApplySample[]>>

typescript

const applySampleListSymbol = Symbol("applySampleList") as InjectionKey<Readonly<Ref<readonly ApplySample[]>>>;

完整修正代码示例

const applySampleTableData = ref<ApplySample[]>([]);
const applySampleListSymbol = Symbol("applySampleList") as InjectionKey<Readonly<Ref<readonly ApplySample[]>>>;
provide(applySampleListSymbol, readonly(applySampleTableData));

完整修正代码示例

typescript

// types.ts
import type { InjectionKey, Ref } from 'vue';// 定义只读接口(核心!)
export interface ApplySample {readonly id: number;readonly sampleName: string;// ...其他字段均声明为readonly
}// 定义注入键类型
export type ApplySampleListType = Readonly<Ref<ApplySample[]>>;
export const applySampleListKey: InjectionKey<ApplySampleListType> = Symbol("applySampleList"
);// 父组件
import { provide, ref, readonly } from 'vue';
import { ApplySample, applySampleListKey } from './types';const applySampleTableData = ref<ApplySample[]>([]); // 注意这里使用接口类型provide(applySampleListKey, readonly(applySampleTableData));// 子组件
const sampleList = inject(applySampleListKey)!;
sampleList.value[0]?.id; // ✅ 可读
sampleList.value.push(); // ❌ TS错误:push不存在于readonly数组

关键修改点说明

  1. 接口属性显式只读
    确保 ApplySample 的每个属性都声明为 readonly,与 readonly() 转换后的类型匹配

  2. 注入键类型精确声明
    使用 Readonly<Ref<ApplySample[]>> 而不是 Readonly<Ref<readonly ApplySample[]>>,因为接口已自带只读属性

  3. 数据源类型一致性
    ref<ApplySample[]> 必须使用已声明只读属性的接口类型


类型安全验证

typescript

// ✅ 允许的操作
sampleList.value.length // 读取数组长度
sampleList.value[0]?.id // 访问属性// ❌ 禁止的操作(TS报错)
sampleList.value = [] // 禁止替换整个Ref
sampleList.value.push({ id: 1 }) // 禁止修改数组结构
sampleList.value[0].id = 123 // 禁止修改属性值(因为接口声明了readonly)

为什么推荐方案三?

方案类型安全防止意外修改IDE提示代码可维护性
方案一⚠️ 部分⚠️
方案二
方案三

方案三通过 接口级只读声明 + 精确类型匹配,实现了:

  1. 开发阶段即捕获非法修改

  2. 明确的类型提示

  3. 可维护的代码结构


总结

你的报错本质是 类型系统的精确校验 在发挥作用。通过:

  1. 接口属性显式声明 readonly

  2. 注入键类型精确匹配

  3. 数据源类型一致性

这三个步骤可以完美解决类型冲突,同时保持代码的类型安全和可维护性。这正是TypeScript在Vue 3项目中的核心价值体现——在编译阶段提前发现问题,而不是等到运行时。

http://www.xdnf.cn/news/1625.html

相关文章:

  • 蓝牙 LE:安全模式和程序说明(蓝牙中的网络安全)
  • 3D模型文件格式之《FBX格式介绍》
  • 密码学的hash函数,哈希碰撞, collision resistance, BTC用到的SHA-256简介
  • elementui日历显示红点及根据日程范围判断是否有红点
  • 实验三 进程间通信实验
  • 504 nginx解决方案
  • arm64适配系列文章-第六章-arm64环境上rabbitmq-management的部署,构建cluster-operator
  • LeetCode238_除自身以外数组的乘积
  • 2025.5.4机器学习笔记:PINN文献阅读
  • React状态提升深度解析:原理、实战与最佳实践
  • 声音分离人声和配乐-从头设计数字生命第4课——仙盟创梦IDE
  • 树莓派安装GStreamer ,opencv支持, 并在虚拟环境中使用的安装方法
  • 从数据到智慧:解密机器学习的自主学习密码
  • springboot基于hadoop的酷狗音乐爬虫大数据分析可视化系统(源码+lw+部署文档+讲解),源码可白嫖!
  • 【Python】Python在Linux上安装等操作流程以及注意事项| 基础知识
  • PTA -L1-001 Hello World
  • 项目班——0419——chrono时间库
  • VIC-3D非接触全场应变测量系统用于小尺寸测量之电子元器件篇—研索仪器DIC数字图像相关技术
  • 前端面经-JS篇(四)--回调地狱、promise异步编程、Proxy 与 Reflect 、模块化
  • JMeter 安装及使用 [软件测试工具]
  • 【数据分析实战】使用 Matplotlib 绘制玫瑰图
  • 什么是机器视觉3D碰撞检测?机器视觉3D碰撞检测是机器视觉3D智能系统中安全运行的核心技术之一
  • 使用 Docker 安装 SQL Server 2022 并解决 Navicat 连接问题
  • Linux漏洞管理:自动化扫描与补丁更新策略
  • 【软件设计师】模拟题一
  • 修改el-select背景颜色
  • wait_event 类接口详解
  • 题目:这不是字符串题
  • 数据库day-07
  • 晶振不集成到芯片内部的原因分析