为了账号安全,请及时绑定邮箱和手机立即绑定

Web可视化开发入门教程

概述

Web可视化开发是一种将数据、信息或结构以图形化、图表化的方式展示在网页上的技术。通过可视化的方式,用户可以更直观地理解复杂的数据和信息。Web可视化开发不仅能够帮助用户理解数据背后的故事,还能增强用户体验,为用户提供了更丰富、更便捷的信息展示方式。可视化在现代Web开发中扮演着重要角色,广泛应用于数据展示、报告生成、仪表板设计、地图绘制、交互设计、游戏开发等多个领域。

Web可视化开发简介

Web可视化开发是一种将数据、信息或结构以图形化、图表化的方式展示在网页上的技术。通过可视化的方式,用户可以更直观地理解复杂的数据和信息。Web可视化开发不仅能够帮助用户理解数据背后的故事,还能增强用户体验,为用户提供了更丰富、更便捷的信息展示方式。可视化在现代Web开发中扮演着重要角色,广泛应用于数据展示、报告生成、仪表板设计、地图绘制、交互设计、游戏开发等多个领域。

Web可视化开发的优势和应用场景

优势

  1. 直观易懂:将复杂的数据转换为图表、图像等形式,使用户更容易理解数据背后的意义。
  2. 增强交互性:通过动态图表、按钮、滑块等交互元素,用户可以与数据进行实时互动,获取更深入的信息。
  3. 提升用户体验:通过美观的设计和流畅的交互,提升用户在浏览网页时的体验感。
  4. 数据洞察:可视化可以帮助用户发现数据中的模式和趋势,为决策提供依据。
  5. 跨平台可用性:可视化应用可以在多种设备上运行,包括桌面、移动设备等。

应用场景

  1. 数据报告和分析:如销售报告、市场分析等,通过图表展示关键指标的变化趋势。
  2. 仪表板和监控系统:实时监控服务器性能、网站流量等,通过仪表盘展示关键数据。
  3. 地图和位置服务:如地图标记、路线规划等,通过可视化地图展示地理位置信息。
  4. 游戏和模拟:通过可视化技术实现游戏界面、模拟实验等。
  5. 教育和培训:如在线课程、教学工具等,通过可视化图表帮助学生理解复杂的概念和数据。
必要的工具和环境搭建

Web可视化开发需要一个完整的开发环境来支持代码的编写、调试和运行。在搭建开发环境时,需要安装必要的软件工具,并确保环境配置正确。

开发环境的搭建

安装Node.js和npm

