0

0

Dash Mantine组件动态表格:结合Dropdown实现交互式数据展示

聖光之護

聖光之護

发布时间:2025-10-22 13:22:19

|

937人浏览过

|

来源于php中文网

原创

Dash Mantine组件动态表格:结合Dropdown实现交互式数据展示

本教程详细介绍了如何在dash应用中利用`dash_mantine_components`库的`dmc.table`组件和`dash_core_components`的`dcc.dropdown`实现动态表格。通过spotify数据集的案例,我们将学习如何根据用户选择的流派和子流派,实时更新并展示top 10艺术家列表。核心在于正确配置回调函数的`output`为`children`属性,并返回符合`dmc.table`结构要求的`html.thead`和`html.tbody`元素列表。

引言:Dash动态表格的魅力

在构建交互式数据可视化应用时,动态表格是不可或缺的组成部分。它允许用户通过各种控件(如下拉菜单、滑块等)实时筛选和查看数据,从而提供更深入的洞察。Dash作为Python领域流行的Web应用框架,结合dash_mantine_components (dmc) 库,能够帮助开发者快速构建美观且功能强大的仪表板。本教程将专注于如何利用dmc.Table组件和dcc.Dropdown组件,实现一个根据用户选择动态更新数据的表格。

环境准备与数据加载

在开始之前,请确保您已安装所需的Python库:dash、pandas、dash_core_components、dash_html_components和dash_mantine_components。

pip install dash pandas dash-core-components dash-html-components dash-mantine-components

我们将使用一个Spotify歌曲数据集,目标是根据用户选择的“流派”和“子流派”,展示该类别下的Top 10艺术家及其歌曲数量。

import pandas as pd
from dash import Dash, dcc, html, Input, Output
import dash_mantine_components as dmc

# 假设您的数据文件名为 'spotify_songs.csv'
# 请替换为您的实际数据路径
try:
    data = pd.read_csv('spotify_songs.csv')
except FileNotFoundError:
    print("请确保 'spotify_songs.csv' 文件存在于当前目录或提供正确路径。")
    # 创建一个模拟数据框用于演示,如果文件不存在
    data = pd.DataFrame({
        'Artist': ['ArtistA', 'ArtistB', 'ArtistC', 'ArtistD', 'ArtistA', 'ArtistB', 'ArtistE', 'ArtistF', 'ArtistA', 'ArtistB'] * 20,
        'Genre': ['Pop', 'Pop', 'Rock', 'Pop', 'Rock', 'Jazz', 'Pop', 'Rock', 'Pop', 'Jazz'] * 20,
        'Subgenre': ['Dance Pop', 'Electro Pop', 'Alternative Rock', 'Synth Pop', 'Classic Rock', 'Smooth Jazz', 'Dance Pop', 'Hard Rock', 'Electro Pop', 'Bebop'] * 20
    })
    # 确保模拟数据足够大,以产生Top 10结果
    data = data.sample(n=300, replace=True).reset_index(drop=True)


app = Dash(__name__)

# 获取流派和子流派的唯一值,用于下拉菜单选项
genres = data['Genre'].unique().tolist()
subgenres = data['Subgenre'].unique().tolist()

Dash应用布局构建

Dash应用的布局由一系列HTML组件构成。我们将创建一个包含两个下拉菜单(用于选择流派和子流派)和一个dmc.Table组件(用于展示动态数据)的简单布局。dmc.Table组件在初始时将是空的,其内容将由回调函数动态填充。

