Private
Public Access
1
0
Files
u-desk/docs/02-架构设计/OOP架构/OOP-Composition组合方案.md

572 lines
14 KiB
Markdown
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
# OOP + Composition API 组合使用方案
**日期**: 2026-01-31
**核心理念**: 取长补短,渐进式迁移
---
## 🎯 设计原则
### OOP 负责什么
-**核心业务逻辑**(复杂的状态管理)
-**需要严格初始化顺序的功能**(如 ZIP 浏览)
-**可复用的服务**(文件操作、预览等)
-**需要依赖注入和测试的模块**
### Composition API 负责什么
-**Vue 组件的响应式状态**
-**简单的 UI 逻辑**
-**生命周期钩子**
-**DOM 事件处理**
### 分层架构
```
┌─────────────────────────────────────┐
│ Vue 组件层 (Composition) │ ← UI 交互、生命周期
│ index.vue | 组件 <script setup> │
└──────────────┬──────────────────────┘
│ 使用
┌──────────────▼──────────────────────┐
│ 适配器层 (Composables) │ ← 轻量桥接、响应式转换
│ useFileSystem() | useZipBrowser() │
└──────────────┬──────────────────────┘
│ 调用
┌──────────────▼──────────────────────┐
│ 服务层 (OOP Classes) │ ← 核心逻辑、初始化保证
│ FileSystemService | ZipService │
└─────────────────────────────────────┘
```
---
## 📝 实际代码示例
### 1. 服务层OOP- 解决初始化问题
```typescript
// services/ZipBrowserService.ts
import { ref, computed, type Ref } from 'vue'
import { Message } from '@arco-design/web-vue'
import type { FileApiService } from './FileApiService'
import type { FilePreviewService } from './FilePreviewService'
import type { FileItem } from '@/types/file-system'
/**
* ZIP 浏览服务
* 使用 OOP 封装,构造函数保证初始化顺序
*/
export class ZipBrowserService {
// ========== 状态(私有 ref ==========
private readonly _isBrowsingZip = ref<boolean>(false)
private readonly _currentZipPath = ref<string>('')
private readonly _currentZipDirectory = ref<string>('')
private readonly _pathBeforeZip = ref<string>('')
// ========== 依赖注入(构造时确保初始化) ==========
constructor(
private readonly fileApi: FileApiService,
private readonly previewService: FilePreviewService, // ✅ 保证已初始化
private readonly fileList: Ref<FileItem[]>,
private readonly filePath: Ref<string>
) {
console.log('[ZipBrowserService] 初始化完成')
}
// ========== 公共接口(访问器) ==========
/** 是否正在浏览 ZIP返回 ref */
get isBrowsingZip(): Ref<boolean> {
return this._isBrowsingZip
}
/** 当前 ZIP 路径(返回 ref */
get currentZipPath(): Ref<string> {
return this._currentZipPath
}
/** 当前 ZIP 目录(返回 ref */
get currentZipDirectory(): Ref<string> {
return this._currentZipDirectory
}
/** 显示路径(计算属性) */
get displayPath(): Ref<string> {
return computed(() => {
if (this._currentZipDirectory.value) {
return `📦 ${this._currentZipPath.value} [${this._currentZipDirectory.value}]`
}
return `📦 ${this._currentZipPath.value}`
})
}
// ========== 公共方法 ==========
/**
* 进入 ZIP 浏览模式
*/
async enterZipMode(zipPath: string): Promise<void> {
this._pathBeforeZip.value = this.filePath.value
this._currentZipPath.value = zipPath
this._isBrowsingZip.value = true
this._currentZipDirectory.value = ''
await this.loadZipDirectory()
Message.success('进入 ZIP 浏览模式')
}
/**
* 退出 ZIP 浏览模式
*/
exitZipMode(): void {
this._isBrowsingZip.value = false
this._currentZipPath.value = ''
this._currentZipDirectory.value = ''
this.filePath.value = this._pathBeforeZip.value
Message.info('退出 ZIP 浏览模式')
}
/**
* 获取 ZIP 文件名
*/
getZipFileName(zipPath: string): string {
const parts = zipPath.split(/[/\\]/)
return parts[parts.length - 1] || zipPath
}
/**
* 获取面包屑导航
*/
getZipBreadcrumbs(): ZipBreadcrumbItem[] {
const crumbs: ZipBreadcrumbItem[] = [
{ name: this.getZipFileName(this._currentZipPath.value), path: '' }
]
if (this._currentZipDirectory.value) {
const parts = this._currentZipDirectory.value.split('/')
let currentPath = ''
for (const part of parts) {
currentPath += (currentPath ? '/' : '') + part
crumbs.push({ name: part, path: currentPath })
}
}
return crumbs
}
/**
* 导航到指定目录
*/
async navigateToZipDirectory(path: string): Promise<void> {
this._currentZipDirectory.value = path
await this.loadZipDirectory()
}
// ========== 私有方法 ==========
private async loadZipDirectory(): Promise<void> {
// 加载目录逻辑
}
}
```
### 2. 适配器层Composable- 桥接服务
```typescript
// composables/useZipBrowser.ts
import { useRef } from '@/utils/singleton'
import { ZipBrowserService } from '@/services/ZipBrowserService'
import type { UseZipBrowserOptions } from './types'
/**
* ZIP 浏览 Composable
* 轻量级适配器,桥接 Vue 和服务层
*/
export function useZipBrowser(options: UseZipBrowserOptions) {
// 使用单例模式,确保服务只创建一次
const service = useRef(() => {
console.log('[useZipBrowser] 创建 ZipBrowserService 实例')
return new ZipBrowserService(
options.fileApi, // 依赖注入
options.previewService, // ✅ 保证预览服务已初始化
options.fileList,
options.filePath
)
}, 'zipBrowserService')
// 返回响应式接口Composition API 风格)
return {
// 状态(直接返回 ref
isBrowsingZip: service.isBrowsingZip,
currentZipPath: service.currentZipPath,
currentZipDirectory: service.currentZipDirectory,
displayPath: service.displayPath,
// 方法(绑定到服务实例)
enterZipMode: (path: string) => service.enterZipMode(path),
exitZipMode: () => service.exitZipMode(),
navigateToZipDirectory: (path: string) => service.navigateToZipDirectory(path),
getZipFileName: (path: string) => service.getZipFileName(path),
getZipBreadcrumbs: () => service.getZipBreadcrumbs(),
// 服务实例(可选,用于高级用法)
$service: service
}
}
```
### 3. 单例工具(避免重复创建)
```typescript
// utils/singleton.ts
const singletons = new Map<string, any>()
/**
* 创建或获取单例
* @param factory 工厂函数
* @param key 单例键名
*/
export function useRef<T>(factory: () => T, key: string): T {
if (!singletons.has(key)) {
const instance = factory()
singletons.set(key, instance)
console.log(`[Singleton] 创建 ${key}`)
} else {
console.log(`[Singleton] 复用 ${key}`)
}
return singletons.get(key) as T
}
/**
* 清除单例(测试用)
*/
export function clearRef(key?: string): void {
if (key) {
singletons.delete(key)
} else {
singletons.clear()
}
}
```
### 4. 在组件中使用Composition API
```typescript
// components/FileSystem/index.vue
<script setup lang="ts">
import { ref, computed } from 'vue'
import { useFileOperations } from './composables/useFileOperations'
import { useZipBrowser } from './composables/useZipBrowser'
// ========== 1. 初始化基础服务 ==========
const { listDirectory, extractZipFile, getFileServerURL, fileApi } =
useFileOperations()
// ========== 2. 初始化依赖服务 ==========
const { previewService } = useFilePreview({ filePath })
// ========== 3. 初始化 ZIP 浏览(依赖预览服务) ==========
const zipBrowser = useZipBrowser({
fileApi,
previewService, // ✅ 依赖注入,保证顺序
fileList,
filePath
})
// ========== 4. 使用(和之前一样) ==========
const toolbarConfig = computed(() => ({
isBrowsingZip: zipBrowser.isBrowsingZip.value, // ✅ ref
displayPath: zipBrowser.displayPath.value, // ✅ ref
zipFileName: zipBrowser.getZipFileName(zipBrowser.currentZipPath.value),
zipBreadcrumbs: zipBrowser.getZipBreadcrumbs()
}))
// ========== 5. 事件处理 ==========
const handleEnterZipMode = async (zipPath: string) => {
await zipBrowser.enterZipMode(zipPath) // ✅ 简单调用
}
const handleExitZip = () => {
zipBrowser.exitZipMode() // ✅ 简单调用
}
// ========== 6. 高级用法(可选) ==========
// 直接访问服务实例
const zipService = zipBrowser.$service
console.log(zipService.currentZipPath.value)
</script>
<template>
<!-- 使用方式不变 -->
<Toolbar
:isBrowsingZip="zipBrowser.isBrowsingZip"
:zipFileName="zipBrowser.getZipFileName(zipBrowser.currentZipPath)"
@enter-zip="handleEnterZipMode"
@exit-zip="handleExitZip"
/>
</template>
```
---
## 🔄 渐进式迁移策略
### 阶段 1新功能使用组合方案立即开始
```typescript
// ✅ 新功能:使用 OOP 服务
const newFeature = useNewFeature({
service: new NewFeatureService()
})
```
### 阶段 2问题模块优先迁移本周
```typescript
// ❌ 旧代码(有问题)
const zipBrowser = useZipBrowser({ ... })
// ✅ 新代码(使用服务)
const zipBrowser = useZipBrowser({
fileApi,
previewService, // 依赖注入
fileList,
filePath
})
```
**优先迁移:**
1. `useZipBrowser` - 初始化顺序问题最多
2. `useFilePreview` - 返回值过多
3. `useFileEdit` - 状态管理复杂
### 阶段 3其他功能逐步迁移1-2周
```typescript
// 老代码保持不变,新代码用新方案
const oldFeature = useOldFeature() // 保持原样
const newFeature = useNewFeature({ // 新方案
service: new NewFeatureService()
})
```
### 阶段 4完全迁移后1个月后
```typescript
// 全部使用组合方案
const { fileSystem, preview, zip, edit } = useServices({
services: {
fileSystem: new FileSystemService(),
preview: new FilePreviewService(),
zip: new ZipBrowserService()
}
})
```
---
## 💡 使用场景对比
### 场景 1简单 UI 逻辑(用 Composition API
```typescript
// ✅ 简单的响应式状态
const showDialog = ref(false)
const dialogMessage = ref('')
const openDialog = (msg: string) => {
dialogMessage.value = msg
showDialog.value = true
}
```
### 场景 2复杂业务逻辑用 OOP 服务)
```typescript
// ✅ 复杂的状态管理
class ZipBrowserService {
constructor(
private preview: FilePreviewService, // 依赖注入
private fileApi: FileApiService
) {}
async enterZipMode(path: string) {
// 复杂的初始化逻辑
await this.preview.cleanup()
this._isBrowsingZip.value = true
await this.loadZipContents()
}
}
```
### 场景 3需要组合两者组合使用
```typescript
// 服务层OOP- 核心逻辑
class FilePreviewService {
async previewImage(path: string) {
const url = await this.fileApi.getImageUrl(path)
this._previewUrl.value = url
}
}
// Composable - 桥接到 Vue
function useFilePreview() {
const service = new FilePreviewService(...)
return {
// 响应式状态Composition API
previewUrl: service.previewUrl,
// 方法(委托给服务)
previewImage: (path: string) => service.previewImage(path),
// 服务实例(可选)
$service: service
}
}
// 组件 - 使用
const { previewUrl, previewImage } = useFilePreview()
```
---
## 🎓 最佳实践
### 1. 服务类设计原则
```typescript
class GoodService {
// ✅ 状态用 ref
private readonly _state = ref<State>(initialState)
// ✅ 构造函数注入依赖
constructor(
private readonly dependency: OtherService
) {}
// ✅ 提供访问器
get state(): Ref<State> {
return this._state
}
// ✅ 方法返回值(不返回 ref
doSomething(): void {
this._state.value = { ... }
}
}
```
### 2. Composable 设计原则
```typescript
function useGoodService(options: Options) {
// ✅ 创建服务实例
const service = new GoodService(options.dependency)
return {
// ✅ 返回 ref响应式
state: service.state,
// ✅ 绑定方法
doSomething: () => service.doSomething(),
// ✅ 可选:暴露服务
$service: service
}
}
```
### 3. 组件使用原则
```typescript
// ✅ 简单场景:只用返回值
const { state, doSomething } = useGoodService()
// ✅ 复杂场景:访问服务实例
const { $service } = useGoodService()
$service.advancedMethod()
// ✅ 生命周期钩子Composition API
onMounted(() => {
$service.initialize()
})
```
---
## 📊 对比总结
| 维度 | OOP 服务 | Composable | 组件使用 |
|-----|---------|-----------|---------|
| **适用场景** | 复杂逻辑、初始化顺序 | 简单逻辑、UI 状态 | 组合使用 |
| **状态管理** | ref 私有字段 | ref 变量 | ref 变量 |
| **依赖注入** | 构造函数 | 函数参数 | 函数参数 |
| **测试性** | ✅ 容易 | ⚠️ 中等 | ⚠️ 中等 |
| **Vue 兼容** | ⚠️ 需要适配 | ✅ 完美 | ✅ 完美 |
| **初始化保证** | ✅ 构造函数 | ❌ 手动保证 | - |
---
## 🚀 快速开始模板
### 创建服务类
```bash
# 1. 创建服务文件
services/MyFeatureService.ts
# 2. 创建 composable 适配器
composables/useMyFeature.ts
# 3. 在组件中使用
components/MyComponent.vue
```
### 模板代码
```typescript
// 1. 服务类
export class MyFeatureService {
constructor(private dep: DependencyService) {}
get state() { return this._state }
doSomething() { ... }
}
// 2. Composable
export function useMyFeature() {
const service = new MyFeatureService(dep)
return {
state: service.state,
doSomething: () => service.doSomething(),
$service: service
}
}
// 3. 组件
const { state, doSomething } = useMyFeature()
```
---
## ✅ 总结
### 组合方案的优势
1. **✅ 解决初始化问题** - OOP 构造函数保证顺序
2. **✅ 保持开发体验** - Composition API 风格
3. **✅ 渐进式迁移** - 不需要大规模重构
4. **✅ 高内聚低耦合** - 服务封装,适配器桥接
5. **✅ 易于测试** - 服务层独立测试
### 核心理念
> **OOP 负责复杂逻辑Composition 负责 UI 交互**
---
**生成时间**: 2026-01-31
**下一步**: 创建第一个 OOP 服务示例ZipBrowserService