Node.js是一个基于Chrome V8引擎的JavaScript运行环境,而npm是Node.js的包管理器,用于管理项目依赖和执行脚本。安装步骤如下:

  1. 访问Node.js官网(https://nodejs.org/)下载最新版本的Node.js,建议选择LTS版本
  2. 安装完成后,打开命令行工具(如Windows的CMD或macOS/Linux的终端),运行以下命令检查安装是否成功:
    node -v
    npm -v

    如果命令返回Node.js和npm的版本号,说明安装成功。

安装VS Code

VS Code是一款免费且开源的代码编辑器,支持多种编程语言,包括JavaScript、HTML和CSS。步骤如下:

  1. 访问VS Code官网(https://code.visualstudio.com/)下载适用于你的操作系统的VS Code。
  2. 安装完成后,启动VS Code,安装必要的插件,如Live Server、Prettier等,来提高开发效率。

安装浏览器扩展

为了更好地进行Web开发,可以安装以下扩展:

  1. Live Server:用于实时预览HTML页面。
  2. EditorConfig for VS Code:帮助保持代码格式的一致性。
  3. ESLint:代码质量检查工具。
  4. JavaScript (ES6) Code Snippets:提供ES6代码片段以加速开发。

常用开发工具介绍

  1. 浏览器开发者工具:所有现代浏览器都内置了开发者工具,可以帮助开发者调试和优化网页。
    • Chrome开发者工具:提供了全面的调试功能,包括元素检查、网络请求、控制台输出等。
    • Firefox开发者工具:同样提供了强大的调试工具,适合不同的开发需求。
  2. Postman:一个强大的API测试工具,可以用来测试和调试Web应用的API接口。
  3. Figma:一个在线设计工具,可以帮助设计者创建和分享UI设计
  4. Git:版本控制系统,用于管理代码版本,确保团队协作的顺畅。
  5. Docker:容器化工具,可以简化开发环境的搭建和维护。
基础概念与术语解析

在开始Web可视化开发之前,理解一些基础概念和术语是非常重要的。这些概念是构建可视化应用的基础。

DOM和CSS的基本概念

DOM

DOM(文档对象模型)是一个跨平台的API,它提供了一种标准的方式来访问并操作HTML和XML文档。通过DOM,JavaScript可以动态地改变页面内容,响应用户输入,或者与服务器进行交互。
DOM将整个HTML文档组织成一颗树形结构,每个节点代表一个HTML元素。通过JavaScript,可以访问和修改这些节点,从而改变页面的结构和内容。

CSS

CSS(层叠样式表)是一种用于描述HTML文档样式的语言。通过CSS,可以定义页面元素的外观样式,如颜色、字体、布局等。
CSS的主要功能包括:

  1. 选择器:用于选择HTML元素,如diva.class等。
  2. 属性和值:定义元素的样式,如color: red;设置文本颜色为红色。
  3. 布局:通过positionmarginpadding等布局属性控制元素的位置和大小。

HTML标签基础

HTML(超文本标记语言)是构建Web页面的基础。以下是常用的HTML标签:

标题标签

<h1>一级标题</h1>
<h2>二级标题</h2>
<h3>三级标题</h3>

段落标签

<p>这是一个段落。</p>

列表标签

<ul>
  <li>项目1</li>
  <li>项目2</li>
</ul>

<ol>
  <li>步骤1</li>
  <li>步骤2</li>
</ol>

链接标签

<a href="https://www.example.com">链接到示例网站</a>

图像标签

<img class="lazyload" src="" data-original="path/to/image.jpg" alt="描述图片的替代文本">

表格标签

<table>
  <tr>
  <th>列1</th>
  <th>列2</th>
</tr>
<tr>
  <td>数据1</td>
  <td>数据2</td>
</tr>
</table>

表单标签

<form>
  <label for="name">姓名:</label>
<input type="text" id="name" name="name">
<br>
<button type="submit">提交</button>
</form>
第一个Web可视化项目实战

在本节中,我们将创建一个简单的Web可视化应用。这个应用将展示一个简单的柱状图,展示不同类别数据的大小。

创建简单的Web可视化应用

使用D3.js创建柱状图

D3.js是一个强大的JavaScript库,用于数据驱动的文档。它提供了丰富的功能来创建各种图表和可视化应用。

  1. 引入D3.js库
    首先需要引入D3.js库。可以通过CDN引入,也可以下载到本地。

    <!DOCTYPE html>
    <html>
    <head>
       <title>简单柱状图</title>
       <script class="lazyload" src="" data-original="https://d3js.org/d3.v6.min.js"></script>
    </head>
    <body>
    <div id="chart"></div>
    <script>
       // JavaScript代码将放在这里
    </script>
    </body>
    </html>
  2. 数据准备
    准备要可视化的数据。例如,数据可以是一个简单的数组,其中每个元素代表一个类别和它的值。

    const data = [
       {category: "A", value: 10},
       {category: "B", value: 20},
       {category: "C", value: 30},
       {category: "D", value: 40},
       {category: "E", value: 50},
    ];
  3. 创建SVG容器
    使用D3.js选择和修改DOM元素,创建一个SVG容器来容纳柱状图。

    const svg = d3.select("#chart")
       .append("svg")
       .attr("width", 500)
       .attr("height", 300);
  4. 创建柱状图
    为每个数据元素创建一个矩形,并设置其高度和位置。

    const x = d3.scaleBand()
       .domain(data.map(d => d.category))
       .range([0, 500])
       .padding(0.1);
    
    const y = d3.scaleLinear()
       .domain([0, d3.max(data, d => d.value)])
       .range([300, 0]);
    
    svg.selectAll("rect")
       .data(data)
       .enter()
       .append("rect")
       .attr("x", d => x(d.category))
       .attr("y", d => y(d.value))
       .attr("width", x.bandwidth())
       .attr("height", d => 300 - y(d.value))
       .attr("fill", "steelblue");
  5. 完成代码
    将所有代码整合在一起,形成一个完整的HTML文件。

    <!DOCTYPE html>
    <html>
    <head>
       <title>简单柱状图</title>
       <script class="lazyload" src="" data-original="https://d3js.org/d3.v6.min.js"></script>
    </head>
    <body>
    <div id="chart"></div>
    <script>
       const data = [
           {category: "A", value: 10},
           {category: "B", value: 20},
           {category: "C", value: 30},
           {category: "D", value: 40},
           {category: "E", value: 50},
       ];
    
       const svg = d3.select("#chart")
           .append("svg")
           .attr("width", 500)
           .attr("height", 300);
    
       const x = d3.scaleBand()
           .domain(data.map(d => d.category))
           .range([0, 500])
           .padding(0.1);
    
       const y = d3.scaleLinear()
           .domain([0, d3.max(data, d => d.value)])
           .range([300, 0]);
    
       svg.selectAll("rect")
           .data(data)
           .enter()
           .append("rect")
           .attr("x", d => x(d.category))
           .attr("y", d => y(d.value))
           .attr("width", x.bandwidth())
           .attr("height", d => 300 - y(d.value))
           .attr("fill", "steelblue");
    </script>
    </body>
    </html>

用户交互与响应式设计

用户交互是指用户与可视化应用之间的交互,包括点击、拖拽、滑动等操作。响应式设计则是指应用能够根据不同的设备和屏幕尺寸进行自适应调整,提供一致的用户体验。

用户交互

用户交互可以通过事件处理来实现。例如,通过监听鼠标点击事件,当用户点击某个图表时,显示相应的详细信息。还可以通过拖拽事件实现图表的拖拽功能,使用户能够更直观地操作图表。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>用户交互示例</title>
</head>
<body>
    <div id="chart"></div>
    <script>
        const svg = d3.select("#chart")
            .append("svg")
            .attr("width", 500)
            .attr("height", 300);

        const data = [
            {category: "A", value: 10},
            {category: "B", value: 20},
            {category: "C", value: 30},
            {category: "D", value: 40},
            {category: "E", value: 50},
        ];

        const x = d3.scaleBand()
            .domain(data.map(d => d.category))
            .range([0, 500])
            .padding(0.1);

        const y = d3.scaleLinear()
            .domain([0, d3.max(data, d => d.value)])
            .range([300, 0]);

        const rects = svg.selectAll("rect")
            .data(data)
            .enter()
            .append("rect")
            .attr("x", d => x(d.category))
            .attr("y", d => y(d.value))
            .attr("width", x.bandwidth())
            .attr("height", d => 300 - y(d.value))
            .attr("fill", "steelblue")
            .on("click", (event, d) => {
                console.log(`点击了${d.category},值为${d.value}`);
            });
    </script>
</body>
</html>
响应式设计

响应式设计可以通过CSS媒体查询实现。媒体查询可以根据不同的屏幕尺寸和设备类型应用不同的样式。例如,对于小屏幕设备,可以缩小图表的大小,简化布局,以适应较小的屏幕尺寸。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>响应式设计示例</title>
    <style>
        @media screen and (max-width: 600px) {
            #chart {
                width: 100%;
                height: 300px;
            }
        }
    </style>
</head>
<body>
    <div id="chart"></div>
    <script>
        const svg = d3.select("#chart")
            .append("svg")
            .attr("width", 500)
            .attr("height", 300);

        const data = [
            {category: "A", value: 10},
            {category: "B", value: 20},
            {category: "C", value: 30},
            {category: "D", value: 40},
            {category: "E", value: 50},
        ];

        const x = d3.scaleBand()
            .domain(data.map(d => d.category))
            .range([0, 500])
            .padding(0.1);

        const y = d3.scaleLinear()
            .domain([0, d3.max(data, d => d.value)])
            .range([300, 0]);

        const rects = svg.selectAll("rect")
            .data(data)
            .enter()
            .append("rect")
            .attr("x", d => x(d.category))
            .attr("y", d => y(d.value))
            .attr("width", x.bandwidth())
            .attr("height", d => 300 - y(d.value))
            .attr("fill", "steelblue");
    </script>
</body>
</html>
常见问题解答

问题:D3.js中的数据绑定是如何工作的?

在D3.js中,数据绑定是通过将数据与DOM元素进行关联来实现的。通过data()函数将数据绑定到选择的元素,然后使用enter()update()exit()方法来分别处理新数据、已更新的数据和要删除的数据。具体步骤如下:

  1. 选择元素:使用selectAll()选择一组元素。
  2. 绑定数据:使用data()函数将数据绑定到选择的元素。
  3. 创建新元素:使用enter().append()为每个新数据项创建新的DOM元素。
  4. 更新元素:使用update选择器更新现有元素的属性。
  5. 删除旧元素:使用exit().remove()移除不再需要的旧元素。

问题:如何优化Web可视化应用的性能?

  1. 减少DOM操作:DOM操作是昂贵的,尽量减少直接操作DOM的次数。
  2. 延迟加载:对于大型数据集或复杂图形,可以考虑延迟加载,只加载显示区域的数据。
  3. 使用事件委托:通过事件委托,可以减少事件处理器的数量,提高性能。
  4. 选择正确的数据结构:优化数据结构可以显著提高数据处理速度。
  5. 使用缓存:对于频繁访问的数据,可以使用缓存来减少重复计算。
  6. 优化CSS和JavaScript:确保CSS样式和JavaScript代码的简洁高效,避免不必要的嵌套和重复。
常见Web可视化库介绍

Web可视化库提供了丰富的功能来创建各种图表和可视化应用。以下是几个常用的Web可视化库:

D3.js简介

D3.js是一个基于SVG、HTML和CSS的JavaScript库,用于创建动态的数据驱动可视化。它提供了强大的数据操作和DOM操作功能,支持各种图表类型,如柱状图、折线图、散点图等。D3.js的灵活性使得它适用于各种复杂的动态数据可视化需求。

D3.js的特点包括:

  • 丰富的数据操作功能:可以通过多种方法处理和操作数据。
  • 强大的DOM操作功能:提供了强大的DOM操作能力,可以动态地改变页面结构。
  • 多种图表类型:支持柱状图、折线图、散点图等多种图表类型。
  • 丰富的事件处理:支持丰富的事件处理功能,可以实现动态交互。
  • 强大的布局功能:提供了丰富的布局算法,可以创建各种复杂的数据可视化。

示例代码:

const data = [
    {category: "A", value: 10},
    {category: "B", value: 20},
    {category: "C", value: 30},
    {category: "D", value: 40},
    {category: "E", value: 50},
];

const svg = d3.select("#chart")
    .append("svg")
    .attr("width", 500)
    .attr("height", 300);

const x = d3.scaleBand()
    .domain(data.map(d => d.category))
    .range([0, 500])
    .padding(0.1);

const y = d3.scaleLinear()
    .domain([0, d3.max(data, d => d.value)])
    .range([300, 0]);

svg.selectAll("rect")
    .data(data)
    .enter()
    .append("rect")
    .attr("x", d => x(d.category))
    .attr("y", d => y(d.value))
    .attr("width", x.bandwidth())
    .attr("height", d => 300 - y(d.value))
    .attr("fill", "steelblue");

ECharts简介

ECharts是由百度开发的一个强大的可视化库,支持多种图表类型,如折线图、柱状图、饼图、散点图、地图等。ECharts的特点是提供了丰富的配置选项,可以满足各种复杂的可视化需求。

ECharts的特点包括:

  • 丰富多样的图表类型:支持多种图表类型,如折线图、柱状图、饼图、散点图、地图等。
  • 强大的配置选项:可以通过配置选项来控制图表的各个方面,包括颜色、字体、样式等。
  • 良好的浏览器兼容性:支持各种现代浏览器。
  • 丰富的事件处理:支持多种事件处理,如鼠标点击、悬停等。
  • 动态效果:提供了多种动态效果,如过渡动画、缩放等。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <script class="lazyload" src="" data-original="https://cdn.jsdelivr.net/npm/echarts/dist/echarts.min.js"></script>
</head>
<body>
    <div id="main" style="width: 600px;height:400px;"></div>
    <script>
        var chartDom = document.getElementById('main');
        var myChart = echarts.init(chartDom);
        var option;

        option = {
            title: {
                text: '简单柱状图'
            },
            tooltip: {},
            xAxis: {
                data: ["A", "B", "C", "D", "E"]
            },
            yAxis: {},
            series: [{
                name: '数据',
                type: 'bar',
                data: [10, 20, 30, 40, 50]
            }]
        };

        option && myChart.setOption(option);
    </script>
</body>
</html>

Three.js简介

Three.js是一个强大的WebGL库,用于创建3D图形和动画。Three.js支持多种3D模型格式,可以用于创建复杂的3D场景。它提供了丰富的API,可以轻松地创建和操作3D对象、光源、摄像机等。

Three.js的特点包括:

  • 丰富的3D对象:支持多种3D对象,如几何体、纹理、光源等。
  • 强大的动画功能:提供了丰富的动画功能,可以创建复杂的3D动画。
  • 支持多种3D模型格式:支持导入多种3D模型格式,如OBJ、FBX等。
  • 强大的材质和光源:提供了多种材质和光源选项,可以创建逼真的3D场景。
  • 支持多种渲染器:支持多种渲染器,如WebGLRenderer、CanvasRenderer等。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>Three.js入门示例</title>
    <script class="lazyload" src="" data-original="https://cdn.jsdelivr.net/npm/three@0.117.0/build/three.min.js"></script>
</head>
<body>
    <script>
        var scene = new THREE.Scene();
        var camera = new THREE.PerspectiveCamera(75, window.innerWidth / window.innerHeight, 0.1, 1000);

        var renderer = new THREE.WebGLRenderer();
        renderer.setSize(window.innerWidth, window.innerHeight);
        document.body.appendChild(renderer.domElement);

        var geometry = new THREE.BoxGeometry();
        var material = new THREE.MeshBasicMaterial({color: 0x00ff00});
        var cube = new THREE.Mesh(geometry, material);
        scene.add(cube);

        camera.position.z = 5;

        function animate() {
            requestAnimationFrame(animate);

            cube.rotation.x += 0.01;
            cube.rotation.y += 0.01;

            renderer.render(scene, camera);
        }

        animate();
    </script>
</body>
</html>
Web可视化开发的进阶方向

在掌握基本的Web可视化开发技术后,可以通过以下方向进一步提升自己的技能。

动态数据绑定与实时更新

动态数据绑定是指将数据源与可视化对象动态绑定,当数据源发生变化时,可视化对象会自动更新。实时更新则是指在数据发生变化时,可视化应用能够实时反映这些变化。

动态数据绑定

动态数据绑定可以通过多种方式实现,常见的方法包括使用WebSocket连接后台数据源、监听数据变化事件等。例如,使用WebSocket连接后台数据源,当数据发生变化时,通过WebSocket发送消息,前端接收到消息后更新可视化对象。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>动态数据绑定示例</title>
</head>
<body>
    <div id="chart"></div>
    <script>
        const data = [
            {category: "A", value: 10},
            {category: "B", value: 20},
            {category: "C", value: 30},
            {category: "D", value: 40},
            {category: "E", value: 50},
        ];

        const svg = d3.select("#chart")
            .append("svg")
            .attr("width", 500)
            .attr("height", 300);

        const x = d3.scaleBand()
            .domain(data.map(d => d.category))
            .range([0, 500])
            .padding(0.1);

        const y = d3.scaleLinear()
            .domain([0, d3.max(data, d => d.value)])
            .range([300, 0]);

        const rects = svg.selectAll("rect")
            .data(data)
            .enter()
            .append("rect")
            .attr("x", d => x(d.category))
            .attr("y", d => y(d.value))
            .attr("width", x.bandwidth())
            .attr("height", d => 300 - y(d.value))
            .attr("fill", "steelblue");

        // 模拟数据变化
        setTimeout(() => {
            data[0].value = 15;
            rects.data(data).transition().duration(500)
                .attr("y", d => y(d.value))
                .attr("height", d => 300 - y(d.value));
        }, 2000);
    </script>
</body>
</html>

实时更新

实时更新可以使用WebSocket、轮询等技术实现。WebSocket是一种双向通信协议,可以在服务器和客户端之间建立持久连接,服务器可以主动推送数据给客户端。轮询则是定期向服务器请求数据,获取最新的数据并更新可视化对象。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>WebSocket实时更新示例</title>
</head>
<body>
    <div id="chart"></div>
    <script>
        const socket = new WebSocket("ws://localhost:8080");

        socket.onmessage = function(event) {
            const data = JSON.parse(event.data);
            updateChart(data);
        };

        const svg = d3.select("#chart")
            .append("svg")
            .attr("width", 500)
            .attr("height", 300);

        const x = d3.scaleBand()
            .domain(["A", "B", "C", "D", "E"])
            .range([0, 500])
            .padding(0.1);

        const y = d3.scaleLinear()
            .domain([0, 100])
            .range([300, 0]);

        const rects = svg.selectAll("rect")
            .data([10, 20, 30, 40, 50])
            .enter()
            .append("rect")
            .attr("x", (d, i) => x(i))
            .attr("y", d => y(d))
            .attr("width", x.bandwidth())
            .attr("height", d => 300 - y(d))
            .attr("fill", "steelblue");

        function updateChart(data) {
            rects.data(data)
                .attr("y", d => y(d))
                .attr("height", d => 300 - y(d));
        }
    </script>
</body>
</html>

用户交互与响应式设计

用户交互是指用户与可视化应用之间的交互,包括点击、拖拽、滑动等操作。响应式设计则是指应用能够根据不同的设备和屏幕尺寸进行自适应调整,提供一致的用户体验。

用户交互

用户交互可以通过事件处理来实现。例如,通过监听鼠标点击事件,当用户点击某个图表时,显示相应的详细信息。还可以通过拖拽事件实现图表的拖拽功能,使用户能够更直观地操作图表。

响应式设计

响应式设计可以通过CSS媒体查询实现。媒体查询可以根据不同的屏幕尺寸和设备类型应用不同的样式。例如,对于小屏幕设备,可以缩小图表的大小,简化布局,以适应较小的屏幕尺寸。

示例代码:

<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8">
    <title>响应式设计示例</title>
    <style>
        @media screen and (max-width: 600px) {
            #chart {
                width: 100%;
                height: 300px;
            }
        }
    </style>
