一文掌握 ECharts:从入门到精通
什么是 ECharts?
ECharts(Enterprise Charts)是百度开源的一个基于 JavaScript 的数据可视化图表库,提供了直观、生动、可交互、可高度个性化定制的数据可视化图表。它兼容当前绝大部分浏览器,底层依赖轻量级的矢量图形库 ZRender。
核心特性
- 丰富的图表类型:支持 30+ 种图表类型
- 多种数据格式:支持键值对、二维表、TypedArray 等格式
- 流数据支持:实时数据更新和动画效果
- 视觉映射:将数据映射到视觉元素(颜色、大小等)
- 动态数据:支持数据动态更新和交互
- 无障碍访问:支持盲人使用的无障碍功能
快速开始
1. 安装 ECharts
CDN 引入:
<script src="https://cdn.jsdelivr.net/npm/echarts@5.4.3/dist/echarts.min.js"></script>
NPM 安装:
npm install echarts
import * as echarts from 'echarts';
2. 基本使用步骤
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>ECharts 示例</title>
<script src="https://cdn.jsdelivr.net/npm/echarts@5.4.3/dist/echarts.min.js"></script>
</head>
<body>
<div id="main" style="width: 600px;height:400px;"></div>
<script>
// 初始化图表实例
const chartDom = document.getElementById('main');
const myChart = echarts.init(chartDom);
// 配置项
const option = {
title: {
text: 'ECharts 入门示例'
},
tooltip: {},
legend: {
data: ['销量']
},
xAxis: {
data: ['衬衫', '羊毛衫', '雪纺衫', '裤子', '高跟鞋', '袜子']
},
yAxis: {},
series: [{
name: '销量',
type: 'bar',
data: [5, 20, 36, 10, 10, 20]
}]
};
// 使用配置项显示图表
myChart.setOption(option);
// 响应窗口大小变化
window.addEventListener('resize', function() {
myChart.resize();
});
</script>
</body>
</html>
常用图表类型详解
1. 折线图 (Line Chart)
const lineOption = {
title: {
text: '月度销售额趋势'
},
tooltip: {
trigger: 'axis'
},
legend: {
data: ['产品A', '产品B']
},
grid: {
left: '3%',
right: '4%',
bottom: '3%',
containLabel: true
},
xAxis: {
type: 'category',
boundaryGap: false,
data: ['1月', '2月', '3月', '4月', '5月', '6月']
},
yAxis: {
type: 'value'
},
series: [
{
name: '产品A',
type: 'line',
data: [120, 132, 101, 134, 90, 230],
smooth: true, // 平滑曲线
areaStyle: {} // 区域填充
},
{
name: '产品B',
type: 'line',
data: [220, 182, 191, 234, 290, 330],
smooth: true,
areaStyle: {}
}
]
};
2. 柱状图 (Bar Chart)
const barOption = {
title: {
text: '各部门人员分布',
subtext: '2023年数据'
},
tooltip: {
trigger: 'axis',
axisPointer: {
type: 'shadow'
}
},
legend: {
data: ['男性', '女性']
},
grid: {
left: '3%',
right: '4%',
bottom: '3%',
containLabel: true
},
xAxis: {
type: 'value'
},
yAxis: {
type: 'category',
data: ['技术部', '市场部', '人事部', '财务部', '运营部']
},
series: [
{
name: '男性',
type: 'bar',
data: [50, 30, 20, 15, 25],
itemStyle: {
color: '#5470c6'
}
},
{
name: '女性',
type: 'bar',
data: [20, 35, 25, 20, 30],
itemStyle: {
color: '#ee6666'
}
}
]
};
3. 饼图 (Pie Chart)
const pieOption = {
title: {
text: '用户访问来源',
left: 'center'
},
tooltip: {
trigger: 'item',
formatter: '{a} <br/>{b}: {c} ({d}%)'
},
legend: {
orient: 'vertical',
left: 'left'
},
series: [
{
name: '访问来源',
type: 'pie',
radius: '50%',
data: [
{ value: 1048, name: '搜索引擎' },
{ value: 735, name: '直接访问' },
{ value: 580, name: '邮件营销' },
{ value: 484, name: '联盟广告' },
{ value: 300, name: '视频广告' }
],
emphasis: {
itemStyle: {
shadowBlur: 10,
shadowOffsetX: 0,
shadowColor: 'rgba(0, 0, 0, 0.5)'
}
}
}
]
};
4. 散点图 (Scatter Chart)
const scatterOption = {
title: {
text: '身高体重分布'
},
tooltip: {
trigger: 'axis',
axisPointer: {
type: 'cross'
}
},
xAxis: {
type: 'value',
name: '身高 (cm)'
},
yAxis: {
type: 'value',
name: '体重 (kg)'
},
series: [{
type: 'scatter',
data: [
[160, 50], [165, 55], [170, 60], [175, 65], [180, 70],
[185, 75], [162, 52], [168, 58], [172, 62], [178, 68]
],
symbolSize: 10,
itemStyle: {
color: '#ff6b6b'
}
}]
};
高级功能
1. 数据异步加载
// 模拟异步数据加载
function fetchData() {
return new Promise((resolve) => {
setTimeout(() => {
resolve({
categories: ['周一', '周二', '周三', '周四', '周五', '周六', '周日'],
data: [120, 200, 150, 80, 70, 110, 130]
});
}, 1000);
});
}
// 显示加载动画
myChart.showLoading();
fetchData().then(data => {
myChart.hideLoading();
myChart.setOption({
xAxis: {
type: 'category',
data: data.categories
},
yAxis: {
type: 'value'
},
series: [{
data: data.data,
type: 'line'
}]
});
});
2. 事件处理
// 图表点击事件
myChart.on('click', function(params) {
console.log('点击了:', params);
alert(`您点击了: ${params.name}, 值: ${params.value}`);
});
// 图例切换事件
myChart.on('legendselectchanged', function(params) {
console.log('图例选择变化:', params);
});
// 数据区域缩放
myChart.on('datazoom', function(params) {
console.log('数据区域缩放:', params);
});
3. 动态数据更新
let currentData = [5, 20, 36, 10, 10, 20];
let dataIndex = 0;
// 定时更新数据
setInterval(() => {
currentData[dataIndex % 6] = Math.round(Math.random() * 100);
dataIndex++;
myChart.setOption({
series: [{
data: currentData
}]
});
}, 2000);
4. 主题定制
// 使用内置主题
const chart = echarts.init(dom, 'dark'); // 使用暗色主题
// 自定义主题
const customTheme = {
color: ['#c23531','#2f4554','#61a0a8','#d48265','#91c7ae'],
backgroundColor: '#f5f5f5',
textStyle: {
fontFamily: 'Arial, sans-serif'
}
};
// 注册主题
echarts.registerTheme('myTheme', customTheme);
const chart = echarts.init(dom, 'myTheme');
实战案例
1. 销售仪表盘
const dashboardOption = {
tooltip: {
trigger: 'axis'
},
legend: {
data: ['线上销售', '线下销售', '总销售额']
},
grid: {
left: '3%',
right: '4%',
bottom: '3%',
containLabel: true
},
xAxis: {
type: 'category',
boundaryGap: false,
data: ['1月', '2月', '3月', '4月', '5月', '6月']
},
yAxis: [
{
type: 'value',
name: '销售额'
},
{
type: 'value',
name: '增长率'
}
],
series: [
{
name: '线上销售',
type: 'line',
data: [120, 132, 101, 134, 90, 230]
},
{
name: '线下销售',
type: 'line',
data: [220, 182, 191, 234, 290, 330]
},
{
name: '总销售额',
type: 'bar',
yAxisIndex: 1,
data: [340, 314, 292, 368, 380, 560]
}
]
};
2. 地图可视化
// 需要额外引入地图数据
// <script src="https://cdn.jsdelivr.net/npm/echarts@5.4.3/map/js/china.js"></script>
const mapOption = {
title: {
text: '全国销售分布',
left: 'center'
},
tooltip: {
trigger: 'item',
formatter: '{b}: {c}'
},
visualMap: {
min: 0,
max: 1000,
left: 'left',
top: 'bottom',
text: ['高', '低'],
calculable: true,
inRange: {
color: ['#e0f3ff', '#006edd']
}
},
series: [{
name: '销售额',
type: 'map',
map: 'china',
roam: true,
emphasis: {
label: {
show: true
}
},
data: [
{ name: '北京', value: 900 },
{ name: '上海', value: 800 },
{ name: '广东', value: 950 },
{ name: '浙江', value: 700 },
{ name: '江苏', value: 600 }
// ... 其他省份数据
]
}]
};
性能优化
1. 大数据量优化
// 使用大数据模式
const largeDataOption = {
dataset: {
source: largeDataArray // 使用 TypedArray 提高性能
},
series: {
type: 'scatter',
large: true, // 开启大数据模式
largeThreshold: 2000, // 数据量阈值
progressiveChunkMode: 'mod' // 分块渲染
}
};
2. 按需引入
// 核心模块
import * as echarts from 'echarts/core';
import { BarChart, LineChart, PieChart } from 'echarts/charts';
import {
TitleComponent,
TooltipComponent,
GridComponent,
LegendComponent
} from 'echarts/components';
import { CanvasRenderer } from 'echarts/renderers';
// 注册必要组件
echarts.use([
TitleComponent,
TooltipComponent,
GridComponent,
LegendComponent,
BarChart,
LineChart,
PieChart,
CanvasRenderer
]);
常见问题解决
1. 图表渲染问题
// 确保容器有尺寸
<div id="chart" style="width: 100%; height: 400px;"></div>
// 在合适的时机初始化
window.addEventListener('load', function() {
const chart = echarts.init(document.getElementById('chart'));
});
// 响应式处理
window.addEventListener('resize', function() {
chart.resize();
});
2. 数据格式处理
// 处理时间序列数据
const timeData = [
['2023-01-01', 100],
['2023-01-02', 200],
// ...
].map(item => [new Date(item[0]), item[1]]);
const option = {
xAxis: {
type: 'time'
},
series: [{
data: timeData,
type: 'line'
}]
};
最佳实践
- 按需引入:只引入需要的图表类型和组件
- 响应式设计:监听窗口变化自动调整图表大小
- 错误处理:添加适当的加载状态和错误提示
- 性能监控:大数据量时使用分页或虚拟滚动
- 无障碍访问:为视障用户提供适当的描述
总结
ECharts 是一个功能强大、灵活性高的数据可视化库。通过本文的学习,你应该能够:
- ✅ 掌握 ECharts 的基本使用方法
- ✅ 理解常用图表类型的配置
- ✅ 实现数据的动态更新和交互
- ✅ 进行性能优化和主题定制
- ✅ 解决常见的开发问题
ECharts 的官方文档非常完善,建议在实际开发中多查阅文档,结合具体业务需求选择最合适的可视化方案。