short_open_tag = On

解决LAMP环境下PHP页面无法加载的问题
本文介绍了一种在LAMP环境下遇到的PHP页面无法正常加载的问题及其解决方案。通过对php.ini配置文件中short_open_tag选项的设置及对PHP测试文件格式的调整,成功解决了该问题。
安装LAMP出现的奇怪问题
今天因为要安装CACTI,需要安装LAMP,依次把MYSQL、APACHE装好了,接下来把PHP也装好
在APACHE的文档目录下建立一个PHP测试文件
内容为
<?
phpinfo();
?>
然后重启apache进行访问,发现PHP的页面出不来,已经加上
AddType application/x-httpd-php .php
AddType image/x-icon .ico
修改DirectoryIndex 行,添加index.php
上面的已经改了,还是不行,
这时将index.php里进行一点小改动
<?php
phpinfo();
?>
在?后面加php,这是php文档的标准格式。再进行访问,就可以出来了
另外一种解决访问是。编辑php.ini文件,修改short_open_tag = On表示支持短标签。
就可以访问?号后面没有加php的格式了

注意 有些php集成开发环境 可能需要同时修改

php 以及 Apache/bin 下的 php.ini 文件

/* * SPDX-FileCopyrightText: 2021-2024 Espressif Systems (Shanghai) CO LTD * * SPDX-License-Identifier: CC0-1.0 */ #include <stdio.h> #include <string.h> #include <stdlib.h> #include "sdkconfig.h" #include "freertos/FreeRTOS.h" #include "freertos/task.h" #include "driver/i2s_std.h" #include "driver/gpio.h" #include "esp_system.h" #include "esp_check.h" #include "es8311.h" #include "example_config.h" #include "esp_vfs_fat.h" #include "sdmmc_cmd.h" #include "driver/sdmmc_host.h" #include "mp3dec.h" static const char *TAG = "i2s_es8311_mp3"; static const char err_reason[][30] = {"input param is invalid", "operation timeout" }; static i2s_chan_handle_t tx_handle = NULL; static i2s_chan_handle_t rx_handle = NULL; // SD卡相关变量 static sdmmc_card_t *card; static esp_err_t sdcard_init(void) { esp_err_t ret; // 配置SD卡GPIO gpio_set_pull_mode(SD_CLK_IO, GPIO_PULLUP_ONLY); gpio_set_pull_mode(SD_CMD_IO, GPIO_PULLUP_ONLY); gpio_set_pull_mode(SD_D0_IO, GPIO_PULLUP_ONLY); // 配置FAT文件系统 esp_vfs_fat_sdmmc_mount_config_t mount_config = { .format_if_mount_failed = false, .max_files = 5, .allocation_unit_size = 16 * 1024 }; // 配置SDMMC主机 sdmmc_host_t host = SDMMC_HOST_DEFAULT(); host.max_freq_khz = SDMMC_FREQ_HIGHSPEED; // 配置SD卡引脚 sdmmc_slot_config_t slot_config = SDMMC_SLOT_CONFIG_DEFAULT(); slot_config.width = 1; slot_config.clk = SD_CLK_IO; slot_config.cmd = SD_CMD_IO; slot_config.d0 = SD_D0_IO; slot_config.d1 = SD_D1_IO; slot_config.d2 = SD_D2_IO; slot_config.d3 = SD_D3_IO; // 挂载文件系统 ret = esp_vfs_fat_sdmmc_mount(MOUNT_POINT, &host, &slot_config, &mount_config, &card); if (ret != ESP_OK) { if (ret == ESP_FAIL) { ESP_LOGE(TAG, "Failed to mount filesystem. " "If you want the card to be formatted, set format_if_mount_failed = true."); } else { ESP_LOGE(TAG, "Failed to initialize the card (%s). " "Make sure SD card lines have pull-up resistors in place.", esp_err_to_name(ret)); } return ret; } ESP_LOGI(TAG, "SD card mounted successfully"); return ESP_OK; } static esp_err_t es8311_codec_init(void) { /* Initialize I2C peripheral */ #if !defined(CONFIG_EXAMPLE_BSP) const i2c_config_t es_i2c_cfg = { .sda_io_num = I2C_SDA_IO, .scl_io_num = I2C_SCL_IO, .mode = I2C_MODE_MASTER, .sda_pullup_en = GPIO_PULLUP_ENABLE, .scl_pullup_en = GPIO_PULLUP_ENABLE, .master.clk_speed = 100000, }; ESP_RETURN_ON_ERROR(i2c_param_config(I2C_NUM, &es_i2c_cfg), TAG, "config i2c failed"); ESP_RETURN_ON_ERROR(i2c_driver_install(I2C_NUM, I2C_MODE_MASTER, 0, 0, 0), TAG, "install i2c driver failed"); #else ESP_ERROR_CHECK(bsp_i2c_init()); #endif /* Initialize es8311 codec */ es8311_handle_t es_handle = es8311_create(I2C_NUM, ES8311_ADDRRES_0); ESP_RETURN_ON_FALSE(es_handle, ESP_FAIL, TAG, "es8311 create failed"); const es8311_clock_config_t es_clk = { .mclk_inverted = false, .sclk_inverted = false, .mclk_from_mclk_pin = true, .mclk_frequency = EXAMPLE_MCLK_FREQ_HZ, .sample_frequency = EXAMPLE_SAMPLE_RATE }; ESP_ERROR_CHECK(es8311_init(es_handle, &es_clk, ES8311_RESOLUTION_16, ES8311_RESOLUTION_16)); ESP_RETURN_ON_ERROR(es8311_sample_frequency_config(es_handle, EXAMPLE_SAMPLE_RATE * EXAMPLE_MCLK_MULTIPLE, EXAMPLE_SAMPLE_RATE), TAG, "set es8311 sample frequency failed"); ESP_RETURN_ON_ERROR(es8311_voice_volume_set(es_handle, EXAMPLE_VOICE_VOLUME, NULL), TAG, "set es8311 volume failed"); ESP_RETURN_ON_ERROR(es8311_microphone_config(es_handle, false), TAG, "set es8311 microphone failed"); #if CONFIG_EXAMPLE_MODE_ECHO ESP_RETURN_ON_ERROR(es8311_microphone_gain_set(es_handle, EXAMPLE_MIC_GAIN), TAG, "set es8311 microphone gain failed"); #endif return ESP_OK; } static esp_err_t i2s_driver_init(void) { #if !defined(CONFIG_EXAMPLE_BSP) i2s_chan_config_t chan_cfg = I2S_CHANNEL_DEFAULT_CONFIG(I2S_NUM, I2S_ROLE_MASTER); chan_cfg.dma_desc_num = 8; chan_cfg.dma_frame_num = 512; chan_cfg.auto_clear = true; // Auto clear the legacy data in the DMA buffer ESP_ERROR_CHECK(i2s_new_channel(&chan_cfg, &tx_handle, &rx_handle)); i2s_std_config_t std_cfg = { .clk_cfg = I2S_STD_CLK_DEFAULT_CONFIG(EXAMPLE_SAMPLE_RATE), .slot_cfg = I2S_STD_PHILIPS_SLOT_DEFAULT_CONFIG(I2S_DATA_BIT_WIDTH_16BIT, I2S_SLOT_MODE_STEREO), .gpio_cfg = { .mclk = I2S_MCK_IO, .bclk = I2S_BCK_IO, .ws = I2S_WS_IO, .dout = I2S_DO_IO, .din = I2S_DI_IO, .invert_flags = { .mclk_inv = false, .bclk_inv = false, .ws_inv = false, }, }, }; std_cfg.clk_cfg.mclk_multiple = EXAMPLE_MCLK_MULTIPLE; ESP_ERROR_CHECK(i2s_channel_init_std_mode(tx_handle, &std_cfg)); ESP_ERROR_CHECK(i2s_channel_init_std_mode(rx_handle, &std_cfg)); ESP_ERROR_CHECK(i2s_channel_enable(tx_handle)); ESP_ERROR_CHECK(i2s_channel_enable(rx_handle)); #else ESP_LOGI(TAG, "Using BSP for HW configuration"); i2s_std_config_t std_cfg = { .clk_cfg = I2S_STD_CLK_DEFAULT_CONFIG(EXAMPLE_SAMPLE_RATE), .slot_cfg = I2S_STD_PHILIPS_SLOT_DEFAULT_CONFIG(I2S_DATA_BIT_WIDTH_16BIT, I2S_SLOT_MODE_STEREO), .gpio_cfg = BSP_I2S_GPIO_CFG, }; std_cfg.clk_cfg.mclk_multiple = EXAMPLE_MCLK_MULTIPLE; ESP_ERROR_CHECK(bsp_audio_init(&std_cfg, &tx_handle, &rx_handle)); ESP_ERROR_CHECK(bsp_audio_poweramp_enable(true)); #endif return ESP_OK; } #if CONFIG_EXAMPLE_MODE_MUSIC static void i2s_music(void *args) { esp_err_t ret = ESP_OK; size_t bytes_read, bytes_write; FILE *mp3_file = NULL; HMP3Decoder mp3_decoder = NULL; MP3FrameInfo mp3_frame_info; int decode_ret; int bytes_left = 0; uint8_t *read_ptr = mp3_input_buffer; // 输入缓冲区指针 short pcm_output[PCM_OUTPUT_BUFFER_SIZE / 2]; // MP3Decode输出为short类型 // 初始化SD卡(SPI模式,仅使用DAT0) if (sdcard_init() != ESP_OK) { ESP_LOGE(TAG, "Failed to initialize SD card"); goto exit; } // 打开MP3文件 mp3_file = fopen(MOUNT_POINT "/music.mp3", "rb"); if (!mp3_file) { ESP_LOGE(TAG, "Failed to open MP3 file"); goto exit; } ESP_LOGI(TAG, "MP3 file opened successfully"); // 初始化MP3解码器 mp3_decoder = MP3InitDecoder(); if (!mp3_decoder) { ESP_LOGE(TAG, "Failed to initialize MP3 decoder"); goto exit; } // 预读取一部分数据到输入缓冲区 bytes_read = fread(mp3_input_buffer, 1, MP3_INPUT_BUFFER_SIZE, mp3_file); if (bytes_read == 0) { ESP_LOGE(TAG, "Failed to read MP3 file"); goto exit; } bytes_left = bytes_read; // 启用I2S发送通道 // ESP_ERROR_CHECK(i2s_channel_enable(tx_handle)); ESP_LOGI(TAG, "Starting MP3 playback"); while (1) { // 确保输入缓冲区有足够数据(至少保留一个帧的可能空间) if (bytes_left < 1024) { // 减小阈值,避免缓冲区溢出 // 移动剩余数据到缓冲区开头 memmove(mp3_input_buffer, read_ptr, bytes_left); // 从文件补充新数据 bytes_read = fread(mp3_input_buffer + bytes_left, 1, MP3_INPUT_BUFFER_SIZE - bytes_left, mp3_file); if (bytes_read == 0) { if (feof(mp3_file)) { ESP_LOGI(TAG, "End of MP3 file reached"); break; } else { ESP_LOGE(TAG, "Error reading MP3 file"); goto exit; } } bytes_left += bytes_read; read_ptr = mp3_input_buffer; // 重置指针到缓冲区开头 } // 解码MP3帧(使用MP3Decode替代MP3DecodeFrame) // 参数说明: // - 解码器句柄:mp3_decoder // - 输入缓冲区指针的指针:&read_ptr(函数会修改指针位置) // - 剩余字节数的指针:&bytes_left(函数会修改剩余字节数) // - 输出PCM缓冲区:pcm_output // - 是否使用固定大小缓冲区:0(自动适应) decode_ret = MP3Decode(mp3_decoder, &read_ptr, &bytes_left, pcm_output, 0); // 处理解码结果 if (decode_ret != ERR_MP3_NONE) { // 错误处理:打印错误信息并尝试跳过错误数据 ESP_LOGW(TAG, "MP3 decode error: %d", decode_ret); if (decode_ret == ERR_MP3_INDATA_UNDERFLOW) { // 输入数据不足,继续读取文件(循环会自动补充数据) continue; } else if (decode_ret == ERR_MP3_INVALID_FRAMEHEADER) { // 无效帧头,跳过1字节继续尝试 read_ptr++; bytes_left--; continue; } else { // 严重错误,终止播放 ESP_LOGE(TAG, "Fatal decode error: %d", decode_ret); goto exit; } } // 获取当前帧信息(采样率、声道数等) MP3GetLastFrameInfo(mp3_decoder, &mp3_frame_info); // 检查采样率是否匹配(可选,不匹配可能导致播放速度异常) if (mp3_frame_info.samprate != EXAMPLE_SAMPLE_RATE) { ESP_LOGW(TAG, "MP3 sample rate (%d) != configured rate (%d)", mp3_frame_info.samprate, EXAMPLE_SAMPLE_RATE); } // 计算PCM数据大小:单帧采样数 × 声道数 × 每个采样的字节数(16位=2字节) size_t pcm_size = mp3_frame_info.outputSamps * mp3_frame_info.nChans * sizeof(short); // 发送PCM数据到I2S(ES8311) ret = i2s_channel_write(tx_handle, pcm_output, pcm_size, &bytes_write, portMAX_DELAY); if (ret != ESP_OK) { ESP_LOGE(TAG, "[music] I2S write failed: %s", err_reason[ret == ESP_ERR_TIMEOUT]); goto exit; } if (bytes_write != pcm_size) { ESP_LOGW(TAG, "[music] Incomplete write: %d/%d bytes", bytes_write, pcm_size); } } exit: // 清理资源 if (mp3_decoder) { MP3FreeDecoder(mp3_decoder); // 使用MP3FreeDecoder释放解码器 } if (mp3_file) { fclose(mp3_file); } esp_vfs_fat_sdcard_unmount(MOUNT_POINT, NULL); ESP_LOGI(TAG, "Playback finished"); vTaskDelete(NULL); } #endif void app_main(void) { gpio_reset_pin(GPIO_NUM_3); gpio_set_direction(GPIO_NUM_3, GPIO_MODE_OUTPUT); gpio_set_level(GPIO_NUM_3, 1); printf("i2s es8311 mp3 player example start\n-----------------------------\n"); /* Initialize i2s peripheral */ if (i2s_driver_init() != ESP_OK) { ESP_LOGE(TAG, "i2s driver init failed"); abort(); } else { ESP_LOGI(TAG, "i2s driver init success"); } /* Initialize i2c peripheral and config es8311 codec by i2c */ if (es8311_codec_init() != ESP_OK) { ESP_LOGE(TAG, "es8311 codec init failed"); abort(); } else { ESP_LOGI(TAG, "es8311 codec init success"); } if(rx_handle){ i2s_channel_disable(rx_handle); } #if EXAMPLE_PA_CTRL_IO >= 0 /* Enable PA by setting the PA_CTRL_IO to high, because the power amplifier on some dev-kits are disabled by default */ gpio_config_t gpio_cfg = { .pin_bit_mask = (1ULL << EXAMPLE_PA_CTRL_IO), .mode = GPIO_MODE_OUTPUT, }; ESP_ERROR_CHECK(gpio_config(&gpio_cfg)); ESP_ERROR_CHECK(gpio_set_level(EXAMPLE_PA_CTRL_IO, 1)); #endif #if CONFIG_EXAMPLE_MODE_MUSIC /* Play MP3 from SD card */ xTaskCreate(i2s_music, "i2s_music", 2048 * 6, NULL, 8, NULL); #else /* Echo the sound from MIC in echo mode */ xTaskCreate(i2s_echo, "i2s_echo", 8192, NULL, 5, NULL); #endif } 我现在配置时双声道还是单声道
最新发布
08-05
simpleui能给个完整代码吗?另外simpleui我的自定义按钮功能我希望是放在simpleui后台管理里的booklist管理页面上的添加类别的功能。这我的表from django.db import models Create your models here. 书籍类型表 class BookType(models.Model): book_type_id = models.AutoField(primary_key=True) book_type_name = models.CharField(max_length=255) def __str__(self): return self.book_type_name class Meta: db_table = 'BookType' #书籍列表 class booklist(models.Model): id = models.AutoField(“id”, primary_key=True) bookId = models.CharField(“书籍编号”, max_length=255, default=“”) tag = models.ForeignKey( BookType, on_delete=models.CASCADE) title = models.CharField(“书名”, max_length=255, default=“”) cover = models.CharField(“封面”, max_length=2555, default=“”) author = models.CharField(“作者”, max_length=255, default=“”) press = models.CharField(“出版社”, max_length=255, default=“”) year = models.CharField(“出版年份”, max_length=255, default=“”) pageNum = models.CharField(“页码”, max_length=255, default=“”) price = models.CharField(“价格”, max_length=255, default=“”) rate = models.CharField(“评分”, max_length=255, default=“”) startList = models.CharField(“星级列表”, max_length=255, default=“”) summary = models.TextField(“描述”,default=“”) detailLink = models.CharField(“详情链接”, max_length=255, default=“”) createTime = models.CharField(“创建时间”, max_length=2555, default=“”) class Meta: db_table = "booklist" #用户表 class User(models.Model): id = models.AutoField(“id”, primary_key=True) username = models.CharField(“用户名”,max_length=255, default=“”) password = models.CharField(“密码”,max_length=255, default=“”) age = models.CharField(“年龄”, max_length=255, default=“”) gender = models.CharField(“性别”, max_length=255, default=“”) createTime = models.DateTimeField(“创建时间”,auto_now_add=True) class Meta: db_table = "user" #将来要做评分表 class Rating(models.Model): id = models.AutoField(“评分ID”, primary_key=True) user_id = models.ForeignKey(User, on_delete=models.CASCADE, verbose_name=“用户ID”) book_id = models.ForeignKey(booklist, on_delete=models.CASCADE, verbose_name=“书籍ID”) rating = models.IntegerField(“评分”, default=0) # 假设评分为1-5的整数 rating_time = models.DateTimeField(“评分时间”, auto_now_add=True) def __str__(self): return f"{self.user_id.username} - {self.book_id.title} - {self.rating}" class Meta: db_table = "rating" 收藏表 class Collection(models.Model): id = models.AutoField(“收藏ID”, primary_key=True) user_id = models.ForeignKey(User, on_delete=models.CASCADE, verbose_name=“用户ID”) book_id = models.ForeignKey(booklist, on_delete=models.CASCADE, verbose_name=“书籍ID”) collection_time = models.DateTimeField(“收藏时间”, auto_now_add=True) def __str__(self): return f"{self.user_id.username} - {self.book_id.title} - 收藏" class Meta: db_table = "collection" 公告表 class Announcement(models.Model): announcement_id = models.AutoField(primary_key=True) content = models.TextField() timestamp = models.DateTimeField(auto_now_add=True) def __str__(self): return f"公告 {self.announcement_id}" class Meta: db_table = "announcement" 评论表 class Comment(models.Model): comment_id = models.AutoField(primary_key=True) user_id = models.ForeignKey(User, on_delete=models.CASCADE) book_id = models.ForeignKey(booklist, on_delete=models.CASCADE) parent_comment_id = models.ForeignKey(‘self’, null=True, blank=True, on_delete=models.CASCADE) content = models.TextField() timestamp = models.DateTimeField(auto_now_add=True) def __str__(self): return f"评论 {self.comment_id} - 用户 {self.user_id.username} - 书籍 {self.book_id.title}" class Meta: db_table = "comment"
03-12
import os import pandas as pd import tkinter as tk from tkinter import ttk, filedialog, scrolledtext, messagebox from tkinter.colorchooser import askcolor from difflib import SequenceMatcher import re import openpyxl import threading import numpy as np from openpyxl.utils import get_column_letter import xlrd import gc import hashlib import json import tempfile from concurrent.futures import ThreadPoolExecutor, as_completed import unicodedata from datetime import datetime class EnhancedSignalComparator: def __init__(self, root): self.root = root self.root.title("增强版信号功能对比工具") self.root.geometry("1200x800") self.root.configure(bg="#f0f0f0") # 初始化变量 self.folder_path = tk.StringVar() self.search_text = tk.StringVar() self.files = [] self.results = {} # 存储信号对比结果 self.highlight_color = "#FFD700" # 默认高亮色 self.search_running = False self.stop_requested = False self.cache_dir = os.path.join(tempfile.gettempdir(), "excel_cache") self.file_cache = {} # 文件缓存 self.column_cache = {} # 列名缓存 self.max_workers = 4 # 最大并发线程数 # 创建缓存目录 os.makedirs(self.cache_dir, exist_ok=True) # 创建界面 self.create_widgets() self.log_file = "comparator.log" self.setup_logging() def setup_logging(self): """初始化日志系统""" with open(self.log_file, "w", encoding="utf-8") as log_file: log_file.write(f"{datetime.now().isoformat()} - 日志初始化\n") def log(self, message): """记录日志""" timestamp = datetime.now().isoformat() log_entry = f"{timestamp} - {message}\n" # 文件记录 with open(self.log_file, "a", encoding="utf-8") as log_file: log_file.write(log_entry) # 状态栏显示(缩短版本) if len(message) > 60: self.status_var.set(message[:57] + "...") else: self.status_var.set(message) def create_widgets(self): # 顶部控制面板 control_frame = ttk.Frame(self.root, padding=10) control_frame.pack(fill=tk.X) # 文件夹选择 ttk.Label(control_frame, text="选择文件夹:").grid(row=0, column=0, sticky=tk.W) folder_entry = ttk.Entry(control_frame, textvariable=self.folder_path, width=50) folder_entry.grid(row=0, column=1, padx=5, sticky=tk.EW) ttk.Button(control_frame, text="浏览...", command=self.browse_folder).grid(row=0, column=2) # 搜索输入 ttk.Label(control_frame, text="搜索信号:").grid(row=1, column=0, sticky=tk.W, pady=(10,0)) search_entry = ttk.Entry(control_frame, textvariable=self.search_text, width=50) search_entry.grid(row=1, column=1, padx=5, pady=(10,0), sticky=tk.EW) search_entry.bind("<Return>", lambda event: self.start_search_thread()) ttk.Button(control_frame, text="搜索", command=self.start_search_thread).grid(row=1, column=2, pady=(10,0)) ttk.Button(control_frame, text="停止", command=self.stop_search).grid(row=1, column=3, pady=(10,0), padx=5) # 高级选项 ttk.Label(control_frame, text="并发线程:").grid(row=2, column=0, sticky=tk.W, pady=(10,0)) self.thread_var = tk.StringVar(value="4") ttk.Combobox(control_frame, textvariable=self.thread_var, values=["1", "2", "4", "8"], width=5).grid(row=2, column=1, sticky=tk.W, padx=5, pady=(10,0)) # 文件过滤 ttk.Label(control_frame, text="文件过滤:").grid(row=2, column=2, sticky=tk.W, pady=(10,0)) self.filter_var = tk.StringVar(value="*.xlsx;*.xlsm;*.xls") ttk.Entry(control_frame, textvariable=self.filter_var, width=20).grid(row=2, column=3, sticky=tk.W, padx=5, pady=(10,0)) # 精确匹配选项 self.exact_match_var = tk.BooleanVar(value=False) exact_match_check = ttk.Checkbutton( control_frame, text="精确匹配", variable=self.exact_match_var ) exact_match_check.grid(row=2, column=4, sticky=tk.W, padx=5, pady=(10,0)) # 高亮颜色选择 ttk.Label(control_frame, text="高亮颜色:").grid(row=3, column=0, sticky=tk.W, pady=(10,0)) self.color_btn = tk.Button(control_frame, bg=self.highlight_color, width=3, command=self.choose_color) self.color_btn.grid(row=3, column=1, sticky=tk.W, padx=5, pady=(10,0)) # 进度条 self.progress = ttk.Progressbar(control_frame, orient="horizontal", length=200, mode="determinate") self.progress.grid(row=3, column=2, columnspan=2, sticky=tk.EW, padx=5, pady=(10,0)) # 结果标签 self.result_label = ttk.Label(control_frame, text="") self.result_label.grid(row=3, column=4, sticky=tk.W, padx=5, pady=(10,0)) # 对比面板 notebook = ttk.Notebook(self.root) notebook.pack(fill=tk.BOTH, expand=True, padx=10, pady=10) # 表格视图 self.table_frame = ttk.Frame(notebook) notebook.add(self.table_frame, text="表格视图") # 文本对比视图 self.text_frame = ttk.Frame(notebook) notebook.add(self.text_frame, text="行内容对比") # 状态栏 self.status_var = tk.StringVar() status_bar = ttk.Label(self.root, textvariable=self.status_var, relief=tk.SUNKEN, anchor=tk.W) status_bar.pack(side=tk.BOTTOM, fill=tk.X) # 初始化表格和文本区域 self.init_table_view() self.init_text_view() def init_table_view(self): """初始化表格视图""" # 创建树状表格 columns = ("信号", "文件", "行内容摘要") self.tree = ttk.Treeview(self.table_frame, columns=columns, show="headings") # 设置列标题 for col in columns: self.tree.heading(col, text=col) self.tree.column(col, width=200, anchor=tk.W) # 添加滚动条 scrollbar = ttk.Scrollbar(self.table_frame, orient=tk.VERTICAL, command=self.tree.yview) self.tree.configure(yscrollcommand=scrollbar.set) self.tree.pack(side=tk.LEFT, fill=tk.BOTH, expand=True) scrollbar.pack(side=tk.RIGHT, fill=tk.Y) # 绑定选择事件 self.tree.bind("<<TreeviewSelect>>", self.on_table_select) def init_text_view(self): """初始化文本对比视图""" self.text_panes = {} self.text_frame.columnconfigure(0, weight=1) self.text_frame.rowconfigure(0, weight=1) # 创建对比容器 self.compare_container = ttk.Frame(self.text_frame) self.compare_container.grid(row=0, column=0, sticky="nsew", padx=5, pady=5) # 添加差异高亮按钮 btn_frame = ttk.Frame(self.text_frame) btn_frame.grid(row=1, column=0, sticky="ew", padx=5, pady=5) ttk.Button(btn_frame, text="高亮显示差异", command=self.highlight_differences).pack(side=tk.LEFT) ttk.Button(btn_frame, text="导出差异报告", command=self.export_report).pack(side=tk.LEFT, padx=5) ttk.Button(btn_frame, text="清除缓存", command=self.clear_cache).pack(side=tk.LEFT, padx=5) ttk.Button(btn_frame, text="手动指定列名", command=self.manual_column_select).pack(side=tk.LEFT, padx=5) def browse_folder(self): """选择文件夹""" folder = filedialog.askdirectory(title="选择包含Excel文件的文件夹") if folder: self.folder_path.set(folder) self.load_files() def load_files(self): """加载文件夹中的Excel文件(优化特殊字符处理)""" folder = self.folder_path.get() if not folder or not os.path.isdir(folder): return # 获取文件过滤模式 filter_patterns = self.filter_var.get().split(';') self.files = [] for file in os.listdir(folder): file_path = os.path.join(folder, file) # 跳过临时文件 if file.startswith('~$'): continue # 检查文件扩展名 file_lower = file.lower() matched = False for pattern in filter_patterns: # 移除通配符并转换为小写 ext = pattern.replace('*', '').lower() if file_lower.endswith(ext): matched = True break if matched: # 规范化文件名处理特殊字符 normalized_path = self.normalize_file_path(file_path) if normalized_path and os.path.isfile(normalized_path): self.files.append(normalized_path) self.status_var.set(f"找到 {len(self.files)} 个Excel文件") def normalize_file_path(self, path): """规范化文件路径,处理特殊字符""" try: # 尝试直接访问文件 if os.path.exists(path): return path # 尝试Unicode规范化 normalized = unicodedata.normalize('NFC', path) if os.path.exists(normalized): return normalized # 尝试不同编码方案 encodings = ['utf-8', 'shift_jis', 'euc-jp', 'cp932'] for encoding in encodings: try: decoded = path.encode('latin1').decode(encoding) if os.path.exists(decoded): return decoded except: continue # 最终尝试原始路径 return path except Exception as e: self.status_var.set(f"文件路径处理错误: {str(e)}") return path def get_file_hash(self, file_path): """计算文件哈希值用于缓存""" try: hash_md5 = hashlib.md5() with open(file_path, "rb") as f: for chunk in iter(lambda: f.read(4096), b""): hash_md5.update(chunk) return hash_md5.hexdigest() except Exception as e: self.status_var.set(f"计算文件哈希失败: {str(e)}") return str(os.path.getmtime(file_path)) def get_cache_filename(self, file_path): """获取缓存文件名""" file_hash = self.get_file_hash(file_path) return os.path.join(self.cache_dir, f"{os.path.basename(file_path)}_{file_hash}.cache") def load_header_cache(self, file_path): """加载列名缓存""" cache_file = self.get_cache_filename(file_path) if os.path.exists(cache_file): try: with open(cache_file, "r", encoding='utf-8') as f: return json.load(f) except: return None return None def save_header_cache(self, file_path, header_info): """保存列名缓存""" cache_file = self.get_cache_filename(file_path) try: with open(cache_file, "w", encoding='utf-8') as f: json.dump(header_info, f) return True except: return False def find_header_row(self, file_path): """查找列名行(增强版)""" # 禁用缓存进行测试 # return None, None # 检查缓存 cache = self.load_header_cache(file_path) if cache: return cache.get("header_row"), cache.get("signal_col") # 没有缓存则重新查找 if file_path.lower().endswith((".xlsx", ".xlsm")): return self.find_header_row_openpyxl(file_path) elif file_path.lower().endswith(".xls"): return self.find_header_row_xlrd(file_path) return None, None def find_header_row_openpyxl(self, file_path): """使用openpyxl查找列名行(增强版)""" try: self.log(f"开始处理文件: {os.path.basename(file_path)}") wb = openpyxl.load_workbook(file_path, read_only=True, data_only=True) ws = wb.active # 尝试多种列名匹配模式 patterns = [ r'データ名', r'データ名', r'信号名', r'Signal Name', r'Data Name', r'信号名称', r'データ名称', r'信号', r'データー名', r'DataItem', r'Signal' # 新增模式 ] # 扩大搜索范围:前100行和前100列 for row_idx in range(1, 101): for col_idx in range(1, 101): try: cell = ws.cell(row=row_idx, column=col_idx) cell_value = cell.value if not cell_value: continue cell_str = str(cell_value) for pattern in patterns: if re.search(pattern, cell_str, re.IGNORECASE): self.log(f"找到匹配模式 '{pattern}' 在行{row_idx}列{col_idx}") # 找到列名行后,尝试确定信号列 signal_col = None # 在同行中查找信号列 for col_idx2 in range(1, 101): # 1-100列 try: cell2 = ws.cell(row=row_idx, column=col_idx2) cell2_value = cell2.value if not cell2_value: continue cell2_str = str(cell2_value) if re.search(pattern, cell2_str, re.IGNORECASE): signal_col = col_idx2 break except: continue # 保存缓存 if signal_col is not None: header_info = {"header_row": row_idx, "signal_col": signal_col} self.save_header_cache(file_path, header_info) wb.close() return row_idx, signal_col except: continue wb.close() except Exception as e: self.log(f"查找列名行出错: {str(e)}") return None, None def add_result_to_table(self, signal_value, file_name, summary): """在主线程中添加结果到表格""" if not self.root: return # 检查是否在主线程 if threading.current_thread() is not threading.main_thread(): self.root.after(0, self.add_result_to_table, signal_value, file_name, summary) return # 在主线程中添加结果 self.tree.insert("", tk.END, values=(signal_value, file_name, summary)) def find_header_row_xlrd(self, file_path): """使用xlrd查找列名行(增强版)""" try: wb = xlrd.open_workbook(file_path) ws = wb.sheet_by_index(0) # 尝试多种列名匹配模式 patterns = [ r'データ名', # 半角片假名 r'データ名', # 全角片假名 r'信号名', # 中文 r'Signal Name', # 英文 r'Data Name', r'信号名称', r'データ名称' ] # 扩大搜索范围:前100行和前100列 for row_idx in range(0, 100): # 0-99行 # 扩大列搜索范围到100列 for col_idx in range(0, 100): # 0-99列 try: cell_value = ws.cell_value(row_idx, col_idx) if not cell_value: continue # 尝试所有匹配模式 cell_str = str(cell_value) for pattern in patterns: if re.search(pattern, cell_str, re.IGNORECASE): # 找到列名行后,尝试确定信号列 signal_col = None # 在同行中查找信号列 for col_idx2 in range(0, 100): # 0-99列 try: cell2_value = ws.cell_value(row_idx, col_idx2) if not cell2_value: continue cell2_str = str(cell2_value) if re.search(pattern, cell2_str, re.IGNORECASE): signal_col = col_idx2 break except: continue # 保存缓存 if signal_col is not None: header_info = {"header_row": row_idx, "signal_col": signal_col} self.save_header_cache(file_path, header_info) return row_idx, signal_col except: continue except Exception as e: self.status_var.set(f"查找列名行出错: {str(e)}") return None, None def start_search_thread(self): """启动搜索线程""" if self.search_running: return self.search_running = True self.stop_requested = False self.max_workers = int(self.thread_var.get()) threading.Thread(target=self.search_files, daemon=True).start() def stop_search(self): """停止搜索""" self.stop_requested = True self.status_var.set("正在停止搜索...") def search_files(self): """在文件中搜索内容(优化特殊文件处理)""" search_term = self.search_text.get().strip() if not search_term: self.status_var.set("请输入搜索内容") self.search_running = False return if not self.files: self.status_var.set("请先选择文件夹") self.search_running = False return # 重置结果和UI self.results = {} for item in self.tree.get_children(): self.tree.delete(item) total_files = len(self.files) processed_files = 0 found_signals = 0 # 使用线程池处理文件 # 在search_files方法中添加详细进度 with ThreadPoolExecutor(max_workers=self.max_workers) as executor: futures = {} for i, file_path in enumerate(self.files): if self.stop_requested: break future = executor.submit(self.process_file, file_path, search_term) futures[future] = (file_path, i) # 保存文件索引 for future in as_completed(futures): if self.stop_requested: break file_path, idx = futures[future] try: found = future.result() found_signals += found processed_files += 1 # 更详细的进度反馈 progress = int(processed_files / total_files * 100) self.progress["value"] = progress self.status_var.set( f"已处理 {processed_files}/{total_files} 个文件 | " f"当前: {os.path.basename(file_path)} | " f"找到: {found_signals} 个匹配" ) self.root.update_idletasks() except Exception as e: self.status_var.set(f"处理文件 {os.path.basename(file_path)} 出错: {str(e)}") # 更新结果 if self.stop_requested: self.status_var.set(f"搜索已停止,已处理 {processed_files}/{total_files} 个文件") elif found_signals == 0: self.status_var.set(f"未找到包含 '{search_term}' 的信号") else: self.status_var.set(f"找到 {len(self.results)} 个匹配信号,共 {found_signals} 处匹配") self.update_text_view() self.progress["value"] = 0 self.search_running = False gc.collect() # 强制垃圾回收释放内存 def manual_find_header_row(self, file_path): """手动查找列名行(需要实现)""" # 这里应该实现手动查找逻辑 # 为简化问题,暂时返回默认值 return 1, 1 # 默认行1列1 def process_file(self, file_path, search_term): """处理单个文件(增强异常处理和调试)""" found = 0 short_name = os.path.basename(file_path) try: # 获取列名行和信号列 header_row, signal_col = self.find_header_row(file_path) self.log(f"文件 {short_name}: 自动查找结果 - 列名行: {header_row}, 信号列: {signal_col}") # 如果自动查找失败,尝试手动模式 if header_row is None or signal_col is None: self.log(f"文件 {short_name} 未找到列名行,尝试手动查找...") header_row, signal_col = self.manual_find_header_row(file_path) self.log(f"文件 {short_name}: 手动查找结果 - 列名行: {header_row}, 信号列: {signal_col}") if header_row is None or signal_col is None: self.log(f"文件 {short_name} 无法确定列名行,已跳过") return found # 使用pandas处理 found = self.process_file_with_pandas(file_path, search_term, header_row, signal_col) self.log(f"文件 {short_name} 处理完成,找到 {found} 个匹配") except Exception as e: error_msg = f"处理文件 {short_name} 出错: {str(e)}" self.log(error_msg) import traceback traceback.print_exc() return found def manual_column_select(self): """手动指定列名位置(增强版)""" if not self.files: messagebox.showinfo("提示", "请先选择文件夹") return # 创建手动选择窗口 manual_window = tk.Toplevel(self.root) manual_window.title("手动指定列名位置") manual_window.geometry("500x400") # 文件选择 ttk.Label(manual_window, text="选择文件:").pack(pady=(10, 5)) file_var = tk.StringVar() file_combo = ttk.Combobox(manual_window, textvariable=file_var, values=[os.path.basename(f) for f in self.files], width=40) file_combo.pack(fill=tk.X, padx=20, pady=5) file_combo.current(0) # 预览框架 preview_frame = ttk.Frame(manual_window) preview_frame.pack(fill=tk.BOTH, expand=True, padx=10, pady=10) # 表格预览 columns = ("列", "值") self.preview_tree = ttk.Treeview(preview_frame, columns=columns, show="headings", height=10) # 设置列标题 for col in columns: self.preview_tree.heading(col, text=col) self.preview_tree.column(col, width=100, anchor=tk.W) # 添加滚动条 scrollbar = ttk.Scrollbar(preview_frame, orient=tk.VERTICAL, command=self.preview_tree.yview) self.preview_tree.configure(yscrollcommand=scrollbar.set) self.preview_tree.pack(side=tk.LEFT, fill=tk.BOTH, expand=True) scrollbar.pack(side=tk.RIGHT, fill=tk.Y) # 加载预览数据 def load_preview(event=None): file_idx = file_combo.current() file_path = self.files[file_idx] # 清空现有预览 for item in self.preview_tree.get_children(): self.preview_tree.delete(item) # 加载前10行数据 try: if file_path.lower().endswith((".xlsx", ".xlsm")): wb = openpyxl.load_workbook(file_path, read_only=True, data_only=True) ws = wb.active # 读取前10行 for row_idx in range(1, 11): for col_idx in range(1, 51): # 前50列 try: cell = ws.cell(row=row_idx, column=col_idx) if cell.value is not None: self.preview_tree.insert("", tk.END, values=( f"行{row_idx}列{col_idx}", str(cell.value)[:50] # 限制显示长度 )) except: continue wb.close() elif file_path.lower().endswith(".xls"): wb = xlrd.open_workbook(file_path) ws = wb.sheet_by_index(0) # 读取前10行 for row_idx in range(0, 10): for col_idx in range(0, 50): # 前50列 try: cell_value = ws.cell_value(row_idx, col_idx) if cell_value: self.preview_tree.insert("", tk.END, values=( f"行{row_idx+1}列{col_idx+1}", str(cell_value)[:50] # 限制显示长度 )) except: continue except Exception as e: messagebox.showerror("错误", f"加载预览失败: {str(e)}") file_combo.bind("<<ComboboxSelected>>", load_preview) load_preview() # 初始加载 # 输入框架 input_frame = ttk.Frame(manual_window) input_frame.pack(fill=tk.X, padx=20, pady=10) # 行号输入 ttk.Label(input_frame, text="列名行号:").grid(row=0, column=0, sticky=tk.W) row_var = tk.StringVar(value="1") row_entry = ttk.Entry(input_frame, textvariable=row_var, width=10) row_entry.grid(row=0, column=1, padx=5) # 列号输入 ttk.Label(input_frame, text="信号列号:").grid(row=0, column=2, sticky=tk.W, padx=(10,0)) col_var = tk.StringVar(value="1") col_entry = ttk.Entry(input_frame, textvariable=col_var, width=10) col_entry.grid(row=0, column=3, padx=5) # 确认按钮 def confirm_selection(): try: file_idx = file_combo.current() file_path = self.files[file_idx] header_row = int(row_var.get()) signal_col = int(col_var.get()) # 保存到缓存 header_info = {"header_row": header_row, "signal_col": signal_col} self.save_header_cache(file_path, header_info) messagebox.showinfo("成功", f"已为 {os.path.basename(file_path)} 设置列名位置:行{header_row} 列{signal_col}") manual_window.destroy() except Exception as e: messagebox.showerror("错误", f"无效输入: {str(e)}") ttk.Button(manual_window, text="确认", command=confirm_selection).pack(pady=10) def process_file_with_pandas(self, file_path, search_term, header_row, signal_col): """使用pandas高效处理Excel文件(修复版)""" found = 0 short_name = os.path.basename(file_path) try: # 添加文件信息日志 file_size = os.path.getsize(file_path) self.log(f"处理文件: {short_name} ({file_size}字节)") # 使用pandas读取Excel文件 file_ext = os.path.splitext(file_path)[1].lower() engine = 'openpyxl' if file_ext in ['.xlsx', '.xlsm'] else 'xlrd' # 修复1: 正确计算列范围 # 计算最大可用列数 max_columns = self.get_max_columns(file_path) # 确保信号列在合理范围内 if signal_col < 1 or signal_col > max_columns: self.log(f"文件 {short_name}: 信号列{signal_col}超出范围(1-{max_columns})") return 0 # 修复2: 简化列范围计算 # 直接读取所有列,避免复杂的范围计算 df = pd.read_excel( file_path, engine=engine, header=header_row-1, # pandas的header是从0开始计数的 dtype=str ) # 修复3: 检查读取结果 if df.empty: self.log(f"文件 {short_name}: 读取到空DataFrame") return 0 # 获取列名 column_names = df.columns.tolist() # 修复4: 检查信号列索引是否有效 if signal_col-1 >= len(column_names): self.log(f"文件 {short_name}: 信号列索引{signal_col-1}超出列数{len(column_names)}") return 0 # 获取信号列名称 signal_col_name = column_names[signal_col-1] self.log(f"文件 {short_name}: 使用信号列 '{signal_col_name}' (位置 {signal_col})") # 获取信号列数据 signal_series = df.iloc[:, signal_col-1].fillna('') # 搜索匹配的信号 # 修复5: 添加精确匹配选项支持 if self.exact_match_var.get(): # 精确匹配 matches = df[signal_series.str.strip().str.lower() == search_term.lower().strip()] else: # 模糊匹配 matches = df[signal_series.str.contains( re.escape(search_term), case=False, na=False, regex=True )] # 处理匹配行 for idx, row in matches.iterrows(): # 只显示有值的列 row_content = [] for col_idx, value in enumerate(row): # 跳过空值 if pd.notna(value) and str(value).strip() != '': # 使用实际列名 if col_idx < len(column_names): col_name = column_names[col_idx] else: col_name = f"列{col_idx+1}" row_content.append(f"{col_name}: {str(value).strip()}") row_content = "\n".join(row_content) signal_value = row.iloc[signal_col-1] # 使用位置索引获取信号值 # 使用更唯一的复合键(包含行索引) signal_key = f"{signal_value}||{short_name}||{idx}" # 生成摘要 summary = row_content[:50] + "..." if len(row_content) > 50 else row_content # 添加到结果集 self.results[signal_key] = { "signal": signal_value, "file": short_name, "content": row_content, "file_path": file_path, # 添加完整路径 "row_idx": idx # 添加行索引 } # 在主线程中添加结果到表格 self.add_result_to_table(signal_value, short_name, summary) found += 1 # 每处理10行更新一次UI if found % 10 == 0: self.status_var.set(f"处理 {short_name}: 找到 {found} 个匹配") self.root.update_idletasks() # 添加完成日志 self.log(f"文件 {short_name} 处理完成: 找到 {found} 个匹配") except Exception as e: import traceback traceback.print_exc() self.log(f"处理文件 {short_name} 出错: {str(e)}") self.status_var.set(f"处理文件 {short_name} 出错: {str(e)}") finally: # 显式释放内存 if 'df' in locals(): del df if 'matches' in locals(): del matches gc.collect() return found def get_max_columns(self, file_path): """获取Excel文件的最大列数""" try: if file_path.lower().endswith((".xlsx", ".xlsm")): wb = openpyxl.load_workbook(file_path, read_only=True) ws = wb.active max_col = ws.max_column wb.close() return max_col elif file_path.lower().endswith(".xls"): wb = xlrd.open_workbook(file_path) ws = wb.sheet_by_index(0) return ws.ncols except: return 100 # 默认值 return 100 # 默认值 def update_text_view(self): """更新文本对比视图(添加空结果检查)""" # 清除现有文本区域 for widget in self.compare_container.winfo_children(): widget.destroy() if not self.results: # 添加空状态提示 empty_label = ttk.Label(self.text_frame, text="未找到匹配结果", font=("Arial", 12)) empty_label.pack(pady=20) return # 获取第一个信号作为默认显示 first_signal_key = next(iter(self.results.keys())) self.display_signal_comparison(first_signal_key) def on_table_select(self, event): """表格选择事件处理""" selected = self.tree.selection() if not selected: return item = self.tree.item(selected[0]) signal_value = item["values"][0] # 获取信号值 # 直接传递信号值给显示方法 self.display_signal_comparison(signal_value) def display_signal_comparison(self, signal_value): """显示指定信号在不同文件中的对比""" # 清除现有文本区域 for widget in self.compare_container.winfo_children(): widget.destroy() # 获取包含该信号的所有结果项 signal_items = [ (key, data) for key, data in self.results.items() if data["signal"] == signal_value ] if not signal_items: return # 按文件名排序 signal_items.sort(key=lambda x: x[1]["file"]) # 创建列框架 for i, (signal_key, signal_data) in enumerate(signal_items): col_frame = ttk.Frame(self.compare_container) col_frame.grid(row=0, column=i, sticky="nsew", padx=5, pady=5) self.compare_container.columnconfigure(i, weight=1) # 文件名标签 file_label = ttk.Label(col_frame, text=signal_data["file"], font=("Arial", 10, "bold")) file_label.pack(fill=tk.X, pady=(0, 5)) # 信号名标签 signal_label = ttk.Label(col_frame, text=signal_data["signal"], font=("Arial", 9, "italic")) signal_label.pack(fill=tk.X, pady=(0, 5)) # 文本区域 text_area = scrolledtext.ScrolledText(col_frame, wrap=tk.WORD, width=30, height=15) text_area.insert(tk.INSERT, signal_data["content"]) text_area.configure(state="disabled") text_area.pack(fill=tk.BOTH, expand=True) # 保存引用 self.text_panes[signal_key] = text_area # 添加"查看完整内容"按钮 btn_frame = ttk.Frame(col_frame) btn_frame.pack(fill=tk.X, pady=(5, 0)) ttk.Button( btn_frame, text="查看完整内容", command=lambda f=signal_data["file_path"], r=signal_data["row_idx"]: self.show_full_content(f, r) ).pack(side=tk.LEFT) def show_full_content(self, file_path, row_idx): """显示行的完整内容""" # 实现完整内容显示逻辑 self.log(f"显示完整内容: 文件={os.path.basename(file_path)}, 行={row_idx}") # [具体实现代码] def highlight_differences(self): """高亮显示文本差异""" if not self.text_panes: return # 获取所有行内容 all_contents = [] for text_area in self.text_panes.values(): text_area.configure(state="normal") text = text_area.get("1.0", tk.END).strip() text_area.configure(state="disabled") all_contents.append(text) # 如果所有内容相同,则不需要高亮 if len(set(all_contents)) == 1: self.status_var.set("所有文件行内容完全一致") return # 使用第一个文件作为基准 base_text = all_contents[0] # 对比并高亮差异 for i, (file, text_area) in enumerate(self.text_panes.items()): if i == 0: # 基准文件不需要处理 continue text_area.configure(state="normal") text_area.tag_configure("diff", background=self.highlight_color) # 清除之前的高亮 text_area.tag_remove("diff", "1.0", tk.END) # 获取当前文本 compare_text = text_area.get("1.0", tk.END).strip() # 使用序列匹配器查找差异 s = SequenceMatcher(None, base_text, compare_text) # 高亮差异部分 for tag in s.get_opcodes(): opcode = tag[0] start = tag[3] end = tag[4] if opcode != "equal": # 添加高亮标签 text_area.tag_add("diff", f"1.0+{start}c", f"1.0+{end}c") text_area.configure(state="disabled") self.status_var.set("差异已高亮显示") def choose_color(self): """选择高亮颜色""" color = askcolor(title="选择高亮颜色", initialcolor=self.highlight_color) if color[1]: self.highlight_color = color[1] self.color_btn.configure(bg=self.highlight_color) def export_report(self): """导出差异报告(修复结果集访问)""" if not self.results: messagebox.showwarning("警告", "没有可导出的结果") return try: # 创建报告数据结构(修复结果集访问方式) report_data = [] for signal_key, data in self.results.items(): report_data.append({ "信号": data["signal"], "文件": data["file"], "行内容": data["content"] }) # 转换为DataFrame df = pd.DataFrame(report_data) # 保存到Excel save_path = filedialog.asksaveasfilename( defaultextension=".xlsx", filetypes=[("Excel文件", "*.xlsx")], title="保存差异报告" ) if save_path: df.to_excel(save_path, index=False) self.status_var.set(f"报告已保存到: {save_path}") except Exception as e: messagebox.showerror("错误", f"导出报告失败: {str(e)}") def clear_cache(self): """清除缓存""" try: for file in os.listdir(self.cache_dir): if file.endswith(".cache"): os.remove(os.path.join(self.cache_dir, file)) self.file_cache = {} self.column_cache = {} self.status_var.set("缓存已清除") except Exception as e: self.status_var.set(f"清除缓存失败: {str(e)}") def manual_column_select(self): """手动指定列名位置""" if not self.files: messagebox.showinfo("提示", "请先选择文件夹") return # 创建手动选择窗口 manual_window = tk.Toplevel(self.root) manual_window.title("手动指定列名位置") manual_window.geometry("400x300") # 文件选择 ttk.Label(manual_window, text="选择文件:").pack(pady=(10, 5)) file_var = tk.StringVar() file_combo = ttk.Combobox(manual_window, textvariable=file_var, values=[os.path.basename(f) for f in self.files]) file_combo.pack(fill=tk.X, padx=20, pady=5) file_combo.current(0) # 行号输入 ttk.Label(manual_window, text="列名行号:").pack(pady=(10, 5)) row_var = tk.StringVar(value="1") row_entry = ttk.Entry(manual_window, textvariable=row_var) row_entry.pack(fill=tk.X, padx=20, pady=5) # 列号输入 ttk.Label(manual_window, text="信号列号:").pack(pady=(10, 5)) col_var = tk.StringVar(value="1") col_entry = ttk.Entry(manual_window, textvariable=col_var) col_entry.pack(fill=tk.X, padx=20, pady=5) # 确认按钮 def confirm_selection(): try: file_idx = file_combo.current() file_path = self.files[file_idx] header_row = int(row_var.get()) signal_col = int(col_var.get()) # 保存到缓存 header_info = {"header_row": header_row, "signal_col": signal_col} self.save_header_cache(file_path, header_info) messagebox.showinfo("成功", f"已为 {os.path.basename(file_path)} 设置列名位置:行{header_row} 列{signal_col}") manual_window.destroy() except Exception as e: messagebox.showerror("错误", f"无效输入: {str(e)}") ttk.Button(manual_window, text="确认", command=confirm_selection).pack(pady=20) if __name__ == "__main__": root = tk.Tk() app = EnhancedSignalComparator(root) root.mainloop() 1、将单信号搜索改为多信号搜索 2、将页面布局进行修改,以更合理的方式进行布局
07-25
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值