</head>
<body>
    <div id="chart"></div>
    <script>
        const svg = d3.select("#chart")
            .append("svg")
            .attr("width", 500)
            .attr("height", 300);

        const data = [
            {category: "A", value: 10},
            {category: "B", value: 20},
            {category: "C", value: 30},
            {category: "D", value: 40},
            {category: "E", value: 50},
        ];

        const x = d3.scaleBand()
            .domain(data.map(d => d.category))
            .range([0, 500])
            .padding(0.1);

        const y = d3.scaleLinear()
            .domain([0, d3.max(data, d => d.value)])
            .range([300, 0]);

        const rects = svg.selectAll("rect")
            .data(data)
            .enter()
            .append("rect")
            .attr("x", d => x(d.category))
            .attr("y", d => y(d.value))
            .attr("width", x.bandwidth())
            .attr("height", d => 300 - y(d.value))
            .attr("fill", "steelblue");
    </script>
</body>
</html>

通过动态数据绑定和实时更新,可以创建更动态、更实时的可视化应用。通过用户交互和响应式设计,可以提升用户的体验,使可视化应用更加直观和易用。

总结

Web可视化开发是一种将数据以图形化的方式展示在网页上的技术,它可以帮助用户更好地理解和分析数据。通过本教程的学习,你将掌握如何搭建开发环境、理解基础概念、创建简单的可视化应用,并了解几种常用的可视化库。希望你能够通过这些知识,开发出更多有趣和实用的可视化应用。

点击查看更多内容
TA 点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
  • 推荐
  • 评论
  • 收藏
  • 共同学习,写下你的评论
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消