Files

337 lines
6.8 KiB
Vue
Raw Permalink Normal View History

<template>
<div class="container-docker-demo">
<div class="demo-header">
<span class="icon">🐳</span>
<span class="title">Docker 容器化演示</span>
<span class="subtitle">理解容器如何让应用"一次打包,到处运行"</span>
</div>
<div class="docker-visualization">
<div
class="layer traditional"
:class="{ active: showTraditional }"
@click="showTraditional = true; showDocker = false"
>
<h5>传统部署</h5>
<div class="server-stack">
<div class="layer-item app">应用 A</div>
<div v-if="showConflict" class="layer-item conflict">依赖冲突!</div>
<div class="layer-item deps">依赖库 v1.0</div>
<div class="layer-item os">操作系统</div>
<div class="layer-item hardware">物理服务器</div>
</div>
</div>
<div class="vs-divider">VS</div>
<div
class="layer docker"
:class="{ active: showDocker }"
@click="showDocker = true; showTraditional = false"
>
<h5>Docker 容器</h5>
<div class="docker-stack">
<div class="containers">
<div class="container-box">
<div class="container-app">应用 A</div>
<div class="container-deps">依赖 v1.0</div>
</div>
<div class="container-box">
<div class="container-app">应用 B</div>
<div class="container-deps">依赖 v2.0</div>
</div>
</div>
<div class="docker-engine">Docker Engine</div>
<div class="host-os">宿主机操作系统</div>
<div class="hardware">物理服务器</div>
</div>
</div>
</div>
<div class="benefits-grid">
<div
v-for="benefit in benefits"
:key="benefit.title"
class="benefit-card"
>
<div class="benefit-icon">
{{ benefit.icon }}
</div>
<div class="benefit-title">
{{ benefit.title }}
</div>
<div class="benefit-desc">
{{ benefit.desc }}
</div>
</div>
</div>
<div class="info-box">
<span class="icon">💡</span>
<strong>核心思想</strong>容器化让应用"一次构建,到处运行"解决了环境一致性和快速部署的问题
</div>
</div>
</template>
<script setup>
import { ref } from 'vue'
const showTraditional = ref(true)
const showDocker = ref(false)
const showConflict = ref(false)
const benefits = [
{
icon: '📦',
title: '环境一致性',
desc: '开发、测试、生产环境完全一致,告别"在我机器上能跑"'
},
{ icon: '🚀', title: '快速部署', desc: '秒级启动,镜像分发,滚动更新无停机' },
{
icon: '📊',
title: '资源隔离',
desc: 'CPU/内存限制,互不干扰,一台机器跑多个应用'
},
{ icon: '🔄', title: '版本管理', desc: '镜像版本化,随时回滚,灰度发布' }
]
</script>
<style scoped>
.container-docker-demo {
border: 1px solid var(--vp-c-divider);
border-radius: 6px;
background: var(--vp-c-bg-soft);
padding: 0.75rem;
margin: 0.5rem 0;
}
.demo-header {
display: flex;
align-items: center;
gap: 0.4rem;
margin-bottom: 0.5rem;
padding-bottom: 0.4rem;
border-bottom: 1px solid var(--vp-c-divider);
}
.demo-header .icon {
font-size: 1rem;
}
.demo-header .title {
font-weight: bold;
font-size: 0.9rem;
color: var(--vp-c-text-1);
}
.demo-header .subtitle {
color: var(--vp-c-text-2);
font-size: 0.75rem;
margin-left: 0.4rem;
}
.docker-visualization {
display: flex;
gap: 0.5rem;
margin-bottom: 0.75rem;
align-items: stretch;
}
.layer {
flex: 1;
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
padding: 0.5rem;
cursor: pointer;
transition: all 0.3s;
}
.layer:hover,
.layer.active {
border-color: var(--vp-c-brand);
}
.layer h5 {
margin: 0 0 0.5rem 0;
text-align: center;
font-size: 0.75rem;
color: var(--vp-c-text-1);
}
.server-stack,
.docker-stack {
display: flex;
flex-direction: column;
gap: 0.25rem;
}
.layer-item {
padding: 0.3rem;
border-radius: 3px;
text-align: center;
font-size: 0.65rem;
}
.layer-item.app {
background: rgba(102, 126, 234, 0.2);
color: var(--vp-c-brand);
font-weight: 600;
}
.layer-item.deps {
background: var(--vp-c-bg-soft);
border: 1px dashed var(--vp-c-divider);
}
.layer-item.os,
.layer-item.hardware {
background: var(--vp-c-bg-soft);
color: var(--vp-c-text-2);
}
.layer-item.conflict {
background: rgba(239, 68, 68, 0.2);
color: var(--vp-c-danger);
font-weight: 600;
animation: pulse 1s infinite;
}
@keyframes pulse {
0%,
100% {
opacity: 1;
}
50% {
opacity: 0.5;
}
}
.containers {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 0.25rem;
}
.container-box {
background: rgba(102, 126, 234, 0.1);
border: 1px solid rgba(102, 126, 234, 0.3);
border-radius: 4px;
padding: 0.25rem;
text-align: center;
}
.container-app {
font-weight: 600;
font-size: 0.65rem;
color: var(--vp-c-brand);
margin-bottom: 0.1rem;
}
.container-deps {
font-size: 0.55rem;
color: var(--vp-c-text-2);
}
.docker-engine {
padding: 0.3rem;
background: rgba(16, 185, 129, 0.1);
border: 1px solid rgba(16, 185, 129, 0.3);
border-radius: 3px;
text-align: center;
font-size: 0.65rem;
font-weight: 600;
color: #059669;
}
.host-os,
.hardware {
padding: 0.3rem;
background: var(--vp-c-bg-soft);
border-radius: 3px;
text-align: center;
font-size: 0.65rem;
color: var(--vp-c-text-2);
}
.vs-divider {
display: flex;
align-items: center;
font-weight: 700;
color: var(--vp-c-text-3);
font-size: 0.75rem;
}
.benefits-grid {
display: grid;
grid-template-columns: repeat(4, 1fr);
gap: 0.4rem;
}
.benefit-card {
background: var(--vp-c-bg);
border: 1px solid var(--vp-c-divider);
border-radius: 4px;
padding: 0.5rem;
text-align: center;
transition: all 0.2s;
}
.benefit-card:hover {
border-color: var(--vp-c-brand);
transform: translateY(-1px);
}
.benefit-icon {
font-size: 1.25rem;
margin-bottom: 0.25rem;
}
.benefit-title {
font-weight: 600;
font-size: 0.75rem;
color: var(--vp-c-text-1);
margin-bottom: 0.1rem;
}
.benefit-desc {
font-size: 0.65rem;
color: var(--vp-c-text-2);
line-height: 1.3;
}
@media (max-width: 768px) {
.docker-visualization {
flex-direction: column;
}
.vs-divider {
justify-content: center;
padding: 0.25rem 0;
}
.benefits-grid {
grid-template-columns: repeat(2, 1fr);
}
}
.info-box {
background: var(--vp-c-bg-alt);
padding: 0.5rem;
border-radius: 4px;
font-size: 0.75rem;
color: var(--vp-c-text-2);
margin-top: 0.5rem;
display: flex;
gap: 0.2rem;
}
.info-box .icon {
flex-shrink: 0;
}
.info-box strong {
color: var(--vp-c-text-1);
}
</style>