ef70b1d8e1
## 新增内容 ### 附录文档扩展 - 扩展前端项目架构文档 (frontend-project-architecture.md) - 扩展后端项目架构文档 (backend-project-architecture.md) - 扩展数据治理文档 (data-governance.md) - 扩展数据可视化文档 (data-visualization.md) - 扩展分布式系统文档 (distributed-systems.md) - 扩展高可用文档 (high-availability.md) - 扩展单体到微服务文档 (monolith-to-microservices.md) - 扩展系统设计方法论文档 (system-design-methodology.md) - 扩展 Docker 容器文档 (docker-containers.md) - 扩展 Kubernetes 文档 (kubernetes.md) - 扩展 Linux 基础文档 (linux-basics.md) - 扩展神经网络文档 (neural-networks.md) ### 新增交互式组件 - 数据治理组件: DataQualityDemo, DataGovernanceFrameworkDemo, DataLineageDemo - 数据可视化组件: ChartTypeSelectorDemo, DashboardLayoutDemo - 分布式系统组件: CAPTheoremDemo, ConsistencyModelsDemo, DistributedChallengesDemo - 高可用组件: AvailabilityCalculatorDemo, FailoverStrategyDemo - 系统设计组件: SystemDesignStepsDemo, CapacityEstimationDemo - Docker 容器组件: DockerArchitectureDemo, DockerLifecycleDemo - Kubernetes 组件: K8sArchitectureDemo, K8sWorkloadsDemo - Linux 基础组件: LinuxFileSystemDemo, LinuxCommandDemo, LinuxPermissionsDemo - 神经网络组件: NeuronDemo, NetworkLayersDemo, NetworkArchitectureDemo - 单体到微服务组件: ArchEvolutionDemo - 项目架构组件: ProjectArchitectureComparisonDemo - 附录导航组件: AppendixFlowMap ### 英文版重构 - 将 en-us 目录重命名为 en - 更新相关配置和组件中的语言代码 ## Bug 修复 - 修复 index.js 中重复的组件导入语句导致的 build 失败 - 恢复被注释的 InvertedIndexDemo 和 SearchRelevanceDemo 导入 - 修复 HomeFeatures.vue 中 en-us 与 config.mjs 中 en 不一致导致的语言切换问题 ## 其他改进 - 添加构建脚本 (scripts/build.mjs) - 更新依赖版本
133 lines
6.6 KiB
Vue
133 lines
6.6 KiB
Vue
<!--
|
||
ConsistencyModelsDemo.vue
|
||
一致性模型演示:展示强一致性、最终一致性、因果一致性的区别
|
||
-->
|
||
<template>
|
||
<div class="consistency-demo">
|
||
<div class="header">
|
||
<div class="title">一致性模型对比</div>
|
||
<div class="subtitle">点击查看不同一致性模型的行为差异</div>
|
||
</div>
|
||
|
||
<div class="model-tabs">
|
||
<div
|
||
v-for="m in models"
|
||
:key="m.key"
|
||
:class="['tab', { active: activeModel === m.key }]"
|
||
@click="activeModel = m.key"
|
||
>
|
||
{{ m.name }}
|
||
</div>
|
||
</div>
|
||
|
||
<div v-if="current" class="model-detail">
|
||
<div class="model-name">{{ current.name }}</div>
|
||
<div class="model-desc">{{ current.desc }}</div>
|
||
|
||
<div class="timeline">
|
||
<div v-for="(step, i) in current.steps" :key="i" class="step">
|
||
<div class="step-time">T{{ i + 1 }}</div>
|
||
<div class="step-nodes">
|
||
<div
|
||
v-for="(node, ni) in step.nodes"
|
||
:key="ni"
|
||
:class="['node', node.status]"
|
||
>
|
||
<div class="node-label">{{ node.name }}</div>
|
||
<div class="node-value">{{ node.value }}</div>
|
||
</div>
|
||
</div>
|
||
<div class="step-desc">{{ step.desc }}</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="model-tradeoff">
|
||
<span class="label">权衡:</span>{{ current.tradeoff }}
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script setup>
|
||
import { ref, computed } from 'vue'
|
||
|
||
const activeModel = ref('strong')
|
||
|
||
const models = [
|
||
{
|
||
key: 'strong',
|
||
name: '强一致性',
|
||
desc: '写入成功后,所有节点立即返回最新值。像单机数据库一样的体验。',
|
||
tradeoff: '延迟高(需要等所有节点确认),可用性低(节点故障时可能阻塞)',
|
||
steps: [
|
||
{ nodes: [{ name: '节点A', value: 'v1', status: 'ok' }, { name: '节点B', value: 'v1', status: 'ok' }, { name: '节点C', value: 'v1', status: 'ok' }], desc: '初始状态,所有节点数据一致' },
|
||
{ nodes: [{ name: '节点A', value: 'v2 ✍️', status: 'writing' }, { name: '节点B', value: '同步中...', status: 'syncing' }, { name: '节点C', value: '同步中...', status: 'syncing' }], desc: '客户端写入 v2,等待所有节点确认' },
|
||
{ nodes: [{ name: '节点A', value: 'v2', status: 'ok' }, { name: '节点B', value: 'v2', status: 'ok' }, { name: '节点C', value: 'v2', status: 'ok' }], desc: '所有节点确认后才返回成功,读任意节点都是 v2' }
|
||
]
|
||
},
|
||
{
|
||
key: 'eventual',
|
||
name: '最终一致性',
|
||
desc: '写入后不等所有节点同步,数据最终会一致,但中间可能读到旧值。',
|
||
tradeoff: '延迟低、可用性高,但可能短暂读到旧数据',
|
||
steps: [
|
||
{ nodes: [{ name: '节点A', value: 'v1', status: 'ok' }, { name: '节点B', value: 'v1', status: 'ok' }, { name: '节点C', value: 'v1', status: 'ok' }], desc: '初始状态' },
|
||
{ nodes: [{ name: '节点A', value: 'v2 ✍️', status: 'writing' }, { name: '节点B', value: 'v1', status: 'stale' }, { name: '节点C', value: 'v1', status: 'stale' }], desc: '写入 A 后立即返回成功,B/C 还是旧值' },
|
||
{ nodes: [{ name: '节点A', value: 'v2', status: 'ok' }, { name: '节点B', value: 'v2', status: 'ok' }, { name: '节点C', value: 'v1→v2', status: 'syncing' }], desc: '后台异步同步,逐渐达到一致' }
|
||
]
|
||
},
|
||
{
|
||
key: 'causal',
|
||
name: '因果一致性',
|
||
desc: '有因果关系的操作保证顺序,无因果关系的操作可以乱序。介于强一致和最终一致之间。',
|
||
tradeoff: '比强一致性延迟低,比最终一致性更可预测',
|
||
steps: [
|
||
{ nodes: [{ name: '用户A', value: '发帖: "你好"', status: 'ok' }, { name: '用户B', value: '看到帖子', status: 'ok' }, { name: '用户C', value: '看到帖子', status: 'ok' }], desc: '用户 A 发帖' },
|
||
{ nodes: [{ name: '用户A', value: '发帖: "你好"', status: 'ok' }, { name: '用户B', value: '回复: "嗨!"', status: 'writing' }, { name: '用户C', value: '看到帖子', status: 'ok' }], desc: '用户 B 回复(因果依赖于 A 的帖子)' },
|
||
{ nodes: [{ name: '用户A', value: '看到回复', status: 'ok' }, { name: '用户B', value: '回复: "嗨!"', status: 'ok' }, { name: '用户C', value: '先看到帖子再看到回复', status: 'ok' }], desc: '所有人都先看到帖子再看到回复(因果顺序保证)' }
|
||
]
|
||
}
|
||
]
|
||
|
||
const current = computed(() => models.find(m => m.key === activeModel.value))
|
||
</script>
|
||
|
||
<style scoped>
|
||
.consistency-demo {
|
||
border: 1px solid var(--vp-c-divider); background: var(--vp-c-bg-soft);
|
||
border-radius: 12px; padding: 1.5rem; margin: 1.5rem 0;
|
||
}
|
||
.header { margin-bottom: 1rem; }
|
||
.title { font-weight: 700; font-size: 1.1rem; }
|
||
.subtitle { color: var(--vp-c-text-2); font-size: 0.9rem; }
|
||
.model-tabs { display: flex; gap: 0.5rem; margin-bottom: 1rem; flex-wrap: wrap; }
|
||
.tab {
|
||
padding: 0.4rem 0.75rem; border-radius: 6px; cursor: pointer;
|
||
font-size: 0.85rem; background: var(--vp-c-bg); border: 1px solid var(--vp-c-divider);
|
||
transition: all 0.2s;
|
||
}
|
||
.tab:hover { border-color: var(--vp-c-brand); }
|
||
.tab.active { border-color: var(--vp-c-brand); background: rgba(var(--vp-c-brand-rgb), 0.05); font-weight: 600; }
|
||
.model-detail { background: var(--vp-c-bg); border-radius: 8px; padding: 1rem; border: 1px solid var(--vp-c-divider); }
|
||
.model-name { font-weight: 700; font-size: 0.95rem; }
|
||
.model-desc { color: var(--vp-c-text-2); font-size: 0.82rem; margin-bottom: 0.75rem; }
|
||
.timeline { display: flex; flex-direction: column; gap: 0.5rem; margin-bottom: 0.75rem; }
|
||
.step { display: flex; gap: 0.5rem; align-items: center; flex-wrap: wrap; }
|
||
.step-time { font-weight: 700; font-size: 0.8rem; color: var(--vp-c-brand); min-width: 28px; }
|
||
.step-nodes { display: flex; gap: 0.4rem; flex: 1; }
|
||
.node {
|
||
padding: 0.3rem 0.5rem; border-radius: 4px; font-size: 0.72rem;
|
||
border: 1px solid var(--vp-c-divider); flex: 1; text-align: center;
|
||
}
|
||
.node.ok { background: rgba(34,197,94,0.08); border-color: #22c55e; }
|
||
.node.writing { background: rgba(var(--vp-c-brand-rgb),0.08); border-color: var(--vp-c-brand); }
|
||
.node.syncing { background: rgba(245,158,11,0.08); border-color: #f59e0b; }
|
||
.node.stale { background: rgba(239,68,68,0.08); border-color: #ef4444; }
|
||
.node-label { font-weight: 600; }
|
||
.node-value { color: var(--vp-c-text-2); }
|
||
.step-desc { font-size: 0.75rem; color: var(--vp-c-text-3); width: 100%; margin-top: 0.15rem; }
|
||
.model-tradeoff { font-size: 0.82rem; }
|
||
.label { font-weight: 600; color: var(--vp-c-text-2); }
|
||
@media (max-width: 640px) { .step { flex-direction: column; } .step-nodes { width: 100%; } }
|
||
</style>
|