0

0

Java Swing JTable:实现选中行永久变色效果

花韻仙語

花韻仙語

发布时间:2025-11-15 20:54:01

|

410人浏览过

|

来源于php中文网

原创

java swing jtable:实现选中行永久变色效果

本文详细介绍了如何在Java Swing的JTable中实现选中行永久变色的功能,即使在取消选择后颜色也能保持。核心方法是利用JTable的TableModel添加一个隐藏的布尔列来存储行的自定义选中状态,并通过自定义TableCellRenderer根据该状态渲染行背景色。同时,结合MouseListener监听用户交互,动态更新隐藏列的布尔值,并触发JTable重绘以实现视觉上的持久化变色效果。

在Java Swing应用程序中,JTable是常用的数据显示组件。默认情况下,当用户选中JTable中的一行时,该行会显示一个临时的选中颜色;一旦失去焦点或选择其他行,其颜色便会恢复。如果需要实现选中行颜色永久保持的效果,即使用户取消了当前行的选择,该行依然保持其自定义颜色,就需要结合自定义渲染器和数据模型进行处理。

核心原理分析

要实现JTable选中行永久变色,不能仅仅依赖JTable内置的选中状态。我们需要一种机制来持久化存储每一行的“自定义选中”状态,并在渲染时根据这个状态来着色。

  1. 扩展数据模型(TableModel):为JTable的数据模型添加一个额外的、通常是隐藏的列。这个列将存储一个布尔值,指示对应行是否处于“自定义选中”状态。
  2. 自定义单元格渲染器(TableCellRenderer):创建一个继承自DefaultTableCellRenderer的自定义渲染器。在getTableCellRendererComponent方法中,渲染器将检查当前行在数据模型中对应隐藏列的布尔值。如果为true,则设置该行的背景色为预设的“选中”颜色;否则,使用默认背景色。
  3. 事件监听器(MouseListener或ListSelectionListener):添加一个监听器到JTable。当用户点击某一行时,监听器会获取当前点击行的索引,并更新数据模型中对应隐藏列的布尔值(例如,从false切换到true,或从true切换到false)。更新数据模型后,需要通知JTable重绘以反映颜色的变化。

通过这种方式,行的颜色不再仅仅取决于JTable的临时选中状态,而是由数据模型中持久化的布尔值控制,从而实现永久变色。

立即学习Java免费学习笔记(深入)”;

实现步骤

下面将通过具体的代码示例来演示如何实现这一功能。

1. 准备JTable与TableModel

首先,我们需要创建一个JTable及其DefaultTableModel。关键在于为TableModel添加一个额外的列来存储行的选中状态。为了不让用户看到这个状态列,我们会在JTable的TableColumnModel中将其移除。

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;

import javax.swing.*;
import javax.swing.table.*;

// 主应用程序入口
public class TestTableRowColor {
    public static void main(String[] args) {
        SwingUtilities.invokeLater(() -> {
            ChangeRowColorPanel mainPanel = new ChangeRowColorPanel();

            JFrame frame = new JFrame("JTable行永久变色示例");
            frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            frame.add(mainPanel);
            frame.pack();
            frame.setLocationRelativeTo(null);
            frame.setVisible(true);
        });
    }
}

// 包含JTable的面板
@SuppressWarnings("serial")
class ChangeRowColorPanel extends JPanel {
    private static final String[] COLUMN_NAMES = { "列一", "列二", "列三", "选中状态" }; // 最后一列用于存储选中状态
    private DefaultTableModel model = new DefaultTableModel(COLUMN_NAMES, 0);
    private JTable table = new JTable(model);

    public ChangeRowColorPanel() {
        // 移除JTable的最后一列,使其在UI上不可见
        TableColumnModel columnModel = table.getColumnModel();
        columnModel.removeColumn(columnModel.getColumn(columnModel.getColumnCount() - 1));

        // 设置自定义渲染器
        table.setDefaultRenderer(Object.class, new RowColorRenderer());
        // 添加鼠标监听器
        table.addMouseListener(new MyMouse());

        // 填充示例数据
        int max = 5;
        for (int i = 0; i < max; i++) {
            Object[] row = new Object[COLUMN_NAMES.length];
            for (int j = 0; j < COLUMN_NAMES.length - 1; j++) {
                row[j] = (int) (100 * Math.random());
            }
            row[COLUMN_NAMES.length - 1] = false; // 初始状态为未选中
            model.addRow(row);
        }

        setLayout(new BorderLayout());
        add(new JScrollPane(table));
    }
}

