2026-01-18 12:21:49 +08:00
|
|
|
|
<template>
|
|
|
|
|
|
<div class="developer-efficiency-demo">
|
|
|
|
|
|
<div class="demo-header">
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<span class="icon">⏱️</span>
|
|
|
|
|
|
<span class="title">开发效率</span>
|
|
|
|
|
|
<span class="subtitle">不同语言完成相同任务的时间成本</span>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="intro-text">
|
|
|
|
|
|
想象你在<span class="highlight">装修房子</span>:有的装修队能快速完工但质量一般(Python、Ruby),有的慢工出细活(Rust、C++),有的速度和质量都不错(Go、Node.js)。
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div class="task-selector">
|
|
|
|
|
|
<label>选择任务:</label>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<select v-model="selectedTask">
|
|
|
|
|
|
<option value="rest">REST API</option>
|
|
|
|
|
|
<option value="web">Web 应用</option>
|
|
|
|
|
|
<option value="script">数据处理脚本</option>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</select>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<div class="efficiency-chart">
|
|
|
|
|
|
<div class="chart-header">
|
|
|
|
|
|
<span>开发时间(小时)</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<div class="bars">
|
|
|
|
|
|
<div
|
|
|
|
|
|
v-for="lang in sortedLanguages"
|
|
|
|
|
|
:key="lang.name"
|
|
|
|
|
|
class="bar-wrapper"
|
|
|
|
|
|
>
|
|
|
|
|
|
<div class="bar-label">{{ lang.name }}</div>
|
|
|
|
|
|
<div class="bar-track">
|
|
|
|
|
|
<div
|
|
|
|
|
|
class="bar-fill"
|
|
|
|
|
|
:style="{ width: (lang.time / maxTime * 100) + '%' }"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span class="bar-value">{{ lang.time }}h</span>
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
<div class="info-box">
|
|
|
|
|
|
<span class="icon">💡</span>
|
|
|
|
|
|
<strong>核心思想:</strong>初创公司选 Python/Ruby 快速验证想法,大厂选 Java/Go 平衡速度和质量。开发效率不只是写代码的速度,还包括调试、测试、维护的时间成本。
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
|
|
<script setup>
|
|
|
|
|
|
import { ref, computed } from 'vue'
|
|
|
|
|
|
|
|
|
|
|
|
const selectedTask = ref('rest')
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
const taskData = {
|
|
|
|
|
|
rest: [
|
|
|
|
|
|
{ name: 'Python', time: 4 },
|
|
|
|
|
|
{ name: 'Ruby', time: 3.5 },
|
|
|
|
|
|
{ name: 'Go', time: 5 },
|
|
|
|
|
|
{ name: 'Node.js', time: 4.5 },
|
|
|
|
|
|
{ name: 'Java', time: 8 },
|
|
|
|
|
|
{ name: 'Rust', time: 10 }
|
|
|
|
|
|
],
|
|
|
|
|
|
web: [
|
|
|
|
|
|
{ name: 'Ruby', time: 9 },
|
|
|
|
|
|
{ name: 'Python', time: 10 },
|
|
|
|
|
|
{ name: 'Node.js', time: 11 },
|
|
|
|
|
|
{ name: 'Go', time: 12 },
|
|
|
|
|
|
{ name: 'Java', time: 20 },
|
|
|
|
|
|
{ name: 'Rust', time: 25 }
|
|
|
|
|
|
],
|
|
|
|
|
|
script: [
|
|
|
|
|
|
{ name: 'Python', time: 1 },
|
|
|
|
|
|
{ name: 'Ruby', time: 1 },
|
|
|
|
|
|
{ name: 'Node.js', time: 1.5 },
|
|
|
|
|
|
{ name: 'Go', time: 2 },
|
|
|
|
|
|
{ name: 'Java', time: 4 },
|
|
|
|
|
|
{ name: 'Rust', time: 4 }
|
|
|
|
|
|
]
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const sortedLanguages = computed(() => {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
return [...taskData[selectedTask.value]].sort((a, b) => a.time - b.time)
|
2026-01-18 12:21:49 +08:00
|
|
|
|
})
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
const maxTime = computed(() => {
|
|
|
|
|
|
return Math.max(...taskData[selectedTask.value].map(l => l.time))
|
|
|
|
|
|
})
|
2026-01-18 12:21:49 +08:00
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
|
|
<style scoped>
|
|
|
|
|
|
.developer-efficiency-demo {
|
|
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
border-radius: 6px;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
background: var(--vp-c-bg-soft);
|
2026-02-14 20:23:34 +08:00
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
margin: 0.5rem 0;
|
|
|
|
|
|
|
|
|
|
|
|
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.demo-header .icon {
|
|
|
|
|
|
font-size: 1.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.demo-header .title {
|
|
|
|
|
|
font-weight: bold;
|
|
|
|
|
|
font-size: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.demo-header .subtitle {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-2);
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
margin-left: 0.5rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.intro-text {
|
|
|
|
|
|
font-size: 0.9rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
line-height: 1.6;
|
|
|
|
|
|
margin-bottom: 1rem;
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.intro-text .highlight {
|
|
|
|
|
|
color: var(--vp-c-brand-1);
|
|
|
|
|
|
font-weight: 500;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.task-selector {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 0.5rem;
|
|
|
|
|
|
margin-bottom: 1rem;
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.5rem 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.task-selector label {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-size: 0.85rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.task-selector select {
|
2026-02-13 22:10:03 +08:00
|
|
|
|
padding: 0.35rem 0.5rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border: 1px solid var(--vp-c-divider);
|
2026-02-13 22:10:03 +08:00
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-1);
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-size: 0.85rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.efficiency-chart {
|
|
|
|
|
|
background: var(--vp-c-bg);
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
|
|
|
|
|
margin-bottom: 1rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.chart-header {
|
|
|
|
|
|
margin-bottom: 0.75rem;
|
|
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
font-weight: 600;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.bars {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
flex-direction: column;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 0.5rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.bar-wrapper {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
gap: 0.5rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.bar-label {
|
|
|
|
|
|
min-width: 70px;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
font-size: 0.8rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
color: var(--vp-c-text-1);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.bar-track {
|
2026-01-18 12:21:49 +08:00
|
|
|
|
flex: 1;
|
|
|
|
|
|
height: 24px;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
background: var(--vp-c-bg-soft);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.bar-fill {
|
|
|
|
|
|
height: 100%;
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
align-items: center;
|
|
|
|
|
|
justify-content: flex-end;
|
|
|
|
|
|
padding-right: 0.5rem;
|
2026-02-13 22:10:03 +08:00
|
|
|
|
background: var(--vp-c-green-1);
|
2026-01-18 12:21:49 +08:00
|
|
|
|
transition: width 0.5s ease;
|
|
|
|
|
|
color: white;
|
|
|
|
|
|
font-weight: 600;
|
|
|
|
|
|
font-size: 0.75rem;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.info-box {
|
|
|
|
|
|
background: var(--vp-c-bg-alt);
|
|
|
|
|
|
padding: 0.75rem;
|
|
|
|
|
|
border-radius: 6px;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
font-size: 0.85rem;
|
|
|
|
|
|
color: var(--vp-c-text-2);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-02-13 22:10:03 +08:00
|
|
|
|
.info-box .icon {
|
|
|
|
|
|
margin-right: 0.25rem;
|
2026-01-18 12:21:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
</style>
|