Files
sanbuphy ef70b1d8e1 feat: add comprehensive backend topics and fix build issues
## 新增内容

### 附录文档扩展
- 扩展前端项目架构文档 (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)
- 更新依赖版本
2026-02-26 04:35:28 +08:00

195 lines
5.3 KiB
Vue

<template>
<div class="url-demo">
<div class="demo-header">
<div class="demo-title">URL 访问全流程</div>
<button class="play-btn" @click="autoPlay" :disabled="playing">
{{ playing ? '播放中...' : '▶ 自动演示' }}
</button>
</div>
<div class="flow">
<div class="flow-side client-side">
<div class="side-label">浏览器</div>
</div>
<div class="flow-steps">
<div
v-for="(step, i) in steps"
:key="step.name"
class="step"
:class="{ active: current >= i, highlight: current === i }"
@click="current = i"
>
<div class="step-line">
<span class="step-dot"></span>
<span v-if="i < steps.length - 1" class="step-connector"></span>
</div>
<div class="step-body">
<div class="step-head">
<span class="step-num">{{ i + 1 }}</span>
<span class="step-name">{{ step.name }}</span>
<span class="step-dir" :class="step.dir">{{ step.dir === 'right' ? '→' : '←' }}</span>
</div>
<div v-if="current >= i" class="step-detail">{{ step.detail }}</div>
</div>
</div>
</div>
<div class="flow-side server-side">
<div class="side-label">服务器</div>
</div>
</div>
</div>
</template>
<script setup>
import { ref, onUnmounted } from 'vue'
const current = ref(-1)
const playing = ref(false)
let timer = null
const steps = [
{ name: 'URL 解析', dir: 'right', detail: 'https://example.com → 协议: https, 域名: example.com, 路径: /' },
{ name: 'DNS 解析', dir: 'right', detail: '向 DNS 服务器查询,将域名翻译为 IP 地址 93.184.216.34' },
{ name: 'TCP 三次握手', dir: 'right', detail: 'SYN → SYN-ACK → ACK,建立可靠的传输连接' },
{ name: 'TLS 握手', dir: 'right', detail: '交换密钥、验证证书,建立 HTTPS 加密通道' },
{ name: '发送 HTTP 请求', dir: 'right', detail: 'GET /index.html HTTP/1.1 Host: example.com' },
{ name: '服务器处理', dir: 'left', detail: '解析请求 → 执行业务逻辑 → 查询数据库 → 组装响应' },
{ name: '返回 HTTP 响应', dir: 'left', detail: 'HTTP/1.1 200 OK Content-Type: text/html' },
{ name: '浏览器渲染', dir: 'left', detail: 'HTML → DOM 树 → 样式计算 → 布局 → 绘制到屏幕' }
]
const autoPlay = () => {
if (timer) clearInterval(timer)
current.value = -1
playing.value = true
let i = 0
timer = setInterval(() => {
current.value = i
i++
if (i >= steps.length) {
if (timer) clearInterval(timer)
playing.value = false
}
}, 800)
}
onUnmounted(() => { if (timer) clearInterval(timer) })
</script>
<style scoped>
.url-demo {
border: 1px solid var(--vp-c-divider);
border-radius: 8px;
background: var(--vp-c-bg-soft);
padding: 1rem 1.2rem;
margin: 1rem 0;
}
.demo-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.8rem;
}
.demo-title {
font-size: 0.75rem;
font-weight: 600;
color: var(--vp-c-text-2);
}
.play-btn {
font-size: 0.65rem;
padding: 0.25rem 0.6rem;
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
background: var(--vp-c-bg);
color: var(--vp-c-text-2);
cursor: pointer;
transition: border-color 0.2s;
}
.play-btn:hover:not(:disabled) { border-color: var(--vp-c-brand); }
.play-btn:disabled { opacity: 0.5; cursor: default; }
.flow {
display: flex;
gap: 0.5rem;
}
.flow-side {
display: flex;
align-items: flex-start;
padding-top: 0.3rem;
}
.side-label {
writing-mode: vertical-rl;
font-size: 0.65rem;
font-weight: 600;
color: var(--vp-c-text-3);
letter-spacing: 0.15em;
}
.flow-steps { flex: 1; display: flex; flex-direction: column; }
.step {
display: flex;
gap: 0.5rem;
opacity: 0.35;
transition: opacity 0.3s;
}
.step.active { opacity: 1; }
.step.highlight .step-dot { box-shadow: 0 0 0 3px rgba(var(--vp-c-brand-rgb, 100, 108, 255), 0.2); }
.step-line {
display: flex;
flex-direction: column;
align-items: center;
width: 0.8rem;
flex-shrink: 0;
}
.step-dot {
width: 0.5rem;
height: 0.5rem;
border-radius: 50%;
background: var(--vp-c-divider);
transition: all 0.3s;
flex-shrink: 0;
margin-top: 0.35rem;
}
.step.active .step-dot { background: var(--vp-c-brand); }
.step-connector {
flex: 1;
width: 1px;
background: var(--vp-c-divider);
min-height: 0.8rem;
}
.step.active .step-connector { background: var(--vp-c-brand); opacity: 0.3; }
.step-body { flex: 1; padding-bottom: 0.5rem; }
.step-head { display: flex; align-items: center; gap: 0.35rem; }
.step-num {
font-size: 0.6rem;
font-weight: 600;
color: var(--vp-c-text-3);
min-width: 1rem;
}
.step.active .step-num { color: var(--vp-c-brand); }
.step-name {
font-size: 0.72rem;
font-weight: 600;
color: var(--vp-c-text-2);
}
.step-dir {
font-size: 0.7rem;
color: var(--vp-c-text-3);
}
.step-dir.right { color: var(--vp-c-brand); }
.step-dir.left { color: #e879a0; }
.step-detail {
font-size: 0.63rem;
color: var(--vp-c-text-3);
margin-top: 0.2rem;
padding: 0.25rem 0.4rem;
background: var(--vp-c-bg);
border-radius: 4px;
font-family: monospace;
line-height: 1.5;
}
@media (max-width: 480px) {
.flow-side { display: none; }
}
</style>