app.layout = dmc.Container(
    [
        dmc.Title("Spotify Top 10 艺术家动态榜单", order=1),
        dmc.Space(h="md"),

        dmc.Grid(
            [
                dmc.Col(
                    dcc.Dropdown(
                        id="genre-dropdown",
                        options=[{"label": g, "value": g} for g in genres],
                        placeholder="选择流派",
                        value=genres[0] if genres else None, # 默认选中第一个流派
                        clearable=False,
                    ),
                    span=6,
                ),
                dmc.Col(
                    dcc.Dropdown(
                        id="subgenre-dropdown",
                        options=[{"label": sg, "value": sg} for sg in subgenres],
                        placeholder="选择子流派",
                        value=subgenres[0] if subgenres else None, # 默认选中第一个子流派
                        clearable=False,
                    ),
                    span=6,
                ),
            ],
            gutter="xl",
        ),
        dmc.Space(h="xl"),

        dmc.Card(
            children=[
                dmc.Text("Top 10 艺术家", size='lg', color='dimmed', weight=500, align='center'),
                # dmc.Table组件将通过回调函数更新其children属性
                dmc.Table(id='top_10_artists', striped=True, highlightOnHover=True, withBorder=True, withColumnBorders=True),
            ],
            withBorder=True,
            shadow='lg',
            radius='md',
        ),
    ],
    fluid=True,
)

关键点: dmc.Table(id='top_10_artists') 在布局中被定义,但没有直接提供数据。其内容将完全由回调函数通过更新其children属性来控制。

实现核心交互:回调函数

回调函数是Dash应用的核心,它负责响应用户输入并更新应用布局。在本例中,我们将创建一个回调函数,监听genre-dropdown和subgenre-dropdown的值变化,然后计算并返回更新后的dmc.Table内容。

网易人工智能
网易人工智能

网易数帆多媒体智能生产力平台

下载

回调函数签名解析

@app.callback(
    Output("top_10_artists", "children"), # 关键:更新dmc.Table的children属性
    Input("genre-dropdown", "value"),
    Input("subgenre-dropdown", "value")
)
def update_top_10_artists_table(selected_genre, selected_subgenre):
    # ... 函数体 ...
  • Output("top_10_artists", "children"): 这是最重要的部分。dmc.Table组件期望其内容(表头和表体)作为其children属性来渲染。因此,我们的回调函数必须返回一个包含html.Thead和html.Tbody元素的列表。尝试更新"table"属性(如原始问题中的尝试)会导致SchemaLengthValidationError,因为dmc.Table组件没有名为"table"的属性来接收这种结构。
  • Input("genre-dropdown", "value")Input("subgenre-dropdown", "value"): 当这两个下拉菜单的选中值发生变化时,回调函数将被触发。

数据处理与HTML元素构建

在回调函数内部,我们将执行以下步骤:

  1. 根据选定的流派和子流派筛选原始数据集。
  2. 计算每个艺术家的歌曲数量,并获取Top 10。
  3. 将这些数据转换为html.Thead(表头)和html.Tbody(表体)的HTML结构。

为了提高代码的健壮性和可维护性,我们将使用循环来生成表格的行,而不是硬编码每一行。

@app.callback(
    Output("top_10_artists", "children"),
    Input("genre-dropdown", "value"),
    Input("subgenre-dropdown", "value")
)
def update_top_10_artists_table(selected_genre, selected_subgenre):
    if not selected_genre or not selected_subgenre:
        return html.Div("请选择流派和子流派以查看数据。", style={'textAlign': 'center', 'marginTop': '20px'})

    # 复制数据以避免修改原始DataFrame
    df_filtered = data.copy()

    # 根据选择的流派和子流派进行筛选
    df_filtered = df_filtered[
        (df_filtered['Genre'] == selected_genre) &
        (df_filtered['Subgenre'] == selected_subgenre)
    ]

    # 计算艺术家歌曲数量并获取Top 10
    artists_counts = df_filtered['Artist'].value_counts().reset_index()
    artists_counts.columns = ['Artist', 'Count'] # 重命名列以便后续访问
    top_10_artists = artists_counts.head(10)

    # 如果没有数据,返回提示信息
    if top_10_artists.empty:
        return html.Div("当前流派和子流派下没有找到艺术家数据。", style={'textAlign': 'center', 'marginTop': '20px'})

    # 构建表头
    header = html.Thead(
        html.Tr(
            [
                html.Th('艺术家'),
                html.Th('歌曲数量')
            ]
        )
    )

    # 构建表体
    # 使用列表推导式动态生成表格行
    rows = []
    for index, row_data in top_10_artists.iterrows():
        rows.append(
            html.Tr([
                html.Td(row_data['Artist']),
                html.Td(row_data['Count'])
            ])
        )
    body = html.Tbody(rows)

    # 返回表头和表体组成的列表
    return [header, body]

