多表联合的查询(实例)、对于前端返回数据有很多表,可以分开操作、debug调试教程

news2025/1/13 2:57:56

2024.7.13

  • 一、 对于多表的更深层的认识
    • 1. 认识
    • 2. 多表联合查询的列子:
    • 3. 对于多表查询的进一步认识
    • 4. 在实现功能的时候,原本对于省市县这样的表,对于项目的要求,是直接全部查询出来,然后开始使用,但我想着能不能直接用树形结构来操作,然后通过递归的形式来实现,让我们看看我的成果!!
    • 4.1 其他的都不看了,直接看service:
    • 4.2 解释 for (addDistrictDTO area : allDistrict)
    • 4.3 成果!
  • 二、 新增
    • 1. 先看需求!
    • 2. 新增的分析(查询和上面大差不差,直接返回就行,就不提了)
    • 3. 开始开发!!
    • 4. 对于不分开,直接进行插入,我没弄出来,一直报错,有机会我会写在这,如果真有人看,能指点我一下!!!
    • 5. 优化我的代码(递归,以及两个sql操作)
  • 三、debug调试
    • 1. 第二步,debug启动!
    • 2. 第一步,对于错误的方法和地方打断点,查看当时的值和状态。(不用像我这样,哈哈哈,我这太夸张了)
    • 3. 第三步,通过这些按钮进行控制
    • 4. 查看数据
  • 四、感谢大佬们的文章给我的提示,也感谢该死的GPT!!
  • 五、感悟

一、 对于多表的更深层的认识

1. 认识

