一文掌握 ECharts:从入门到精通

133 阅读6分钟

一文掌握 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'
    }]
};

最佳实践

  1. 按需引入:只引入需要的图表类型和组件
  2. 响应式设计:监听窗口变化自动调整图表大小
  3. 错误处理:添加适当的加载状态和错误提示
  4. 性能监控:大数据量时使用分页或虚拟滚动
  5. 无障碍访问:为视障用户提供适当的描述

总结

ECharts 是一个功能强大、灵活性高的数据可视化库。通过本文的学习,你应该能够:

  • ✅ 掌握 ECharts 的基本使用方法
  • ✅ 理解常用图表类型的配置
  • ✅ 实现数据的动态更新和交互
  • ✅ 进行性能优化和主题定制
  • ✅ 解决常见的开发问题

ECharts 的官方文档非常完善,建议在实际开发中多查阅文档,结合具体业务需求选择最合适的可视化方案。