e7b3fa8001
- 解锁 29 个归档组件 (Project Architecture, Rate Limiting, Search Engines, File Storage, Async Task Queues, Scheduled Tasks, Computer Fundamentals) - 修复 RateLimitAlgorithmDemo build 卡住问题 (移除末尾 reset() 调用) - 修复 RuntimeEnvironmentDemo eval 安全警告 - 添加 Vite build chunkSizeWarningLimit 配置 - 添加 Vue 组件开发规范文档 (VUE_COMPONENT_RULES.md)
98 lines
3.8 KiB
Vue
98 lines
3.8 KiB
Vue
<template>
|
|
<div class="lock-demo">
|
|
<div class="header">
|
|
<div class="title">分布式锁演示</div>
|
|
<div class="subtitle">多节点互斥访问共享资源</div>
|
|
</div>
|
|
<div class="controls">
|
|
<button @click="acquire" class="acquire-btn">获取锁</button>
|
|
<button @click="release" class="release-btn">释放锁</button>
|
|
</div>
|
|
<div class="nodes">
|
|
<div v-for="n in nodes" :key="n.id" :class="['node', { active: n.hasLock, waiting: n.waiting }]">
|
|
<div class="node-name">{{ n.name }}</div>
|
|
<div class="node-status">{{ n.hasLock ? '持有锁' : n.waiting ? '等待中' : '空闲' }}</div>
|
|
</div>
|
|
</div>
|
|
<div class="resource">
|
|
<div class="resource-label">共享资源</div>
|
|
<div :class="['resource-status', { locked: locked }]">{{ locked ? '🔒 已占用' : '✅ 可访问' }}</div>
|
|
</div>
|
|
<div class="log-area">
|
|
<div v-for="(log, i) in logs" :key="i" class="log-item">{{ log }}</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup>
|
|
import { ref, onUnmounted } from 'vue'
|
|
|
|
const nodes = ref([
|
|
{ id: 1, name: 'Node A', hasLock: false, waiting: false },
|
|
{ id: 2, name: 'Node B', hasLock: false, waiting: false },
|
|
{ id: 3, name: 'Node C', hasLock: false, waiting: false },
|
|
])
|
|
const locked = ref(false)
|
|
const logs = ref([])
|
|
let timer = null
|
|
|
|
function acquire() {
|
|
const idleNode = nodes.value.find(n => !n.hasLock && !n.waiting)
|
|
if (!idleNode) return
|
|
|
|
if (locked.value) {
|
|
idleNode.waiting = true
|
|
logs.value.unshift(`${idleNode.name} 等待获取锁...`)
|
|
} else {
|
|
idleNode.hasLock = true
|
|
locked.value = true
|
|
logs.value.unshift(`${idleNode.name} 成功获取锁!`)
|
|
}
|
|
}
|
|
|
|
function release() {
|
|
const holder = nodes.value.find(n => n.hasLock)
|
|
if (holder) {
|
|
holder.hasLock = false
|
|
locked.value = false
|
|
logs.value.unshift(`${holder.name} 释放了锁`)
|
|
|
|
const waiter = nodes.value.find(n => n.waiting)
|
|
if (waiter) {
|
|
waiter.waiting = false
|
|
waiter.hasLock = true
|
|
locked.value = true
|
|
logs.value.unshift(`${waiter.name} 获取到锁`)
|
|
}
|
|
}
|
|
}
|
|
|
|
onUnmounted(() => {
|
|
if (timer) clearInterval(timer)
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.lock-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; }
|
|
.controls { display: flex; gap: 0.5rem; margin-bottom: 1rem; }
|
|
.acquire-btn, .release-btn { padding: 0.4rem 0.8rem; border-radius: 6px; border: 1px solid var(--vp-c-divider); background: var(--vp-c-bg); cursor: pointer; font-size: 0.85rem; }
|
|
.acquire-btn { background: var(--vp-c-brand); color: #fff; border-color: var(--vp-c-brand); }
|
|
.release-btn { background: #ef4444; color: #fff; border-color: #ef4444; }
|
|
.nodes { display: flex; gap: 1rem; margin-bottom: 1rem; }
|
|
.node { flex: 1; padding: 1rem; border-radius: 8px; background: var(--vp-c-bg); text-align: center; border: 2px solid var(--vp-c-divider); transition: all 0.3s; }
|
|
.node.active { border-color: #22c55e; background: #dcfce7; }
|
|
.node.waiting { border-color: #f59e0b; background: #fef3c7; }
|
|
.node-name { font-weight: 600; margin-bottom: 0.25rem; }
|
|
.node-status { font-size: 0.85rem; }
|
|
.resource { padding: 1rem; background: var(--vp-c-bg); border-radius: 8px; text-align: center; margin-bottom: 1rem; }
|
|
.resource-label { font-size: 0.9rem; color: var(--vp-c-text-2); margin-bottom: 0.5rem; }
|
|
.resource-status { font-weight: 600; font-size: 1.1rem; }
|
|
.resource-status.locked { color: #22c55e; }
|
|
.log-area { background: var(--vp-c-bg); border-radius: 8px; padding: 0.75rem; max-height: 100px; overflow-y: auto; }
|
|
.log-item { font-size: 0.8rem; padding: 0.25rem 0; border-bottom: 1px solid var(--vp-c-divider); }
|
|
.log-item:last-child { border-bottom: none; }
|
|
</style>
|