注意事项:

  • 我们添加了对selected_genre和selected_subgenre是否为空的检查,以及对top_10_artists是否为空的检查,以提供更好的用户体验和错误处理。
  • 使用artists_counts.columns = ['Artist', 'Count']重命名了列,使得在构建html.Td时可以更清晰地通过名称访问数据。
  • 通过for循环和列表推导式生成html.Tr和html.Td,这比硬编码10行更加灵活和易于维护,尤其当结果数量不固定时。

完整示例代码

将所有部分整合,以下是完整的Dash应用代码:

import pandas as pd
from dash import Dash, dcc, html, Input, Output
import dash_mantine_components as dmc

# 假设您的数据文件名为 'spotify_songs.csv'
# 请替换为您的实际数据路径
try:
    data = pd.read_csv('spotify_songs.csv')
except FileNotFoundError:
    print("请确保 'spotify_songs.csv' 文件存在于当前目录或提供正确路径。")
    # 创建一个模拟数据框用于演示,如果文件不存在
    data = pd.DataFrame({
        'Artist': ['ArtistA', 'ArtistB', 'ArtistC', 'ArtistD', 'ArtistA', 'ArtistB', 'ArtistE', 'ArtistF', 'ArtistA', 'ArtistB'] * 20,
        'Genre': ['Pop', 'Pop', 'Rock', 'Pop', 'Rock', 'Jazz', 'Pop', 'Rock', 'Pop', 'Jazz'] * 20,
        'Subgenre': ['Dance Pop', 'Electro Pop', 'Alternative Rock', 'Synth Pop', 'Classic Rock', 'Smooth Jazz', 'Dance Pop', 'Hard Rock', 'Electro Pop', 'Bebop'] * 20
    })
    data = data.sample(n=300, replace=True).reset_index(drop=True)


app = Dash(__name__)

genres = data['Genre'].unique().tolist()
subgenres = data['Subgenre'].unique().tolist()

app.layout = dmc.Container(
    [
        dmc.Title("Spotify Top 10 艺术家动态榜单", order=1, align='center'),
        dmc.Space(h="md"),

        dmc.Grid(
            [
                dmc.Col(
                    dcc.Dropdown(
                        id="genre-dropdown",
                        options=[{"label": g, "value": g} for g in genres],
                        placeholder="选择流派",
                        value=genres[0] if genres else None,
                        clearable=False,
                    ),
                    span=6,
                ),
                dmc.Col(
                    dcc.Dropdown(
                        id="subgenre-dropdown",
                        options=[{"label": sg, "value": sg} for sg in subgenres],
                        placeholder="选择子流派",
                        value=subgenres[0] if subgenres else None,
                        clearable=False,
                    ),
                    span=6,
                ),
            ],
            gutter="xl",
        ),
        dmc.Space(h="xl"),

        dmc.Card(
            children=[
                dmc.Text("Top 10 艺术家", size='lg', color='dimmed', weight=500, align='center'),
                dmc.Space(h="sm"),
                dmc.Table(id='top_10_artists', striped=True, highlightOnHover=True, withBorder=True, withColumnBorders=True),
            ],
            withBorder=True,
            shadow='lg',
            radius='md',
        ),
    ],
    fluid=True,
    size="lg"
)

@app.callback(
    Output("top_10_artists", "children"),
    Input("genre-dropdown", "value"),
    Input("subgenre-dropdown", "value")
)
def update_top_10_artists_table(selected_genre, selected_subgenre):
    if not selected_genre or not selected_subgenre:
        return html.Div("请选择流派和子流派以查看数据。", style={'textAlign': 'center', 'marginTop': '20px'})

    df_filtered = data.copy()

    df_filtered = df_filtered[
        (df_filtered['Genre'] == selected_genre) &
        (df_filtered['Subgenre'] == selected_subgenre)
    ]

    artists_counts = df_filtered['Artist'].value_counts().reset_index()
    artists_counts.columns = ['Artist', 'Count']
    top_10_artists = artists_counts.head(10)

    if top_10_artists.empty:
        return html.Div("当前流派和子流派下没有找到艺术家数据。", style={'textAlign': 'center', 'marginTop': '20px'})

    header = html.Thead(
        html.Tr(
            [
                html.Th('艺术家'),
                html.Th('歌曲数量')
            ]
        )
    )

    rows = []
    for index, row_data in top_10_artists.iterrows():
        rows.append(
            html.Tr([
                html.Td(row_data['Artist']),
                html.Td(row_data['Count'])
            ])
        )
    body = html.Tbody(rows)

    return [header, body]

