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

Vue 3 源码阅读笔记:ref.ts

文章目录一、文件概览二、核心数据结构1. Ref 接口定义三、核心函数实现1. isRef - 类型守卫2. r[ReactiveFlags.IS_REF]详解一、 r[ReactiveFlags.IS_REF] 是什么意思二、这个标记是怎么来的三、为什么需要这个标记四、完整的标记系统五、为什么用不 Symbol 而用字符串六、手写简化版来理解七、总结2. ref - 主入口函数源码解析2.1 [T] extends [Ref] 详解2.2 条件分支一IfAnyT, RefT, T详解2.3 条件分支二RefUnwrapRefT, UnwrapRefT | T详解3. shallowRef - 浅层响应式 ref 的实现一、文件概览元数据说明文件路径packages/reactivity/src/ref.ts核心功能实现 Vue 3 的 ref 相关 API依赖模块vue/shared,./dep,./reactive导出 APIref,shallowRef,isRef,unref,toRef,toRefs,customRef,triggerRef二、核心数据结构1. Ref 接口定义export interface RefT any, S T { get value(): T // 读取时返回类型 T set value(_: S) // 写入时接受类型 S默认等于 T [RefSymbol]: true // 唯一类型标记用于类型识别 }我的理解使用两个泛型参数T和S是为了支持只读场景S never[RefSymbol]: true是一个类型层面的标记编译后消失为什么这样设计这样可以精确控制 ref 的读写类型比如计算属性可以是只读的三、核心函数实现1. isRef - 类型守卫export function isRef(r: any): r is Ref { return r ? r[ReactiveFlags.IS_REF] true : false }源码解析类型谓词r is Ref告诉 TypeScript如果返回 true则 r 是 Ref 类型内部标记ReactiveFlags.IS_REF __v_isRef每个 ref 实例上都有这个属性为什么不用 instanceof因为对象可能被 Proxy 代理原型链会丢失笔记// 使用示例 const count ref(0) console.log(isRef(count)) // true console.log(isRef(100)) // false2.r[ReactiveFlags.IS_REF]详解一、r[ReactiveFlags.IS_REF]是什么意思1. 基本概念// packages/reactivity/src/constants.ts export enum ReactiveFlags { SKIP __v_skip, IS_REACTIVE __v_isReactive, IS_READONLY __v_isReadonly, IS_SHALLOW __v_isShallow, RAW __v_raw, IS_REF __v_isRef, }r[ReactiveFlags.IS_REF] true // 等价于 r[__v_isRef] true这行代码的意思是检查对象r上是否有__v_isRef**这个属性并且它的值是否为 **true。2. 为什么用__v_isRef这种奇怪的属性名__v_是 Vue 内部属性的命名约定v 代表 Vue这样做是为了避免和用户自定义的属性名冲突用户几乎不可能恰好定义一个叫__v_isRef的属性二、这个标记是怎么来的在RefImpl类中设置的// packages/reactivity/src/ref.ts class RefImplT { private _value: T public dep?: Dep undefined // 重点在这里这个标记是在创建 ref 时自动添加的 public readonly [ReactiveFlags.IS_REF] true // -- 就是这里 constructor(value: T) { this._value value } get value() { trackRefValue(this) return this._value } set value(newVal) { if (hasChanged(newVal, this._value)) { this._value newVal triggerRefValue(this) } } }每个 ref 对象在创建时都会被自动加上 __v_isRef: true 这个属性。三、为什么需要这个标记1. 快速识别 ref 对象// 假设没有这个标记怎么判断是不是 ref function isRef(r: any) { // 方法1检查构造函数但 Proxy 代理后不行 // 方法2检查有没有 value 属性但普通对象也可能有 value // 方法3检查内部属性最可靠 return r r.__v_isRef true }2. 运行时快速判断const count ref(0) const obj { value: 100 } // 普通对象碰巧也有 value console.log(isRef(count)) // true因为有 __v_isRef console.log(isRef(obj)) // false没有 __v_isRef console.log(isRef(null)) // false3. 在响应式系统中做特殊处理// 在 reactive 中遇到 ref 时会自动解包 function reactive(target) { return new Proxy(target, { get(target, key, receiver) { const res Reflect.get(target, key, receiver) // 重点如果获取到的值是一个 ref自动返回 .value if (isRef(res)) { return res.value // 自动解包 } return res } }) } // 使用时的效果 const count ref(0) const state reactive({ count: count // 传入 ref }) console.log(state.count) // 0直接拿到值不需要 .value // 这全靠 __v_isRef 标记来判断四、完整的标记系统Vue 用了多个类似的标记export enum ReactiveFlags { SKIP __v_skip, // 跳过响应式转换 IS_REACTIVE __v_isReactive, // 是否是 reactive 对象 IS_READONLY __v_isReadonly, // 是否是 readonly 对象 IS_SHALLOW __v_isShallow, // 是否是 shallow 对象 RAW __v_raw, // 获取原始对象 IS_REF __v_isRef, // 是否是 ref 对象 }其他标记的使用// reactive 对象也有自己的标记 class ReactiveProxy { public readonly [ReactiveFlags.IS_REACTIVE] true public readonly [ReactiveFlags.RAW] this.target // ... } // 使用 const state reactive({ count: 0 }) console.log(state[ReactiveFlags.IS_REACTIVE]) // true console.log(isReactive(state)) // true内部就是检查这个标记五、为什么用不 Symbol 而用字符串你可能会想为什么不用 Symbol 而用字符串// 理论上可以用 Symbol export const IS_REF Symbol(vue.isRef) // 但 Vue 选择了字符串为什么字符串的好处可序列化字符串可以在 JSON 中传输调试友好__v_isRef在控制台直接可见跨框架边界如果 ref 传到其他框架字符串属性仍然存在简单可靠兼容性更好// 控制台直接查看 const count ref(0) console.log(count) // 可以直接看到 __v_isRef: true // 如果是 Symbol控制台显示 [Symbol()]: true可读性差六、手写简化版来理解// 1. 定义标记常量 const IS_REF __v_isRef // 2. 实现 ref class MyRefT { private _value: T // 添加标记 readonly [IS_REF] true constructor(value: T) { this._value value } get value() { return this._value } set value(newVal) { this._value newVal } } // 3. 实现 isRef function isRef(r: any): boolean { return r r[IS_REF] true } // 4. 使用 const myRef new MyRef(100) console.log(isRef(myRef)) // true const normalObj { value: 100 } console.log(isRef(normalObj)) // false七、总结问题答案r[ReactiveFlags.IS_REF]是什么访问 ref 对象上的内部标记__v_isRef为什么要判断这个属性快速、可靠地判断一个对象是不是 ref为这个属性哪里来的RefImpl类在创建实例时自动添加的为什么不用instanceofProxy 代理后会丢失原型链而且跨 iframe 不工作为什么用字符串可序列化、调试友好、简单可靠通俗理解这就像给每个 ref 对象贴了一个我是 ref的防伪标签。isRef函数就是检查这个标签存不存在、是不是真。这种方式比instanceof更可靠因为即使对象被 Proxy 代理、被传到不同的 iframe、甚至被序列化后再解析这个字符串属性依然存在。2. ref - 主入口函数export function refT( value: T, ): [T] extends [Ref] ? IfAnyT, RefT, T : RefUnwrapRefT, UnwrapRefT | T export function refT any(): RefT | undefined export function ref(value?: unknown) { return createRef(value, false) // 第二个参数 false 表示非浅层 ref }源码解析重载作用第一个处理传入值的情况防止嵌套 ref第二个处理不传参的情况创建值为 undefined 的 ref实现体统一调用 createRef重点理解[T] extends [Ref]检查 T 是否已经是 Ref 类型避免创建RefRefIfAnyT, RefT, T如果 T 是 any返回 Ref否则返回 T 本身UnwrapRefT递归解包嵌套的 ref2.1 [T] extends [Ref] 详解[T] extends [Ref] 是什么意思它是TS的条件类型意思是判断类型 T 是否是 Ref 类型。1. 基本语法[T] extends [Ref] ? A : B2. 为什么用[T]而不是T目的避免联合类型的分发做严格的一次性判断// 如果直接写 T extends Ref type Test1T T extends Ref ? true : false // 当 T string | Ref 时会分别判断 // string extends Ref? false // Ref extends Ref? true // 结果boolean联合类型 // 用 [T] extends [Ref] type Test2T [T] extends [Ref] ? true : false // 当 T string | Ref 时作为整体判断 // [string | Ref] 是不是 [Ref] 的子类型否 // 结果false2.2 条件分支一IfAnyT, RefT, T详解我的追问为什么[T] extends [Ref]为true不就说明T是Ref类型了吗为什么还需要判断T是否是any并给any类型包装Ref这样不会造成重复包装吗关键在于[T] extends [Ref]为 true 时T可能是Ref也可能是any为什么会有 any 的情况// 场景传入 any const value: any 123 const refAny ref(value) // T any // 此时判断 [any] extends [Ref] 是 true 还是 falseTypeScript 中 any 的特殊性// any 可以赋值给任何类型 let x: any 123 let y: string x // ✅ 允许any 可以赋值给 string // 所以 type Test [any] extends [Ref] ? true : false // true // 因为 any 可以当作 Ref 来用尽管实际不是用代码验证// 写个简单的类型测试 type IsRefT [T] extends [Ref] ? true : false // 测试1真正的 ref type Test1 IsRefRefnumber // true ✅ // 测试2any type Test2 IsRefany // true ✅any 万能匹配 // 测试3普通类型 type Test3 IsRefnumber // false type Test4 IsRefstring // false问题就在这里any会让[T] extends [Ref]也返回 trueIfAny 的作用// IfAny 的定义来自 vue/shared type IfAnyT, Y, N 0 extends (1 T) ? Y : N // 这是一个判断 T 是否是 any 的类型工具如果不判断 any// 假设没有 IfAny直接返回 T function badRefT(value: T): [T] extends [Ref] ? T : RefT { // ... } const value: any 123 const result badRef(value) // 返回类型any // 类型丢失我们不知道这是个 ref 了为什么不能直接返回 Ref// 如果直接返回 RefT function badRefT(value: T): [T] extends [Ref] ? RefT : RefT { // ... } // 问题1传入真正的 ref 时 const count ref(0) const result badRef(count) // RefRefnumber ❌ 嵌套了 // 问题2传入 any 时 const val: any 123 const result2 badRef(val) // Refany ✅ 这个是对的 // 问题3传入普通值时 const num 123 const result3 badRef(num) // Refnumber ✅ 这个也是对的所以需要分支处理传入真正 ref 时 → 返回 T 本身防止嵌套传入 any 时 → 返回 Ref保留 ref 信息总结传入值T 的类型[T] extends [Ref]IfAny 结果最终类型ref(0)Refnumbertrue不是 any → 返回 TRefnumberany值anytrue是 any → 返回RefanyRefany123numberfalse-Refnumber关键点[T] extends [Ref]为 true 时T 可能是真正的 ref也可能是 anyIfAny用来区分这两种情况真正 ref 要返回本身防止嵌套any 要包装成 Ref保留信息2.3 条件分支二RefUnwrapRefT, UnwrapRefT | T详解[T] extends [Ref] 为 falseT 不是 Ref: RefUnwrapRefT, UnwrapRefT | T // 创建一个 Ref值的类型要经过 UnwrapRef 处理RefUnwrapRefT, UnwrapRefT | T的含义RefUnwrapRefT, UnwrapRefT | T // 第一个参数读取类型UnwrapRefT // 第二个参数写入类型UnwrapRefT | TUnwrapRef 是什么定义export type UnwrapRefT T extends Refinfer V ? UnwrapRefSimpleV : UnwrapRefSimpleTUnwrapRef 的作用递归地解包嵌套的 ref// 基本类型 type A UnwrapRefnumber // number type B UnwrapRefstring // string // 一层 ref type C UnwrapRefRefnumber // number解包了 // 嵌套 ref type D UnwrapRefRefRefnumber // number递归解包 // 对象中的 ref type E UnwrapRef{ count: Refnumber } // { count: number }对象属性也被解包UnwrapRef 定义详解先看整体结构这是 TypeScript 的递归类型解包export type UnwrapRefT T extends Refinfer V ? UnwrapRefSimpleV : UnwrapRefSimpleT这是一个条件类型意思是如果 T 是一个 Ref 类型就解包出它里面的值 V然后递归处理 V如果 T 不是 Ref 类型就直接用 UnwrapRefSimple 处理 T拆解关键语法T extends Refinfer V这是 TypeScript 的模式匹配语法// 假设有一个 Ref 类型 type RefT { value: T } // infer V 的意思是把 Ref 里面的类型提取出来命名为 V type GetRefValueT T extends Refinfer V ? V : never // 使用 type A GetRefValueRefnumber // number提取出来了 type B GetRefValuestring // never不是 RefUnwrapRefSimpleV这是另一个类型工具用来处理非 Ref 类型的解包比如对象、数组等执行流程示例例子1基本类型type T1 UnwrapRefnumber // number 不是 Ref → 走第二支UnwrapRefSimplenumber // 结果number例子2一层 Reftype T2 UnwrapRefRefnumber // Refnumber 是 Ref → 走第一支UnwrapRefSimplenumber // 结果number例子3嵌套 Reftype T3 UnwrapRefRefRefnumber // 第一次RefRefnumber 是 Ref → 提取 V Refnumber // 调用 UnwrapRefSimpleRefnumber // UnwrapRefSimple 内部又会调用 UnwrapRef递归 // 第二次Refnumber 是 Ref → 提取 V number // 调用 UnwrapRefSimplenumber // 结果number例子4对象包含 Reftype T4 UnwrapRef{ count: Refnumber } // { count: Refnumber } 不是 Ref → 走第二支 // UnwrapRefSimple{ count: Refnumber } // UnwrapRefSimple 会遍历对象属性对每个属性递归调用 UnwrapRef // 最终{ count: number }配合 UnwrapRefSimple 看export type UnwrapRefSimpleT T extends Builtin ? T : // 基本类型直接返回 T extends Mapinfer K, infer V ? MapK, UnwrapRefSimpleV : // Map 特殊处理 T extends Setinfer V ? SetUnwrapRefSimpleV : // Set 特殊处理 T extends object ? { [P in keyof T]: UnwrapRefT[P] } : // 对象递归解包 T完整流程示例type Test UnwrapRef{ count: Refnumber, user: { name: Refstring, age: number }, tags: RefSetRefstring } // 执行过程 // 1. { count: Ref... } 不是 Ref → UnwrapRefSimple // 2. 遍历对象属性 // - count: UnwrapRefRefnumber → number // - user: UnwrapRef{ name: Refstring, age: number } // → 递归处理 user 对象 // * name: UnwrapRefRefstring → string // * age: number → number // - tags: UnwrapRefRefSetRefstring // → 提取 SetRefstring // → UnwrapRefSimpleSetRefstring // → 处理 SetSetUnwrapRefSimpleRefstring // → UnwrapRefSimpleRefstring → string // → 最终Setstring // 结果 { count: number, user: { name: string, age: number }, tags: Setstring }为什么需要递归解包没有递归解包导致的问题// 假设只有一层解包 type ShallowUnwrapT T extends Refinfer V ? V : T const obj ref({ user: ref({ name: ref(vue) }) }) // 使用时 obj.value.user // 类型还是 Ref{ name: Refstring } ❌ obj.value.user.value.name.value // 要写一堆 .value有递归解包const obj ref({ user: ref({ name: ref(vue) }) }) // 使用时 obj.value.user.name // 直接是 string✅ // 所有层级的 ref 都被自动解开了类比理解把UnwrapRef想象成一个剥洋葱的过程// 洋葱RefRefRefnumber UnwrapRefRefRefRefnumber // 第一层发现是 Ref → 剥开得到 RefRefnumber // 第二层发现还是 Ref → 再剥开得到 Refnumber // 第三层发现还是 Ref → 再剥开得到 number // 结果number把UnwrapRefSimple想象成处理各种食材的工具基本类型Builtin→ 直接吃Map/Set → 特殊处理对象 → 每个属性都剥一遍其他 → 保持原样回到RefUnwrapRefT, UnwrapRefT | TRefUnwrapRefT, UnwrapRefT | T // 第一个参数读取类型UnwrapRefT // 第二个参数写入类型UnwrapRefT | T为什么写入类型要允许两种看例子理解例子传入普通值const count ref(0) // T number // RefUnwrapRefnumber, UnwrapRefnumber | number // Refnumber, number | number // Refnumber, number count.value 10 // ✅ 写入 number count.value 20 // ❌ 类型错误只能写 number例子赋值带 ref 的对象const state ref({ name: ref(vue), age: ref(3) }) // 类型Ref{ name: string, age: number }, // { name: string, age: number } | { name: Refstring, age: Refnumber } // 读取时自动解包 console.log(state.value.name) // string // 写入时两种都支持 state.value { name: react, age: 10 } // ✅ 普通对象 state.value { name: ref(angular), age: ref(5) } // ✅ 带 ref 的对象如果不这样设计会怎样方案A只允许 UnwrapRefRefUnwrapRefT, UnwrapRefT // 写入只能写解包后的类型 const obj ref({ count: ref(0) }) obj.value { count: 10 } // ✅ 可以 obj.value { count: ref(20) } // ❌ 类型错误不能写 ref // 但有时我们需要写入 ref比如从另一个 ref 赋值方案B只允许 TRefUnwrapRefT, T // 写入只能写原始类型 const obj ref({ count: ref(0) }) obj.value { count: ref(20) } // ✅ 可以 obj.value { count: 10 } // ❌ 类型错误不能写普通值 // 但大多数时候我们从 API 拿到的是普通对象Vue 的方案两者都允许RefUnwrapRefT, UnwrapRefT | T // 两种都支持 // 既可以从 API 赋值普通对象 // 也可以从其他 ref 赋值总结参数作用为什么这样设计UnwrapRefT读取类型让使用者直接拿到解包后的值方便使用UnwrapRefT或T写入类型既支持普通对象也支持带 ref 的对象更灵活通俗理解读取时Vue 帮你把里面的 ref 都解开了你直接拿值用写入时Vue 很宽容你想传普通对象也行想传带 ref 的对象也行这就是 Vue 响应式系统读取方便写入灵活的设计哲学3. shallowRef - 浅层响应式 ref 的实现持续更新中未完待续~

相关文章:

Vue 3 源码阅读笔记:ref.ts

文章目录一、文件概览二、核心数据结构1. Ref 接口定义三、核心函数实现1. isRef - 类型守卫2. r[ReactiveFlags.IS_REF]详解一、 r[ReactiveFlags.IS_REF] 是什么意思?二、这个标记是怎么来的?三、为什么需要这个标记?四、完整的标记系统五、…...

Java面向对象—反射

反射1、反射(Reflection):是Java被视为“动态”语言的关键,反射机制允许在执行期间借助于Reflection的API取得任何类(接口)的内部信息,并能直接操作任意对象的内部信息。2、Java反射机制主要提供了以下功能:(1&#xf…...

MATLAB高效声发射多通道数据分离与新数据集构建

matlab高效分离声发射各通道数据,构建新的数据集,亲测运行有效,小样本和大样本(百万级别)均适用,专业性和针对性强,确保运行无误可以直接最近在实验室折腾声发射数据,发现多通道采集的数据处理起来特别费劲…...

距离提交只剩3天,查重48%:毕业之家AI工具把我从延毕边缘拉了回来

毕业之家(官网https://www.biye.com)是聚焦国内高校论文全生命周期的 AI 学术服务平台,作为 PaperRed 核心合作与技术支撑方,以学术合规为核心、高效便捷为导向,打造了选题到答辩的一站式闭环服务,尤其针对…...

跨境卖家如何用品类矩阵规划减少对单一类目的依赖

在波谲云诡的跨境电商领域,许多卖家曾凭借一款爆品迅速崛起,却又因市场风向突变、政策调整或供应链断裂而骤然跌落。这种“成也萧何,败也萧何”的单一品类依赖症,已成为悬在众多跨境企业头上的达摩克利斯之剑。要构建可持续、抗风…...

Vue3+Element Plus实战:给el-dialog加个『老板键』(一键全屏/拖拽/记忆位置)

Vue3 Element Plus 弹窗『老板键』:全屏、拖拽与位置记忆的工程化实现 你是否遇到过这样的场景?正在一个复杂的后台管理系统中处理数据,弹窗里展示着关键图表或表单,突然需要快速切换到另一个应用,或者临时需要隐藏当…...

Java 面向对象设计题3

11. 用户类设计 11.定义一个用户类(User),包含用户名(username)和密码(password)属性,提供静态方法验证密码是否有效(长度至少为6),并提供getter和setter方法。 class User {private String username;private String password;public User(St…...

Visual Studio Code 安装和配置

一、VS Code 1. VS Code下载地址 官方下载地址:https://code.visualstudio.com/访问后页面会自动识别你的系统(Windows/macOS/Linux),显示对应版本的下载按钮,直接点击即可。 2. 安装步骤(以 Windows 为…...

回形取数-进阶题5

回形取数 题目 问题描述 回形取数就是沿矩阵的边取数,若当前方向上无数可取或已经取过,则左转90度。一开始位于矩阵左上角,方向向下。输入说明 输入第一行是两个不超过200的正整数m, n,表示矩阵的行和列。接下来m行每行n个整数&am…...

ESP8266+Blinker打造智能家居远程控制开关

1. 从零开始:为什么选择ESP8266和Blinker? 如果你对智能家居感兴趣,想自己动手做个远程开关,控制家里的灯、风扇或者小电器,但又觉得那些成品智能插座太贵,或者功能不够灵活,那你今天算是来对地…...

OpenBMC实战指南(一):obmc-console服务端与客户端的深度解析

1. 初识obmc-console:它到底是什么,能帮你做什么? 如果你刚开始接触OpenBMC,可能会被一堆服务名搞得晕头转向。今天咱们就来聊聊其中一个非常核心,但又常常被误解的组件:obmc-console。简单来说&#xff0c…...

Ubuntu环境下离线部署Docker生态全攻略:从安装到镜像迁移

1. 为什么需要离线部署Docker?从企业内网说起 大家好,我是老张,在运维和开发这个行当里摸爬滚打了十几年,经手过不少企业级项目。今天想和大家聊聊一个非常实际,但又常常让新手头疼的场景:在完全没有外网的…...

【LWIP】MCU通过ICMP协议实现主动PING检测网络设备状态

1. 为什么你的MCU需要主动PING?一个真实的故事 大家好,我是老张,在嵌入式网络这块摸爬滚打了十几年。今天想和大家聊聊一个看似简单,但在实际项目中却至关重要的功能:让MCU主动去PING网络里的其他设备。 你可能已经用L…...

Flutter 三方库 dart_json_annotations 的鸿蒙化适配指南 - 定义严谨的数据契约、在鸿蒙端实现自动化 JSON 注解实战

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.csdn.net Flutter 三方库 dart_json_annotations 的鸿蒙化适配指南 - 定义严谨的数据契约、在鸿蒙端实现自动化 JSON 注解实战 前言 在进行 Flutter for OpenHarmony 的全场景应用开发时&#xff0…...

一文看懂AI智能体协议家族:MCP、A2A、ACP全解析,小白程序员必收藏

在AI智能体(Agent)迅猛发展的当下,MCP、A2A、ACP、UTCP、ANP……各种协议层出不穷,几乎每隔一段时间,科技公司就会为“字母家族”增添新成员。归根结底,所有AI智能体协议的目标都是标准化智能体的通信方式&…...

拒绝黑盒!一文看懂大模型底层原理与产品区别,小白程序员必收藏

在当今数字化时代,AI 大模型早已不是陌生词汇 —— 从日常聊天的 ChatGPT,到帮我们处理工作的智能助手,它正悄悄改变着我们的生活与工作节奏。但对大多数人来说,AI 大模型就像个 “黑盒子”:知道它好用,却搞…...

Flutter 三方库 w_transport 的鸿蒙化适配指南 - 构建高可靠网络传输层、实现鸿蒙端复杂协议交互实战

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.csdn.net Flutter 三方库 w_transport 的鸿蒙化适配指南 - 构建高可靠网络传输层、实现鸿蒙端复杂协议交互实战 前言 在开发 Flutter for OpenHarmony 大型商业应用时,简单的 HTTP 请求…...

Flutter 三方库 codenic_bloc_use_case 的鸿蒙化适配指南 - 践行整洁架构、在 BLoC 中优雅封装鸿蒙业务用例实战

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.csdn.net Flutter 三方库 codenic_bloc_use_case 的鸿蒙化适配指南 - 践行整洁架构、在 BLoC 中优雅封装鸿蒙业务用例实战 前言 在进行 Flutter for OpenHarmony 的大型项目开发时,复杂…...

Flutter 三方库 kiss_dependencies 的鸿蒙化适配指南 - 践行极简依赖注入、实现鸿蒙跨平台工程的高效解耦

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.csdn.net Flutter 三方库 kiss_dependencies 的鸿蒙化适配指南 - 践行极简依赖注入、实现鸿蒙跨平台工程的高效解耦 前言 在 Flutter for OpenHarmony 的实际开发中,随着业务逻辑从单一…...

3秒解锁百度网盘资源:零技术门槛的提取码查询工具使用指南

3秒解锁百度网盘资源:零技术门槛的提取码查询工具使用指南 【免费下载链接】baidupankey 项目地址: https://gitcode.com/gh_mirrors/ba/baidupankey 资源获取的隐形墙:你是否也遇到这些困境? 想象这样的场景:设计师小陈…...

200年前的蒸汽机工人,其实早就预言了今天程序员的命运

最近看到一篇很有意思的文章,作者在读 OpenAI 关于“线束工程”(Harness Engineering)的博客时,突然意识到一件事:这个模式他见过,不止一次,而是三次。这三次跨越了两百多年,但本质上…...

告别提取码焦虑:零门槛百度网盘资源解锁工具让你秒级获取文件

告别提取码焦虑:零门槛百度网盘资源解锁工具让你秒级获取文件 【免费下载链接】baidupankey 项目地址: https://gitcode.com/gh_mirrors/ba/baidupankey 一、被提取码困住的三个真实故事 医生王主任的紧急时刻 凌晨两点,急诊科王主任收到同事发…...

Windows Subsystem for Android (WSA) 实战指南:从环境搭建到高效应用

Windows Subsystem for Android (WSA) 实战指南:从环境搭建到高效应用 【免费下载链接】WSA Developer-related issues and feature requests for Windows Subsystem for Android 项目地址: https://gitcode.com/gh_mirrors/ws/WSA 一、WSA技术解析&#xff…...

SpringBoot + 腾讯地图实战:打造全能型地理位置服务平台,开箱即用!

大家好,我是小悟。 什么是腾讯地图 腾讯地图(Tencent Map)是腾讯公司推出的一款数字地图服务,提供丰富的地图展示、定位、搜索、导航等功能。作为国内领先的地图服务提供商,腾讯地图拥有以下特点: 海量数据…...

基于STM32的多屏可编程HID控制键盘设计

1. 项目概述MultiPad 是一款基于 STM32F103VET6 微控制器构建的高自由度桌面控制键盘系统,其设计目标是为开发者、内容创作者及效率追求者提供一套可深度定制、即插即用、软硬协同的物理交互层解决方案。与传统机械键盘或商用宏键盘不同,MultiPad 并非以…...

De Boor算法实战:从理论到B样条曲线点计算的完整实现

1. 从“搭积木”到“画曲线”:为什么你需要De Boor算法? 如果你玩过3D建模、做过动画路径设计,或者搞过机器人轨迹规划,那你肯定遇到过“画一条光滑曲线”这个看似简单、实则让人头疼的问题。直接用直线段连接控制点?太…...

信号与系统 - 从方波到频谱:周期信号傅里叶级数的几何与物理诠释

1. 从方波说起:一个工程直觉的切入点 很多朋友一听到“傅里叶级数”、“频谱”这些词,第一反应可能就是头疼,满眼的积分号和复数,感觉离实际工程应用很远。我刚开始学信号与系统的时候也是这种感觉,直到我遇到了方波这…...

Windows系统下Typora的安装与激活全流程解析

1. 从零开始:为什么选择Typora以及如何获取它 如果你经常需要写点东西,无论是技术文档、学习笔记,还是日常的随笔,那你大概率听说过Markdown。这种用简单符号就能搞定排名的轻量级标记语言,简直是文字工作者的福音。而…...

小学生玩转Arduino---------智能避障小助手

1. 从“倒车指挥员”到“智能避障小助手” 上次我们一起做了一个“倒车指挥员”,用超声波测距器和蜂鸣器模拟了倒车雷达,是不是觉得特别酷?很多小朋友做完之后跑来问我:“老师,这个只能装在‘车’后面吗?能…...

Redis单机多实例部署:从端口隔离到资源优化实战

1. 为什么要在单台机器上跑多个Redis?聊聊我的真实经历 你可能觉得,一台服务器上装一个Redis,让它监听默认的6379端口,这不是天经地义的事情吗?我以前也是这么想的,直到我遇到了下面这些“甜蜜的烦恼”。 最…...