在ChangeRowColorPanel的构造函数中,我们通过columnModel.removeColumn()方法将名为“选中状态”的最后一列从JTable的视图中移除,但它仍然存在于TableModel中,可以用于数据操作。

What-the-Diff
What-the-Diff

检查请求差异,自动生成更改描述

下载

2. 自定义表格单元格渲染器

接下来,创建RowColorRenderer,它将负责根据数据模型中的“选中状态”列来为行设置背景色。

@SuppressWarnings("serial")
class RowColorRenderer extends DefaultTableCellRenderer {
    private static final Color SELECTED_COLOR = Color.PINK; // 自定义选中颜色

    public RowColorRenderer() {
        setOpaque(true); // 确保渲染器组件是不透明的,以便背景色能显示出来
    }

    @Override
    public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus,
            int row, int column) {
        // 调用父类方法获取默认渲染组件,处理文本、字体等
        Component renderer = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);

        // 获取数据模型
        TableModel model = table.getModel();
        // 获取隐藏的“选中状态”列的索引
        int selectedColumn = model.getColumnCount() - 1;
        // 获取当前行的“选中状态”布尔值
        boolean customSelected = (boolean) model.getValueAt(row, selectedColumn);

        // 根据自定义选中状态设置背景色
        Color background = customSelected ? SELECTED_COLOR : null; // null表示使用JTable默认背景色
        renderer.setBackground(background);

        return renderer; // 返回修改后的渲染组件
    }
}

注意事项:

  • setOpaque(true) 是非常重要的,它确保渲染器组件能够显示其背景色。如果设置为false,背景色可能不会被绘制。
  • 在getTableCellRendererComponent方法中,我们首先调用super.getTableCellRendererComponent来获取默认的渲染组件,这保证了单元格的其他属性(如文本、字体、对齐方式等)能够正确处理。
  • 通过model.getValueAt(row, selectedColumn)获取当前行的自定义选中状态,并据此设置背景色。

3. 监听鼠标事件更新选中状态

最后,实现MyMouse监听器来响应用户点击,并更新数据模型中隐藏列的布尔值。

class MyMouse extends MouseAdapter {
    @Override
    public void mousePressed(MouseEvent e) {
        JTable table = (JTable) e.getSource();
        TableModel model = table.getModel();

        // 获取当前点击的行索引
        int selectedRow = table.getSelectedRow();
        if (selectedRow == -1) { // 如果没有选中行,则不处理
            return;
        }

        // 获取隐藏的“选中状态”列的索引
        int selectedColumnIndex = model.getColumnCount() - 1;
        // 获取当前行的自定义选中状态
        boolean currentCustomSelected = (boolean) model.getValueAt(selectedRow, selectedColumnIndex);

        // 切换选中状态
        model.setValueAt(!currentCustomSelected, selectedRow, selectedColumnIndex);

        // 强制JTable重绘,以立即显示颜色变化
        table.repaint();
    }
}

在mousePressed方法中:

  • 我们通过table.getSelectedRow()获取用户点击的行索引。
  • 然后,获取该行在隐藏的“选中状态”列中的当前布尔值。
  • 使用model.setValueAt()方法将该布尔值取反,从而切换行的自定义选中状态。
  • 最后,调用table.repaint()强制JTable重新绘制,触发RowColorRenderer重新渲染该行,从而显示新的颜色。

完整示例代码

将以上所有部分整合,即可得到一个完整的可运行示例。

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;

import javax.swing.*;
import javax.swing.table.*;

// 主应用程序入口
public class TestTableRowColor {
    public static void main(String[] args) {
        SwingUtilities.invokeLater(() -> {
            ChangeRowColorPanel mainPanel = new ChangeRowColorPanel();

            JFrame frame = new JFrame("JTable行永久变色示例");
            frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            frame.add(mainPanel);
            frame.pack();
            frame.setLocationRelativeTo(null);
            frame.setVisible(true);
        });
    }
}