if __name__ == '__main__':
    app.run_server(debug=True)

运行此代码,您将在浏览器中看到一个Dash应用,其中包含两个下拉菜单和一个动态更新的表格。

注意事项与最佳实践

  1. Output属性的选择: 理解dmc.Table如何接收其内容至关重要。当需要完全控制表格的HTML结构(包括和 )时,应将Output指向组件的children属性,并返回一个由html.Thead和html.Tbody组成的列表。如果dmc.Table提供了更高级的data或columns属性(类似于dash_table.DataTable),则可以考虑使用它们来简化数据绑定,但这取决于dash_mantine_components的具体实现。
  2. 数据为空或不足的处理: 在回调函数中加入逻辑来处理筛选结果

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python 时间序列分析与预测
Python 时间序列分析与预测

本专题专注讲解 Python 在时间序列数据处理与预测建模中的实战技巧,涵盖时间索引处理、周期性与趋势分解、平稳性检测、ARIMA/SARIMA 模型构建、预测误差评估,以及基于实际业务场景的时间序列项目实操,帮助学习者掌握从数据预处理到模型预测的完整时序分析能力。

71

2025.12.04

Python 数据清洗与预处理实战
Python 数据清洗与预处理实战

本专题系统讲解 Python 在数据清洗与预处理中的核心技术,包括使用 Pandas 进行缺失值处理、异常值检测、数据格式化、特征工程与数据转换,结合 NumPy 高效处理大规模数据。通过实战案例,帮助学习者掌握 如何处理混乱、不完整数据,为后续数据分析与机器学习模型训练打下坚实基础。

1

2026.01.31

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

198

2023.11.20

点击input框没有光标怎么办
点击input框没有光标怎么办

点击input框没有光标的解决办法:1、确认输入框焦点;2、清除浏览器缓存;3、更新浏览器;4、使用JavaScript;5、检查硬件设备;6、检查输入框属性;7、调试JavaScript代码;8、检查页面其他元素;9、考虑浏览器兼容性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

186

2023.11.24

2026赚钱平台入口大全
2026赚钱平台入口大全

2026年最新赚钱平台入口汇总,涵盖任务众包、内容创作、电商运营、技能变现等多类正规渠道,助你轻松开启副业增收之路。阅读专题下面的文章了解更多详细内容。

28

2026.01.31

高干文在线阅读网站大全
高干文在线阅读网站大全

汇集热门1v1高干文免费阅读资源,涵盖都市言情、京味大院、军旅高干等经典题材,情节紧凑、人物鲜明。阅读专题下面的文章了解更多详细内容。

7

2026.01.31

无需付费的漫画app大全
无需付费的漫画app大全

想找真正免费又无套路的漫画App?本合集精选多款永久免费、资源丰富、无广告干扰的优质漫画应用,涵盖国漫、日漫、韩漫及经典老番,满足各类阅读需求。阅读专题下面的文章了解更多详细内容。

20

2026.01.31

漫画免费在线观看地址大全
漫画免费在线观看地址大全

想找免费又资源丰富的漫画网站?本合集精选2025-2026年热门平台,涵盖国漫、日漫、韩漫等多类型作品,支持高清流畅阅读与离线缓存。阅读专题下面的文章了解更多详细内容。

2

2026.01.31

漫画防走失登陆入口大全
漫画防走失登陆入口大全

2026最新漫画防走失登录入口合集,汇总多个稳定可用网址,助你畅享高清无广告漫画阅读体验。阅读专题下面的文章了解更多详细内容。

8

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 3.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号