我今天早上,基本上一直都在进行查询的书写,写了四五个查询的接口吧,根据着页面和F12进行推断,我需要哪些表,一开始很慢,因为要去找对应的字段,需要自己写xml。后面结合gpt就快很多了(不要依赖gpt,我就是因为太依赖gpt,导致于我在一个逻辑上,没有仔细想,出错了,就问他,他给我建议,离我最初的逻辑越来越偏!!!gpt只是工具。

2. 多表联合查询的列子:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">

<mapper namespace="com.hengyin.ship.mapper.AddMapper">
    <!-- Result Map 定义 -->
    <resultMap id="MaterialsResultMap" type="com.hengyin.ship.domain.Materials">
        <result property="outTermShipBomNo" column="out_term_ship_bom_no" />
        <result property="outTermShipBomName" column="out_term_ship_bom_name" />
        <result property="outTermTypeName" column="out_term_type_name" />
        <result property="outModelOuterName" column="out_model_outer_name" />
        <result property="outModelInnerName" column="out_model_inner_name" />
    </resultMap>

    <!-- 查询语句定义 -->
    <select id="selectMaterials" resultMap="MaterialsResultMap">
        SELECT
            t1.out_term_ship_bom_no,
            t1.out_term_ship_bom_name,
            t1.out_term_type_name,
            t1.out_model_outer_name,
            t1.out_model_inner_name
        FROM
            om_term_info t1
                JOIN
            out_term_ship t2 ON t1.out_term_ship_bom_no = t2.bom_no
                JOIN
            out_bom_info t3 ON t1.out_term_ship_bom_name = t3.bom_name
                JOIN
            out_term_type t4 ON t1.out_term_type_name = t4.name
                JOIN
            out_model_outer t5 ON t1.out_model_outer_name = t5.model_outer
                JOIN
            out_model_inner t6 ON t1.out_model_inner_name = t6.model_inner
        WHERE
            t1.out_term_ship_bom_no = #{outTermShipBomNo}
    </select>




    <insert id="add">
        INSERT INTO om_term_info (
            term_sn,
            out_term_ship_bom_no,
            out_term_ship_bom_name,
            out_term_type_name,
            out_model_outer_name,
            out_model_inner_name,
            out_custom_info_name,
            out_sales_contract_sales_uid,
            sub_branch_contacts,
            out_region_id,
            out_region_name,
            out_region_parent_code,
            ship_state,
            out_term_ship_ship_time,
            out_term_ship_take_address,
            branch_name,
            sub_branch_tel,
            out_logistics_provider_info_id,
            out_logistics_provider_info_name
        )
        VALUES (
                   #{termSn},
                   #{outTermShipBomNo},
                   #{outTermShipBomName},
                   #{outTermTypeName},
                   #{outModelOuterName},
                   #{outModelInnerName},
                   #{customName},
                   #{salesUid},
                   #{code},
                   #{name},
                    #{parentCode}
                   #{shipState},
                   #{shipTime, jdbcType=TIMESTAMP},
                   #{takeAddress},
                   #{branchName},
                   #{bankContacts},
                   #{bankTel},
                   #{LoCode},
                   #{LoName}
               )
    </insert>
    <insert id="addDTO">
        INSERT INTO om_term_info (
            term_sn, out_term_ship_bom_no, out_term_ship_bom_name,
            out_term_type_name, out_model_outer_name, out_model_inner_name,
            custom_num, out_sales_contract_sales_uid, ship_state, out_term_ship_ship_time,
            out_term_ship_take_address, branch_name, sub_branch_contacts, sub_branch_tel,
            out_logistics_provider_info_name
        ) VALUES (
                     #{termSn}, #{outTermShipBomNo}, #{outTermShipBomName},
                     #{outTermTypeName}, #{outModelOuterName}, #{outModelInnerName},
                     #{customName}, #{salesUid}, #{shipState}, #{shipTime},
                     #{takeAddress}, #{branchName}, #{bankContacts}, #{bankTel},
                     #{loName}
                 )
    </insert>
    <update id="addDistrict">
        UPDATE om_term_info
        SET
            out_region_id = #{code},
            out_region_name = #{name},
            out_region_parent_code = #{parentCode}
        WHERE term_sn = #{termSn};
    </update>
</mapper>

3. 对于多表查询的进一步认识

我一直对于这种关联表和基础表认识的不是特别透彻,当我操作的一层表里有和基础表一样的数据的时候,我在想,他们两个关联,是不是证明,他们就会数据共享,我修改一个另一个也会修改?那我到底该操作哪个表?底层表还是我的一层表。(虽然现在我说出来,你可能会觉得,这不肯定是一层表吗?但我确实绕进去了,哈哈哈)
然后开始理清楚,我可以从底层表里拿到数据,返回,也能从一层表拿啊,但底层表很多都是提供的公共数据,虽然我们小组成员之间会使用同一张表,但我们创建的数据,和修改的数据,不应该去修改底层表,而是应该修改自己的主表(主要操作的表),比如我现在进行发货管理,对于发货信息表和信息台账表,**我看了一下,信息台账表里我需要的数据会更多,应该定为主表。**所以,在后面新增操作的时候,都是直接操作自己的一层表,对于查询的话,可能两个都会使用到。

4. 在实现功能的时候,原本对于省市县这样的表,对于项目的要求,是直接全部查询出来,然后开始使用,但我想着能不能直接用树形结构来操作,然后通过递归的形式来实现,让我们看看我的成果!!

4.1 其他的都不看了,直接看service:

@Service
public class AddServiceImpl implements AddService {
@Autowired
private AddMapper addMapper;


    @Override
    public addLogistics selectLogistics() {
        return addMapper.selectLogistics();
    }

    // 构建层级结构的方法
    private List<addDistrictDTO> buildHierarchy(List<addDistrictDTO> allDistrict, String parentCode) {
        //这里开始进入方法,传入的是我们的allDistrict(含值的对象),和parentcode的值
        List<addDistrictDTO> result = new ArrayList<>();
        //定义一个集合 result用于存放结果
        System.out.println("Building hierarchy for parentCode: " + parentCode);
        // 打印当前递归层级的父节点代码

        // 遍历所有的地区对象
        for (addDistrictDTO area : allDistrict) {
            System.out.println("Checking area: " + area.getCode() + ", parentCode: " + area.getParent_code());
            // 打印正在检查的地区代码和其父节点代码
            if ((parentCode == null && area.getParent_code()== null) || (parentCode != null && parentCode.equals(area.getParent_code()))) {
                //  如果是子地区,则递归调用 buildHierarchy 方法,构建子地区的树形结构
                List<addDistrictDTO> children = buildHierarchy(allDistrict, area.getCode());
                // 如果子地区列表不为空,将其设置为当前地区的子地区
                if (!children.isEmpty()) {
                    area.setChildren(children);
                }
                // 将当前地区加入结果列表
                result.add(area);
            }
        }
        // 打印当前父节点下构建的结果
        System.out.println("Result for parentCode " + parentCode + ": " + result);
        return result;
    }
    @Override
    public List<addDistrictDTO> getAllDistrict() {//这个是我们的切入点
        List<addDistrictDTO> allAllDistrict = addMapper.getAllDistrict();//我们先调用getAllDistrict,查出所有的数据包含code,name,parent_id
        return buildHierarchy(allAllDistrict, null);//然后调用我们的递归。最开始的parentcode的值为null
    }

    @Override
    public List<User> selectUser() {
        return addMapper.selectUser();
    }

    @Override
    public List<Materials> selectMaterials(String outTermShipBomNo) {
        return addMapper.selectMaterials(outTermShipBomNo);

4.2 解释 for (addDistrictDTO area : allDistrict)

解释:这段代码中的 for (addDistrictDTO area : allDistrict) 是 Java 中的增强型 for 循环,也称为 for-each 循环。它是一种简化遍历数组或集合的语法形式,其基本结构为:
addDistrictDTO 是迭代过程中每个元素的类型。
area 是循环每次迭代时,代表集合 allDistrict 中的一个元素。
allDistrict 是需要被遍历的集合,其中存储了 addDistrictDTO 类型的对象。

for (ElementType element : collection) {
    // 循环体内的操作
}

4.3 成果!

[
    {
        "code": "1",
        "name": "上海市",
        "parent_code": null,
        "children": [
            {
                "code": "2",
                "name": "嘉定区",
                "parent_code": "1",
                "children": []
            }
        ]
    },
    {
        "code": "3",
        "name": "贵州省",
        "parent_code": null,
        "children": [
            {
                "code": "4",
                "name": "六盘水市",
                "parent_code": "3",
                "children": [
                    {
                        "code": "5",
                        "name": "盘州市",
                        "parent_code": "4",
                        "children": []
                    }
                ]
            },
            {
                "code": "6",
                "name": "贵阳市",
                "parent_code": "3",
                "children": [
                    {
                        "code": "7",
                        "name": "花溪区",
                        "parent_code": "6",
                        "children": []
                    }
                ]
            }
        ]
    }
]

二、 新增

1. 先看需求!

在这里插入图片描述

2. 新增的分析(查询和上面大差不差,直接返回就行,就不提了)

首先,我对表进行了修改,因为我在我的基础表上,只有一两个字段没有(偷个懒,写入接口文档找关联吧),然后开始分析,因为返回的数据Json为:(这个json写的也真费劲,让gpt写,写的一点都不匹配!!!一定先修改json对于属性。)

{
    "termSn": "21",
    "outTermShipBomNo": "BOM123456",
    "outTermShipBomName": "Bom Name",
    "outTermTypeName": "Type Name",
    "outModelOuterName": "Outer Model Name",
    "outModelInnerName": "Inner Model Name",
    "customName": "Customer Name",
    "salesUid": 1001,
    "phoneNumber": "1234567890",
    "shipState": 1,
    "shipTime": "2024-07-13",
    "takeAddress": "123 Shipping Address",
    "branchName": "Branch Name",
    "bankContacts": "Bank Contacts",
    "bankTel": "0987654321",
    "loCode": "LOG123",
    "loName": "Logistics Name",
    "children": [
        {
            "code": "110000",
            "name": "Beijing",
            "parentCode": "000000",
            "children": [
                {
                    "code": "110100",
                    "name": "Beijing City",
                    "parentCode": "110000",
                    "children": []
                }
            ]
        },
        {
            "code": "120000",
            "name": "Tianjin",
            "parentCode": "000000",
            "children": [
                {
                    "code": "120100",
                    "name": "Tianjin City",
                    "parentCode": "120000",
                    "children": []
                }
            ]
        }
    ]
}

3. 开始开发!!

对于我们传回后端进行处理的数据里,有一个树形结构(也不知道我犟什么,不用这个早写完了),开始进行设计。
我想到,我定义一个总的类(包含所有的属性),然后用行政区域的DTO把树形结构拿走,其他的再封装为一个DTO进行操作。然后实体类的DTO直接进行插入操作,然后行政区域的根据SN进行修改,开始操作!对了,我首先想到的就是 BeanUtils.copyProperties(a,b)用大去赋小,这样就可以把不用的省去,就不用stream流了。
首先,树形数据也是需要进行递归把数据拿出来,插入到表里面,那我们老规矩,直接看service:

   @Transactional
    @Override
    public void add(AddPage addPage) {
        AddPageDTO addPageDTO = new AddPageDTO();
        System.out.println("Received addPage termSn: " + addPage.getTermSn());

        // 获取 termSn 并设置到 addPage 和 addPageDTO
        String termSn = addPage.getTermSn();
        addDistrict.settermSn(termSn);
        addPageDTO.setTermSn(termSn);

        addDistrict addDistrict = new addDistrict();
        BeanUtils.copyProperties(addPage, addPageDTO);
        System.out.println("Received addPageDTO termSn: " + addPageDTO.getTermSn());
        BeanUtils.copyProperties(addPage, addDistrict);
        addDistrict.setTermSn(termSn); //这里把sn放到addDistrict中,下面查不到!!!
        // 检查 termSn 字段是否有值
        if (addPageDTO.getTermSn() == null) {
            throw new IllegalArgumentException("termSn cannot be null");
        }

        // 插入主记录
        addMapper.addDTO(addPageDTO);

            addDistrictTree(addPage.getChildren(), termSn); // 传递 termSn

    }

    private void addDistrictTree(List<addDistrict> districts, String termSn) {
        for (addDistrict district : districts) {

            district.setTermSn(termSn); // 设置 termSn
            // 插入子记录
            addMapper.addDistrict(district);
            if (district.getChildren() != null && !district.getChildren().isEmpty()) {
                addDistrictTree(district.getChildren(), district.getTermSn()); // 传递当前节点的 termSn
            }
        }
    }

然后就是XML!:

 <insert id="addDTO">
        INSERT INTO om_term_info (
            term_sn, out_term_ship_bom_no, out_term_ship_bom_name,
            out_term_type_name, out_model_outer_name, out_model_inner_name,
            custom_num, out_sales_contract_sales_uid, ship_state, out_term_ship_ship_time,
            out_term_ship_take_address, branch_name, sub_branch_contacts, sub_branch_tel,
            out_logistics_provider_info_name
        ) VALUES (
                     #{termSn}, #{outTermShipBomNo}, #{outTermShipBomName},
                     #{outTermTypeName}, #{outModelOuterName}, #{outModelInnerName},
                     #{customName}, #{salesUid}, #{shipState}, #{shipTime},
                     #{takeAddress}, #{branchName}, #{bankContacts}, #{bankTel},
                     #{loName}
                 )
    </insert>
    <update id="addDistrict">
        UPDATE om_term_info
        SET
            out_region_id = #{code},
            out_region_name = #{name},
            out_region_parent_code = #{parentCode}
        WHERE term_sn = #{termSn};
    </update>

这样就可以完美插入了,然后再让gpt生成一条查询这些语句的sql,来查看我们是否完成插入。

SELECT
    term_sn,
    out_term_ship_bom_no,
    out_term_ship_bom_name,
    out_term_type_name,
    out_model_outer_name,
    out_model_inner_name,
    custom_num AS custom_name, -- 注意这里假设myBatis中的#{customName}对应于数据库中的custom_num
    out_sales_contract_sales_uid AS sales_uid,
    ship_state,
    out_term_ship_ship_time AS ship_time,
    out_term_ship_take_address AS take_address,
    branch_name,
    sub_branch_contacts,
    sub_branch_tel,
    out_logistics_provider_info_name AS lo_name,
    out_region_id,
    out_region_name,
    out_region_parent_code
FROM
    om_term_info
WHERE
        term_sn = '20';

在这里插入图片描述

4. 对于不分开,直接进行插入,我没弄出来,一直报错,有机会我会写在这,如果真有人看,能指点我一下!!!

5. 优化我的代码(递归,以及两个sql操作)

三、debug调试

1. 第二步,debug启动!

在这里插入图片描述

2. 第一步,对于错误的方法和地方打断点,查看当时的值和状态。(不用像我这样,哈哈哈,我这太夸张了)

在这里插入图片描述

3. 第三步,通过这些按钮进行控制

在这里插入图片描述

4. 查看数据

在这里插入图片描述

四、感谢大佬们的文章给我的提示,也感谢该死的GPT!!

Debug断点调试
DTO到entity赋值

五、感悟

当你遇到问题的时候,一定要先独立思考,思考不出来,就要考虑换个思路,起来走走,问问别人,不要和gpt死磕,更多的用自己的思维。
还有就是代码规范的问题,写的一点都不规范,命名和方法名称的规范一定要注意。新增模块还有需求,明天再写!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1924674.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

在 CentOS 6.4 VPS 上安装和保护 phpMyAdmin 的方法

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 简介 许多网站和应用程序需要数据库来存储和管理大量信息。MySQL 和 MariaDB 是流行的数据库管理系统&#xff0c;因为它们具有灵活性、…

redisTemplate报错为nil,通过redis-cli查看前缀有乱码

public void set(String key, String value, long timeout) {redisTemplate.opsForValue().set(key, value, timeout, TimeUnit.SECONDS);} 改完之后 public void set(String key, String value, long timeout) {redisTemplate.setKeySerializer(new StringRedisSerializer()…

前端工程化10-webpack静态的模块化打包工具之各种loader处理器

9.1、案例编写 我们创建一个component.js 通过JavaScript创建了一个元素&#xff0c;并且希望给它设置一些样式&#xff1b; 我们自己写的css,要把他加入到Webpack的图结构当中&#xff0c;这样才能被webpack检测到进行打包&#xff0c; style.css–>div_cn.js–>main…

万界星空科技MES系统:食品加工安全的实时监控与智能管理

万界星空科技MES系统通过集成多种技术和功能&#xff0c;能够实时监控食品加工过程中各环节的安全风险。以下是对该系统如何实现实时监控的详细分析&#xff1a; 一、集成传感器和数据分析技术 万界星空科技MES系统利用集成的传感器和数据分析技术&#xff0c;实时监控生产过程…

Linux rsync文件同步工具

scp的不足 1. 性能问题 单线程传输 SCP只使用单线程进行传输&#xff0c;这意味着在传输大文件或大量小文件时&#xff0c;其传输速度和效率可能不如其他多线程工具。 无法压缩数据传输 SCP不支持内置的压缩机制&#xff0c;这在传输大文件时会导致带宽使用效率较低。 2.…

神经网络以及简单的神经网络模型实现

神经网络基本概念&#xff1a; 神经元&#xff08;Neuron&#xff09;&#xff1a; 神经网络的基本单元&#xff0c;接收输入&#xff0c;应用权重并通过激活函数生成输出。 层&#xff08;Layer&#xff09;&#xff1a; 神经网络由多层神经元组成。常见的层包括输入层、隐藏层…

【MySQL 进阶】MySQL 程序 -- 详解

一、MySQL 程序简介 MySQL 安装完成通常会包含如下程序&#xff1a; 1、Linux 系统 程序⼀般在 /usr/bin 目录下&#xff0c;可以通过命令查看&#xff1a; 2、Windows系统 目录&#xff1a;你的安装路径\MySQL Server 8.0\bin&#xff0c;可以通过命令查看&#xff1a; 可…

Vue el-input 限制输入内容

&#x1f914;日常项目中经常遇到既要el-input的样式&#xff0c;又要el-input-number限制&#xff0c;所以需要绑定input事件进行约束输入限制。 以下使用自定义指令进行约束el-input输入的值&#xff0c;便于后期统一管理和拓展。 预览 代码 <!DOCTYPE html> <ht…

STM32入门开发操作记录(二)——LED与蜂鸣器

目录 一、工程模板二、点亮主板1. 配置寄存器2. 调用库函数 三、LED1. 闪烁2. 流水灯 四、蜂鸣器 一、工程模板 参照第一篇&#xff0c;新建工程目录ProjectMould&#xff0c;将先前打包好的Start&#xff0c;Library和User文件^C^V过来&#xff0c;并在Keil5内完成器件支持包的…

[MySQL][表操作]详细讲解

目录 1.创建表1.基本语法2.创建表案例 2.查看表结构3.修改表1.语法2.示例3.modify和change区别 4.删除表 1.创建表 1.基本语法 语法&#xff1a; CREATE TABLE table_name (field1 datatype,field2 datatype,field3 datatype ) character set 字符集 collate 校验规则 engin…

阿里云产品流转

本文主要记述如何使用阿里云对数据进行流转&#xff0c;这里只是以topic流转&#xff08;再发布&#xff09;为例进行说明&#xff0c;可能还会有其他类型的流转&#xff0c;不同服务器的流转也可能会不一样&#xff0c;但应该大致相同。 1 创建设备 具体细节可看&#xff1a;…

STM32F103定时器中断详解

目录 目录 目录 前言 一.什么是定时器 1.1 STM32F103定时器概述 1.2基本定时器 1.2通用定时器 1.3高级定时器 1.4 三种定时器区别 基本定时器&#xff08;Basic Timer&#xff09; 通用定时器&#xff08;General-Purpose Timer&#xff09; 高级定时器&#xff08;Advanced Ti…

企业网三层架构

企业网三层架构&#xff1a;是一种层次化模型设计&#xff0c;旨在将复杂的网络设计分成三个层次&#xff0c;每个层次都着重于某些特定的功能&#xff0c;以提高效率和稳定性。 企业网三层架构层次&#xff1a; 接入层&#xff1a;使终端设备接入到网络中来&#xff0c;提供…

提高使用安全,智慧校园在线用户功能概述

智慧校园系统融入了一个查看当前在线用户的功能&#xff0c;这一设计旨在为管理人员提供一个实时的窗口&#xff0c;洞悉校园平台的即时活跃情况&#xff0c;确保系统的高效运作与环境安全。通过这一功能&#xff0c;管理员能够一目了然地看到所有正活跃在平台上的用户群体&…

『 Linux 』匿名管道应用 - 简易进程池

文章目录 池化技术进程池框架及基本思路进程的描述组织管道通信建立的潜在问题 任务的描述与组织子进程读取管道信息控制子进程进程退出及资源回收 池化技术 池化技术是一种编程技巧,一般用于优化资源的分配与复用; 当一种资源需要被使用时这意味着这个资源可能会被进行多次使…

GuLi商城-商品服务-API-品牌管理-JSR303分组校验

注解:@Validated 实体类: package com.nanjing.gulimall.product.entity;import com.baomidou.mybatisplus.annotation.TableId; import com.baomidou.mybatisplus.annotation.TableName; import com.nanjing.common.valid.ListValue; import com.nanjing.common.valid.Updat…

代码随想录——不同路径Ⅱ(Leetcode 63)

题目链接 动态规划 class Solution {public int uniquePathsWithObstacles(int[][] obstacleGrid) {int m obstacleGrid.length;int n obstacleGrid[0].length;int[][] dp new int[m][n];// 遇到障碍则从(0,0)到达for(int i 0; i < m && obstacleGrid[i][0] …

【C++】初始化列表”存在的意义“和“与构造函数体内定义的区别“

构造函数是为了方便类的初始化而存在&#xff0c;而初始化时会遇到const成员变量、引用成员变量等&#xff0c;这些变量不允许函数内赋值&#xff0c;必须要在初始化时进行赋值&#xff0c;所以就有了初始化列表&#xff0c;初始化列表只能存在于类的构造函数中&#xff0c;用于…

百日筑基第二十天-一头扎进消息队列3-RabbitMQ

百日筑基第二十天-一头扎进消息队列3-RabbitMQ 如上图所示&#xff0c;RabbitMQ 由 Producer、Broker、Consumer 三个大模块组成。生产者将数据发送到 Broker&#xff0c;Broker 接收到数据后&#xff0c;将数据存储到对应的 Queue 里面&#xff0c;消费者从不同的 Queue 消费数…

PySide(PyQt),csv文件的显示

1、正常显示csv文件 import sys import csv from PySide6.QtWidgets import QApplication, QMainWindow, QTableWidget, QTableWidgetItem, QWidgetclass CSVTableWidgetDemo(QMainWindow):def __init__(self):super().__init__()# 创建显示控件self.widget QWidget(self)sel…