// 包含JTable的面板
@SuppressWarnings("serial")
class ChangeRowColorPanel extends JPanel {
    private static final String[] COLUMN_NAMES = { "列一", "列二", "列三", "选中状态" }; // 最后一列用于存储选中状态
    private DefaultTableModel model = new DefaultTableModel(COLUMN_NAMES, 0);
    private JTable table = new JTable(model);

    public ChangeRowColorPanel() {
        // 移除JTable的最后一列,使其在UI上不可见
        TableColumnModel columnModel = table.getColumnModel();
        columnModel.removeColumn(columnModel.getColumn(columnModel.getColumnCount() - 1));

        // 设置自定义渲染器
        table.setDefaultRenderer(Object.class, new RowColorRenderer());
        // 添加鼠标监听器
        table.addMouseListener(new MyMouse());

        // 填充示例数据
        int max = 5;
        for (int i = 0; i < max; i++) {
            Object[] row = new Object[COLUMN_NAMES.length];
            for (int j = 0; j < COLUMN_NAMES.length - 1; j++) {
                row[j] = (int) (100 * Math.random());
            }
            row[COLUMN_NAMES.length - 1] = false; // 初始状态为未选中
            model.addRow(row);
        }

        setLayout(new BorderLayout());
        add(new JScrollPane(table));
    }
}

// 鼠标监听器,用于切换行的自定义选中状态
class MyMouse extends MouseAdapter {
    @Override
    public void mousePressed(MouseEvent e) {
        JTable table = (JTable) e.getSource();
        TableModel model = table.getModel();

        // 获取当前点击的行索引
        int selectedRow = table.getSelectedRow();
        if (selectedRow == -1) { // 如果没有选中行,则不处理
            return;
        }

        // 获取隐藏的“选中状态”列的索引
        int selectedColumnIndex = model.getColumnCount() - 1;
        // 获取当前行的自定义选中状态
        boolean currentCustomSelected = (boolean) model.getValueAt(selectedRow, selectedColumnIndex);

        // 切换选中状态
        model.setValueAt(!currentCustomSelected, selectedRow, selectedColumnIndex);

        // 强制JTable重绘,以立即显示颜色变化
        table.repaint();
    }
}

// 自定义单元格渲染器,根据隐藏列的布尔值设置行背景色
@SuppressWarnings("serial")
class RowColorRenderer extends DefaultTableCellRenderer {
    private static final Color SELECTED_COLOR = Color.PINK; // 自定义选中颜色

    public RowColorRenderer() {
        setOpaque(true); // 确保渲染器组件是不透明的,以便背景色能显示出来
    }

    @Override
    public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus,
            int row, int column) {
        // 调用父类方法获取默认渲染组件,处理文本、字体等
        Component renderer = super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);

        // 获取数据模型
        TableModel model = table.getModel();
        // 获取隐藏的“选中状态”列的索引
        int selectedColumn = model.getColumnCount() - 1;
        // 获取当前行的“选中状态”布尔值
        boolean customSelected = (boolean) model.getValueAt(row, selectedColumn);

        // 根据自定义选中状态设置背景色
        Color background = customSelected ? SELECTED_COLOR : null; // null表示使用JTable默认背景色
        renderer.setBackground(background);

        return renderer; // 返回修改后的渲染组件
    }
}

总结

通过以上方法,我们成功地在Java Swing的JTable中实现了选中行颜色永久变色的效果。这种方法的核心在于:

  • 数据与视图分离:通过在TableModel中维护一个独立的“选中状态”数据,将行的选中逻辑与JTable的默认选择行为解耦。
  • 自定义渲染:利用TableCellRenderer的强大功能,根据数据模型中的自定义状态来灵活控制单元格乃至整行的外观。
  • 事件驱动:通过MouseListener(或ListSelectionListener)响应用户交互,及时更新数据模型并触发视图重绘。

这种模式不仅适用于行颜色变化,还可以扩展到其他更复杂的自定义渲染需求,是JTable高级定制中非常常用的技术。

相关文章

Windows激活工具
Windows激活工具

Windows激活工具是正版认证的激活工具,永久激活,一键解决windows许可证即将过期。可激活win7系统、win8.1系统、win10系统、win11系统。下载后先看完视频激活教程,再进行操作,100%激活成功。

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

16

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

23

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

75

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

95

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

218

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

420

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

168

2026.03.04

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

222

2026.03.03

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

33

2026.03.03

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.6万人学习

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

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