当前位置:首页>python>Python实现的春节对联生成器

Python实现的春节对联生成器

  • 2026-02-06 12:58:31
Python实现的春节对联生成器

用 tkinter 实现一个对联生成器,核心功能包括:上传图片作为背景、输入汉字内容、点击生成对联(叠加在背景图上)、点击保存最终成品。

实现这个完整功能,用到 tkinter 做界面,PIL(Pillow)处理图片和文字叠加,步骤清晰且代码可直接运行。

前置准备

首先安装需要的依赖库(处理图片必须用到 Pillow):

pip install pillow

完整代码实现

import tkinter as tkfrom tkinter import filedialog, messageboxfrom PIL import Image, ImageTk, ImageDraw, ImageFontimport os# 对联生成器主类classCoupletGenerator:def__init__(self, root):# 初始化主窗口        self.root = root        self.root.title("对联生成器(背景图+自定义文字)")        self.root.geometry("800x900")  # 窗口大小# 定义全局变量存储关键数据        self.bg_image_path = None# 背景图片路径        self.bg_image = None# 原始背景图片对象        self.result_image = None# 生成后的最终图片对象# 1. 创建界面组件        self._create_widgets()def_create_widgets(self):"""创建所有界面控件"""# 第一排:上传背景图片按钮        self.upload_btn = tk.Button(            self.root,            text="📷 上传背景图片",            command=self.upload_background,            font=("宋体"12),            width=20,            height=2        )        self.upload_btn.pack(pady=10)# 第二排:背景图片预览区域        self.preview_label = tk.Label(self.root, text="背景图片预览(上传后显示)", font=("宋体"10))        self.preview_label.pack(pady=5)# 限制预览区域大小,避免图片过大撑爆窗口        self.preview_canvas = tk.Canvas(self.root, width=600, height=400, bg="#f0f0f0", bd=1, relief="solid")        self.preview_canvas.pack(pady=5)# 第三排:对联文字输入区域        self.frame_input = tk.Frame(self.root)        self.frame_input.pack(pady=10)# 上联输入        tk.Label(self.frame_input, text="上联:", font=("宋体"12)).grid(row=0, column=0, padx=5, pady=5)        self.upper_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=30)        self.upper_couplet.grid(row=0, column=1, padx=5, pady=5)# 下联输入        tk.Label(self.frame_input, text="下联:", font=("宋体"12)).grid(row=1, column=0, padx=5, pady=5)        self.lower_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=30)        self.lower_couplet.grid(row=1, column=1, padx=5, pady=5)# 横批输入        tk.Label(self.frame_input, text="横批:", font=("宋体"12)).grid(row=2, column=0, padx=5, pady=5)        self.horizontal_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=30)        self.horizontal_couplet.grid(row=2, column=1, padx=5, pady=5)# 第四排:功能按钮(生成+保存)        self.frame_btn = tk.Frame(self.root)        self.frame_btn.pack(pady=20)        self.generate_btn = tk.Button(            self.frame_btn,            text="✨ 生成对联",            command=self.generate_couplet,            font=("宋体"12),            width=15,            bg="#4CAF50",            fg="white"        )        self.generate_btn.grid(row=0, column=0, padx=20)        self.save_btn = tk.Button(            self.frame_btn,            text="💾 保存成品",            command=self.save_couplet,            font=("宋体"12),            width=15,            bg="#2196F3",            fg="white"        )        self.save_btn.grid(row=0, column=1, padx=20)defupload_background(self):"""上传背景图片并显示预览"""# 打开文件选择对话框,限定图片格式        file_path = filedialog.askopenfilename(            title="选择背景图片",            filetypes=[("图片文件""*.jpg *.jpeg *.png *.bmp"), ("所有文件""*.*")]        )ifnot file_path:  # 用户取消选择return# 存储背景图片路径        self.bg_image_path = file_pathtry:# 打开图片并缩放(适配预览画布大小,保持宽高比)            self.bg_image = Image.open(file_path)            self.bg_image.thumbnail((600400), Image.Resampling.LANCZOS)  # 高质量缩放# 转换为tkinter可显示的格式            tk_image = ImageTk.PhotoImage(self.bg_image)# 更新预览画布            self.preview_canvas.delete("all")  # 清空原有内容            self.preview_canvas.create_image(300200, image=tk_image)  # 居中显示            self.preview_canvas.tk_image = tk_image  # 保存引用,防止被垃圾回收# 更新预览标签文字            self.preview_label.config(text=f"已上传:{os.path.basename(file_path)}")except Exception as e:            messagebox.showerror("错误"f"图片打开失败:{str(e)}")            self.bg_image_path = None            self.bg_image = Nonedefgenerate_couplet(self):"""生成对联(将文字叠加到背景图片上)"""# 校验前置条件ifnot self.bg_image_path ornot self.bg_image:            messagebox.warning("提示""请先上传背景图片!")return        upper_text = self.upper_couplet.get().strip()        lower_text = self.lower_couplet.get().strip()        horizontal_text = self.horizontal_couplet.get().strip()ifnot (upper_text or lower_text or horizontal_text):            messagebox.warning("提示""请至少输入一项对联内容!")returntry:# 1. 重新打开原始背景图片(避免预览缩放影响最终成品)            original_bg = Image.open(self.bg_image_path).convert("RGB")  # 转为RGB格式,避免透明通道问题            draw = ImageDraw.Draw(original_bg)  # 创建绘图对象# 2. 设置字体(关键:支持中文显示,这里使用系统自带宋体,可替换为自定义字体文件)# 如果你有自定义中文字体(如simhei.ttf),可以替换为字体文件路径try:# Windows 系统默认宋体路径                font = ImageFont.truetype("simsun.ttc"48)  # 48为字体大小except:try:# Mac 系统默认中文字体路径                    font = ImageFont.truetype("STHeitiLight.ttc"48)except:# 无自定义字体时,使用默认字体(可能不支持中文)                    font = ImageFont.load_default(size=48)                    messagebox.warning("提示""未找到系统中文字体,生成的文字可能无法正常显示!")# 3. 获取背景图片尺寸,用于计算文字位置(居中/对称排列)            bg_width, bg_height = original_bg.size# 4. 绘制上联(左侧垂直排列,可根据需求调整位置)if upper_text:                upper_x = bg_width * 0.2# 上联x坐标(左侧20%位置)                upper_y_step = 60# 每个字的垂直间距                start_upper_y = (bg_height - len(upper_text) * upper_y_step) / 2# 垂直居中for i, char in enumerate(upper_text):                    draw.text(                        (upper_x, start_upper_y + i * upper_y_step),                        char,                        font=font,                        fill=(25500)  # 文字颜色:红色                    )# 5. 绘制下联(右侧垂直排列)if lower_text:                lower_x = bg_width * 0.8# 下联x坐标(右侧80%位置)                lower_y_step = 60# 每个字的垂直间距                start_lower_y = (bg_height - len(lower_text) * lower_y_step) / 2# 垂直居中for i, char in enumerate(lower_text):                    draw.text(                        (lower_x, start_lower_y + i * lower_y_step),                        char,                        font=font,                        fill=(25500)  # 文字颜色:红色                    )# 6. 绘制横批(顶部水平居中排列)if horizontal_text:# 计算横批文字宽度,实现水平居中                text_bbox = draw.textbbox((00), horizontal_text, font=font)                text_width = text_bbox[2] - text_bbox[0]                horizontal_x = (bg_width - text_width) / 2                horizontal_y = bg_height * 0.1# 横批y坐标(顶部10%位置)                draw.text(                    (horizontal_x, horizontal_y),                    horizontal_text,                    font=font,                    fill=(25500)  # 文字颜色:红色                )# 7. 存储生成后的图片,并更新预览            self.result_image = original_bg# 缩放结果图用于预览            preview_result = self.result_image.copy()            preview_result.thumbnail((600400), Image.Resampling.LANCZOS)            tk_result = ImageTk.PhotoImage(preview_result)            self.preview_canvas.delete("all")            self.preview_canvas.create_image(300200, image=tk_result)            self.preview_canvas.tk_image = tk_result            messagebox.showinfo("成功""对联生成完成!可点击保存按钮导出成品。")except Exception as e:            messagebox.showerror("错误"f"对联生成失败:{str(e)}")            self.result_image = Nonedefsave_couplet(self):"""保存生成后的对联图片"""ifnot self.result_image:            messagebox.warning("提示""请先生成对联!")return# 打开保存文件对话框,默认格式为PNG        save_path = filedialog.asksaveasfilename(            title="保存对联图片",            defaultextension=".png",            filetypes=[("PNG图片""*.png"), ("JPG图片""*.jpg"), ("所有文件""*.*")]        )ifnot save_path:  # 用户取消保存returntry:            self.result_image.save(save_path)            messagebox.showinfo("成功"f"对联已保存至:\n{save_path}")except Exception as e:            messagebox.showerror("错误"f"图片保存失败:{str(e)}")# 程序入口if __name__ == "__main__":    root = tk.Tk()    app = CoupletGenerator(root)    root.mainloop()

功能说明与关键细节

  1. 界面布局:分为4个区域(上传按钮、图片预览、文字输入、功能按钮),布局简洁美观,适配新手操作。
  2. 图片处理
    • 支持上传 jpg/jpeg/png/bmp 格式图片,预览时自动缩放到合适大小,不撑爆窗口。
    • 生成对联时使用原始图片,保证成品清晰度,避免预览缩放的画质损失。
    • 最终保存支持选择 PNG/JPG 格式,自定义保存路径。
  3. 中文支持:关键是使用 ImageFont.truetype() 加载系统中文字体(Windows 默认为 simsun.ttc,Mac 默认为 STHeitiLight.ttc),解决 tkinter + PIL 中文乱码问题。
    • 如果你需要自定义字体(如更美观的书法字体),只需下载 .ttf/.ttc 格式中文字体,将 ImageFont.truetype() 中的路径替换为你的字体文件路径即可。
  4. 对联排列
    • 上联:左侧20%位置,垂直居中排列(竖排,符合对联传统格式)。
    • 下联:右侧80%位置,垂直居中排列(竖排)。
    • 横批:顶部10%位置,水平居中排列(横排)。
    • 文字颜色为红色,符合对联的视觉习惯,字体大小为48,可根据需求调整。
  5. 异常处理:对所有关键步骤(图片打开、文字生成、图片保存)添加了异常捕获,并用 messagebox 给出清晰提示,避免程序崩溃。

使用步骤

  1. 运行上述代码,打开对联生成器窗口。
  2. 点击「📷 上传背景图片」,选择一张你喜欢的图片(建议选择浅色、留白较多的图片,对联文字更清晰)。
  3. 在输入框中填写「上联」「下联」「横批」(可填部分内容,不强制全部填写)。
  4. 点击「✨ 生成对联」,等待片刻,预览区域会显示叠加文字后的成品。
  5. 点击「💾 保存成品」,选择保存路径和格式,完成保存。

可优化扩展方向(可选)

  1. 增加文字颜色、字体大小、排列位置的自定义设置。
  2. 增加对联文字的旋转、阴影效果,提升美观度。
  3. 支持批量生成、模板选择(如春联、婚联模板)。
  4. 增加预览区域的放大、缩小功能。

总结

  1. 该程序基于 tkinter 构建界面,Pillow 处理图片与文字叠加,实现了对联生成器的核心需求,且代码可直接运行。
  2. 关键要点是解决中文显示问题(加载中文字体)、保证图片成品清晰度(使用原始图片生成)、提供友好的用户交互(异常提示、保存引导)。
  3. 支持背景图上传、文字输入、对联生成、成品保存四大核心功能,符合传统对联的排版习惯,可直接用于日常使用。

问题分析

  1. 字体路径问题(最常见):系统找不到指定的中文字体,导致文字无法绘制,看似“没有显示”。
  2. 文字坐标偏移:部分图片尺寸特殊,文字可能超出图片可视范围。
  3. 代码逻辑遗漏:确认随机生成的内容是否正确传递到绘制函数。

修复后的完整代码(保证可运行)

import tkinter as tkfrom tkinter import filedialog, messageboxfrom PIL import Image, ImageTk, ImageDraw, ImageFontimport osimport random# 对联词库UPPER_LINES = ["春风入喜财入户","岁月更新福满门","一帆风顺年年好","万事如意步步高","红梅含苞傲冬雪"]LOWER_LINES = ["岁月更新福满门","春风入喜财入户","万事如意步步高","一帆风顺年年好","绿柳吐絮迎新春"]HORIZONTAL_LINES = ["新春大吉","万事如意","四季平安","五福临门","恭喜发财"]classCoupletGenerator:def__init__(self, root):        self.root = root        self.root.title("对联生成器(背景图+随机/自定义)")        self.root.geometry("800x950")        self.bg_image_path = None        self.bg_image = None        self.result_image = None        self._create_widgets()def_create_widgets(self):# 上传背景图        self.upload_btn = tk.Button(            self.root, text="📷 上传背景图片", command=self.upload_background,            font=("宋体"12), width=20, height=2        )        self.upload_btn.pack(pady=10)# 预览        self.preview_label = tk.Label(self.root, text="背景图片预览(上传后显示)", font=("宋体"10))        self.preview_label.pack(pady=5)        self.preview_canvas = tk.Canvas(self.root, width=600, height=400, bg="#f0f0f0", bd=1, relief="solid")        self.preview_canvas.pack(pady=5)# 输入区        self.frame_input = tk.Frame(self.root)        self.frame_input.pack(pady=10)        tk.Label(self.frame_input, text="上联:", font=("宋体"12)).grid(row=0, column=0, padx=5, pady=5)        self.upper_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=30)        self.upper_couplet.grid(row=0, column=1, padx=5, pady=5)        tk.Label(self.frame_input, text="下联:", font=("宋体"12)).grid(row=1, column=0, padx=5, pady=5)        self.lower_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=30)        self.lower_couplet.grid(row=1, column=1, padx=5, pady=5)        tk.Label(self.frame_input, text="横批:", font=("宋体"12)).grid(row=2, column=0, padx=5, pady=5)        self.horizontal_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=30)        self.horizontal_couplet.grid(row=2, column=1, padx=5, pady=5)# 随机对联按钮        self.random_btn = tk.Button(            self.frame_input, text="🎲 随机生成对联", command=self.random_couplet,            font=("宋体"12), bg="#FF9800", fg="white"        )        self.random_btn.grid(row=3, column=0, columnspan=2, pady=10)# 生成/保存        self.frame_btn = tk.Frame(self.root)        self.frame_btn.pack(pady=20)        self.generate_btn = tk.Button(            self.frame_btn, text="✨ 生成对联", command=self.generate_couplet,            font=("宋体"12), width=15, bg="#4CAF50", fg="white"        )        self.generate_btn.grid(row=0, column=0, padx=20)        self.save_btn = tk.Button(            self.frame_btn, text="💾 保存成品", command=self.save_couplet,            font=("宋体"12), width=15, bg="#2196F3", fg="white"        )        self.save_btn.grid(row=0, column=1, padx=20)defupload_background(self):        file_path = filedialog.askopenfilename(            title="选择背景图片",            filetypes=[("图片文件""*.jpg *.jpeg *.png *.bmp"), ("所有文件""*.*")]        )ifnot file_path:return        self.bg_image_path = file_pathtry:            self.bg_image = Image.open(file_path)            self.bg_image.thumbnail((600400), Image.Resampling.LANCZOS)            tk_image = ImageTk.PhotoImage(self.bg_image)            self.preview_canvas.delete("all")            self.preview_canvas.create_image(300200, image=tk_image)            self.preview_canvas.tk_image = tk_image            self.preview_label.config(text=f"已上传:{os.path.basename(file_path)}")except Exception as e:            messagebox.showerror("错误"f"图片打开失败:{str(e)}")            self.bg_image_path = None            self.bg_image = Nonedefrandom_couplet(self):"""随机生成对联并填入输入框"""        upper = random.choice(UPPER_LINES)        lower = random.choice(LOWER_LINES)        horizontal = random.choice(HORIZONTAL_LINES)# 清空输入框并填入随机内容(确保完整覆盖)        self.upper_couplet.delete(0, tk.END)        self.upper_couplet.insert(0, upper)        self.lower_couplet.delete(0, tk.END)        self.lower_couplet.insert(0, lower)        self.horizontal_couplet.delete(0, tk.END)        self.horizontal_couplet.insert(0, horizontal)defgenerate_couplet(self):"""生成对联(核心修复:字体+文字坐标+调试信息)"""ifnot self.bg_image_path ornot self.bg_image:            messagebox.showwarning("提示""请先上传背景图片!")return        upper_text = self.upper_couplet.get().strip()        lower_text = self.lower_couplet.get().strip()        horizontal_text = self.horizontal_couplet.get().strip()ifnot (upper_text or lower_text or horizontal_text):            messagebox.showwarning("提示""请至少输入一项对联内容!")returntry:# 1. 打开原始背景图(保证清晰度)            original_bg = Image.open(self.bg_image_path).convert("RGB")            draw = ImageDraw.Draw(original_bg)            bg_w, bg_h = original_bg.size# 2. 修复字体问题(优先级:自定义字体 > 系统字体 > 默认字体,增加字体大小适配)            font = None            font_size = min(48, int(bg_w/20), int(bg_h/20))  # 自适应图片尺寸,避免文字过大try:# Windows 系统宋体(兼容大多数环境)                font = ImageFont.truetype("simsun.ttc", font_size)except Exception as e1:try:# Mac 系统中文字体                    font = ImageFont.truetype("STHeitiLight.ttc", font_size)except Exception as e2:try:# Linux 系统中文字体                        font = ImageFont.truetype("wqy-microhei.ttc", font_size)except:# 最终兜底:默认字体(虽不支持中文,但提示用户)                        font = ImageFont.load_default(size=font_size)                        messagebox.showwarning("提示"f"未找到中文字体:{e1}/{e2},文字可能无法正常显示!")# 3. 修复文字坐标(避免超出图片范围,增加偏移量适配)            text_color = (25500)  # 红色(醒目,确保可见)            step = font_size + 12# 字间距(适配字体大小)# 上联(左竖排:调整x坐标,避免贴边)if upper_text:                x = max(50, int(bg_w * 0.1))  # 左侧至少留50像素边距                start_y = (bg_h - len(upper_text) * step) / 2# 确保文字在图片内                start_y = max(50, min(start_y, bg_h - len(upper_text) * step - 50))for i, c in enumerate(upper_text):                    draw.text((x, start_y + i * step), c, font=font, fill=text_color, stroke_width=1, stroke_fill=(0,0,0))# 下联(右竖排:调整x坐标,避免贴边)if lower_text:                x = min(bg_w - 50, int(bg_w * 0.9))  # 右侧至少留50像素边距                start_y = (bg_h - len(lower_text) * step) / 2                start_y = max(50, min(start_y, bg_h - len(lower_text) * step - 50))for i, c in enumerate(lower_text):                    draw.text((x, start_y + i * step), c, font=font, fill=text_color, stroke_width=1, stroke_fill=(0,0,0))# 横批(上中横排:居中对齐,增加描边更醒目)if horizontal_text:# 计算文字宽度,实现精准居中                bbox = draw.textbbox((00), horizontal_text, font=font)                text_w = bbox[2] - bbox[0]                x = (bg_w - text_w) / 2                y = max(30, int(bg_h * 0.05))  # 顶部至少留30像素边距                draw.text((x, y), horizontal_text, font=font, fill=text_color, stroke_width=1, stroke_fill=(0,0,0))# 4. 保存结果并更新预览            self.result_image = original_bg# 缩放预览(适配画布)            preview = self.result_image.copy()            preview.thumbnail((600400), Image.Resampling.LANCZOS)            tk_img = ImageTk.PhotoImage(preview)            self.preview_canvas.delete("all")            self.preview_canvas.create_image(300200, image=tk_img)            self.preview_canvas.tk_image = tk_img  # 保存引用,防止被回收            messagebox.showinfo("成功"f"对联生成完成!\n上联:{upper_text}\n下联:{lower_text}\n横批:{horizontal_text}")except Exception as e:            messagebox.showerror("错误"f"生成失败:{str(e)}")            self.result_image = Nonedefsave_couplet(self):ifnot self.result_image:            messagebox.showwarning("提示""请先生成对联!")return        save_path = filedialog.asksaveasfilename(            title="保存对联", defaultextension=".png",            filetypes=[("PNG图片""*.png"), ("JPG图片""*.jpg"), ("所有文件""*.*")]        )ifnot save_path:returntry:            self.result_image.save(save_path)            messagebox.showinfo("成功"f"已保存至:\n{save_path}")except Exception as e:            messagebox.showerror("错误"f"保存失败:{str(e)}")if __name__ == "__main__":    root = tk.Tk()    app = CoupletGenerator(root)    root.mainloop()

关键修复点(确保文字显示)

  1. 字体适配优化

    • 增加 Linux 系统字体(wqy-microhei.ttc)支持,覆盖更多操作系统。
    • 字体大小自适应图片尺寸,避免文字过大超出图片范围。
    • 增加文字描边效果stroke_width=1, stroke_fill=(0,0,0)),即使背景颜色较红,文字也能清晰可见。
  2. 文字坐标安全校验

    • 上下联左右两侧至少留 50 像素边距,横批顶部留 30 像素边距。
    • 限制文字起始 y 坐标,确保不会超出图片上下边界。
    • 字间距(step)适配字体大小,避免文字重叠。
  3. 输入框内容确保完整

    • 随机生成对联时,明确 delete(0, tk.END) 清空输入框,再 insert(0, 内容),确保内容完整传递。
    • 生成时增加调试信息,弹窗显示当前要绘制的对联内容,确认数据未丢失。

测试步骤(确保成功)

  1. 运行代码,上传一张浅色、留白较多的图片(推荐白底或浅红色背景,对比更明显)。
  2. 点击「🎲 随机生成对联」,确认输入框已填入内容。
  3. 点击「✨ 生成对联」,此时预览区应能看到红色带黑边的对联文字。
  4. 点击「💾 保存成品」,查看保存的图片,文字应完整清晰。

补充说明

如果仍然无法显示文字,大概率是系统缺少对应的中文字体文件,解决方案:

  1. 下载一个中文字体文件(如 simhei.ttf 黑体、simsun.ttc 宋体)。
  2. 将字体文件放在代码同一目录下。
  3. 修改代码中 ImageFont.truetype() 的路径为:font = ImageFont.truetype("simhei.ttf", font_size)(替换为你的字体文件名)。

总结

  1. 核心问题是字体适配不足和文字坐标可能超出图片范围,修复后可正常显示随机对联。
  2. 关键优化点包括多系统字体兼容、文字自适应尺寸、描边增强可见性。
  3. 若仍无文字显示,需手动引入自定义中文字体文件,确保 PIL 能加载并绘制中文。

如果你希望在原有对联生成器基础上,增加字体大小选择功能,可以自定义调整对联文字的大小,适配不同尺寸的背景图。

在修复版代码的基础上,添加字体大小下拉选择框(可选16-128号字体,步长8),并让文字间距、边距随字体大小自动适配,保证排版美观,以下是完整可运行的最终版代码

完整代码(含字体大小选择+随机对联+所有修复)

import tkinter as tkfrom tkinter import filedialog, messageboxfrom PIL import Image, ImageTk, ImageDraw, ImageFontimport osimport random# 对联词库(可自行扩展)UPPER_LINES = ["春风入喜财入户""岁月更新福满门""一帆风顺年年好""万事如意步步高","红梅含苞傲冬雪""绿柳吐絮迎新春""迎喜迎春迎富贵""接财接福接平安"]LOWER_LINES = ["岁月更新福满门""春风入喜财入户""万事如意步步高""一帆风顺年年好","绿柳吐絮迎新春""红梅含苞傲冬雪""接财接福接平安""迎喜迎春迎富贵"]HORIZONTAL_LINES = ["新春大吉""万事如意""四季平安""五福临门""恭喜发财","财源广进""吉星高照""合家欢乐""迎春接福"]classCoupletGenerator:def__init__(self, root):        self.root = root        self.root.title("对联生成器(背景图+随机/自定义+字体大小选择)")        self.root.geometry("800x1000")  # 微调窗口高度适配新控件# 核心变量        self.bg_image_path = None        self.bg_image = None        self.result_image = None        self.default_font_size = 48# 默认字体大小        self._create_widgets()def_create_widgets(self):"""创建所有界面控件(新增字体大小选择)"""# 1. 上传背景图片按钮        self.upload_btn = tk.Button(            self.root, text="📷 上传背景图片", command=self.upload_background,            font=("宋体"12), width=20, height=2        )        self.upload_btn.pack(pady=10)# 2. 背景图片预览区域        self.preview_label = tk.Label(self.root, text="背景图片预览(上传后显示)", font=("宋体"10))        self.preview_label.pack(pady=5)        self.preview_canvas = tk.Canvas(self.root, width=600, height=400, bg="#f0f0f0", bd=1, relief="solid")        self.preview_canvas.pack(pady=5)# 3. 对联文字输入+字体大小选择区域        self.frame_input = tk.Frame(self.root)        self.frame_input.pack(pady=10)# 3.1 对联输入(3行)        tk.Label(self.frame_input, text="上联:", font=("宋体"12)).grid(row=0, column=0, padx=5, pady=5)        self.upper_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=25)        self.upper_couplet.grid(row=0, column=1, padx=5, pady=5)        tk.Label(self.frame_input, text="下联:", font=("宋体"12)).grid(row=1, column=0, padx=5, pady=5)        self.lower_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=25)        self.lower_couplet.grid(row=1, column=1, padx=5, pady=5)        tk.Label(self.frame_input, text="横批:", font=("宋体"12)).grid(row=2, column=0, padx=5, pady=5)        self.horizontal_couplet = tk.Entry(self.frame_input, font=("宋体"14), width=25)        self.horizontal_couplet.grid(row=2, column=1, padx=5, pady=5)# 3.2 字体大小选择(新增:下拉框+标签,和输入框同排)        tk.Label(self.frame_input, text="字体大小:", font=("宋体"12)).grid(row=1, column=2, padx=10, pady=5)# 字体大小选项:16-128,步长8,覆盖小/中/大尺寸需求        self.font_size_options = [16,24,32,40,48,56,64,72,80,88,96,104,112,128]        self.font_size_var = tk.StringVar(value=str(self.default_font_size))  # 默认选中48        self.font_size_combobox = tk.OptionMenu(            self.frame_input, self.font_size_var, *self.font_size_options        )        self.font_size_combobox.config(font=("宋体"12), width=6)        self.font_size_combobox.grid(row=1, column=3, padx=5, pady=5)# 4. 随机生成对联按钮        self.random_btn = tk.Button(            self.frame_input, text="🎲 随机生成对联", command=self.random_couplet,            font=("宋体"12), bg="#FF9800", fg="white"        )        self.random_btn.grid(row=3, column=0, columnspan=4, pady=10)# 5. 生成+保存功能按钮        self.frame_btn = tk.Frame(self.root)        self.frame_btn.pack(pady=20)        self.generate_btn = tk.Button(            self.frame_btn, text="✨ 生成对联", command=self.generate_couplet,            font=("宋体"12), width=15, bg="#4CAF50", fg="white"        )        self.generate_btn.grid(row=0, column=0, padx=20)        self.save_btn = tk.Button(            self.frame_btn, text="💾 保存成品", command=self.save_couplet,            font=("宋体"12), width=15, bg="#2196F3", fg="white"        )        self.save_btn.grid(row=0, column=1, padx=20)defupload_background(self):"""上传背景图片并显示预览"""        file_path = filedialog.askopenfilename(            title="选择背景图片",            filetypes=[("图片文件""*.jpg *.jpeg *.png *.bmp"), ("所有文件""*.*")]        )ifnot file_path:return        self.bg_image_path = file_pathtry:            self.bg_image = Image.open(file_path)            self.bg_image.thumbnail((600400), Image.Resampling.LANCZOS)            tk_image = ImageTk.PhotoImage(self.bg_image)            self.preview_canvas.delete("all")            self.preview_canvas.create_image(300200, image=tk_image)            self.preview_canvas.tk_image = tk_image            self.preview_label.config(text=f"已上传:{os.path.basename(file_path)}")except Exception as e:            messagebox.showerror("错误"f"图片打开失败:{str(e)}")            self.bg_image_path = None            self.bg_image = Nonedefrandom_couplet(self):"""随机生成对联并填入输入框"""        upper = random.choice(UPPER_LINES)        lower = random.choice(LOWER_LINES)        horizontal = random.choice(HORIZONTAL_LINES)# 清空并填入,确保无残留内容        self.upper_couplet.delete(0, tk.END)        self.upper_couplet.insert(0, upper)        self.lower_couplet.delete(0, tk.END)        self.lower_couplet.insert(0, lower)        self.horizontal_couplet.delete(0, tk.END)        self.horizontal_couplet.insert(0, horizontal)defgenerate_couplet(self):"""生成对联(核心:读取选择的字体大小,自动适配排版)"""# 前置校验ifnot self.bg_image_path ornot self.bg_image:            messagebox.showwarning("提示""请先上传背景图片!")return        upper_text = self.upper_couplet.get().strip()        lower_text = self.lower_couplet.get().strip()        horizontal_text = self.horizontal_couplet.get().strip()ifnot (upper_text or lower_text or horizontal_text):            messagebox.showwarning("提示""请至少输入一项对联内容!")returntry:# 1. 读取并校验选择的字体大小(防止非法输入)try:                font_size = int(self.font_size_var.get())except:                font_size = self.default_font_size                messagebox.showwarning("提示""字体大小选择异常,使用默认48号字体")# 2. 打开原始背景图,创建绘图对象            original_bg = Image.open(self.bg_image_path).convert("RGB")            draw = ImageDraw.Draw(original_bg)            bg_w, bg_h = original_bg.size# 3. 多系统字体兼容加载            font = Nonetry:                font = ImageFont.truetype("simsun.ttc", font_size)  # Windows 宋体except Exception as e1:try:                    font = ImageFont.truetype("STHeitiLight.ttc", font_size)  # Mac 黑体except Exception as e2:try:                        font = ImageFont.truetype("wqy-microhei.ttc", font_size)  # Linux 文泉驿except:                        font = ImageFont.load_default(size=font_size)                        messagebox.showwarning("提示"f"未找到中文字体:{e1}/{e2},文字可能显示异常!")# 4. 排版参数(随字体大小自动适配,无需手动调整)            text_color = (25500)  # 对联经典红色            step = font_size + 8# 字间距:字体越大,间距越大            stroke_w = max(1, font_size // 24)  # 描边宽度:随字体大小适配,更美观# 5. 绘制上联(左竖排,防超出边界)if upper_text:                x = max(30, int(bg_w * 0.1))  # 左侧最小边距30                start_y = (bg_h - len(upper_text) * step) / 2                start_y = max(30, min(start_y, bg_h - len(upper_text) * step - 30))  # 上下防超出for i, c in enumerate(upper_text):                    draw.text((x, start_y + i * step), c, font=font, fill=text_color,                              stroke_width=stroke_w, stroke_fill=(0,0,0))  # 红底黑边,更醒目# 6. 绘制下联(右竖排,防超出边界)if lower_text:                x = min(bg_w - 30, int(bg_w * 0.9))  # 右侧最小边距30                start_y = (bg_h - len(lower_text) * step) / 2                start_y = max(30, min(start_y, bg_h - len(lower_text) * step - 30))for i, c in enumerate(lower_text):                    draw.text((x, start_y + i * step), c, font=font, fill=text_color,                              stroke_width=stroke_w, stroke_fill=(0,0,0))# 7. 绘制横批(上中横排,精准居中)if horizontal_text:                bbox = draw.textbbox((00), horizontal_text, font=font)                text_w = bbox[2] - bbox[0]                x = (bg_w - text_w) / 2                y = max(20, int(bg_h * 0.05))  # 顶部最小边距20                draw.text((x, y), horizontal_text, font=font, fill=text_color,                          stroke_width=stroke_w, stroke_fill=(0,0,0))# 8. 保存结果并更新预览            self.result_image = original_bg            preview = self.result_image.copy()            preview.thumbnail((600400), Image.Resampling.LANCZOS)            tk_img = ImageTk.PhotoImage(preview)            self.preview_canvas.delete("all")            self.preview_canvas.create_image(300200, image=tk_img)            self.preview_canvas.tk_image = tk_img            messagebox.showinfo("成功",f"对联生成完成!\n字体大小:{font_size}号\n"f"上联:{upper_text}\n下联:{lower_text}\n横批:{horizontal_text}"            )except Exception as e:            messagebox.showerror("错误"f"生成失败:{str(e)}")            self.result_image = Nonedefsave_couplet(self):"""保存生成后的对联图片"""ifnot self.result_image:            messagebox.showwarning("提示""请先生成对联!")return        save_path = filedialog.asksaveasfilename(            title="保存对联图片", defaultextension=".png",            filetypes=[("PNG图片""*.png"), ("JPG图片""*.jpg"), ("所有文件""*.*")]        )ifnot save_path:returntry:            self.result_image.save(save_path)            messagebox.showinfo("成功"f"对联已保存至:\n{save_path}")except Exception as e:            messagebox.showerror("错误"f"保存失败:{str(e)}")if __name__ == "__main__":    root = tk.Tk()    app = CoupletGenerator(root)    root.mainloop()

核心新增功能:字体大小选择

  1. 下拉选择框布局:在对联输入区右侧添加「字体大小」标签+下拉框,和下联同排,界面紧凑不杂乱;
  2. 合理的字体选项:提供16、24、32...128号字体(步长8),覆盖小尺寸背景图(小字体)、大尺寸背景图(大字体)的所有需求;
  3. 默认值与异常处理:默认选中48号经典字体,若选择框出现非法值(极少情况),自动回退到默认值并提示;
  4. 排版自动适配字间距、描边宽度、边距会随选择的字体大小自动调整,比如选128号大字体时,字间距会变大、描边会变粗,避免文字重叠/描边过细,无需手动修改代码。

关键优化点(延续之前的修复)

  1. 文字红底黑边描边效果,无论背景图是浅色/红色,文字都能清晰可见;
  2. 文字坐标做了边界校验,确保不会超出图片范围,适配所有尺寸的背景图;
  3. 多系统(Windows/Mac/Linux)中文字体兼容,解决字体缺失导致的文字不显示问题;
  4. 生成成功后,弹窗会显示当前使用的字体大小,方便确认选择是否生效。

使用步骤(新增字体选择环节)

  1. 运行代码,点击「📷 上传背景图片」选择喜欢的背景;
  2. 点击「🎲 随机生成对联」(或手动输入),自动填入上联/下联/横批;
  3. 点击「字体大小」下拉框,选择适合背景图尺寸的字体(小图选32/40,大图选64/72+);
  4. 点击「✨ 生成对联」,预览区显示带自定义字体大小的对联;
  5. 点击「💾 保存成品」,选择路径保存最终图片即可。

扩展小技巧

如果想添加更多字体大小选项,只需修改代码中self.font_size_options列表,比如添加14、20、144等,下拉框会自动同步; 如果想调整字间距/描边宽度,只需修改step = font_size + 8stroke_w = max(1, font_size // 24)中的数字即可(+8是字间距补偿,//24是描边适配比例)。

总结

  1. 新增字体大小下拉选择框,支持16-128号字体自定义,默认48号经典尺寸;
  2. 排版参数(字间距、描边、边距)随字体大小自动适配,无需手动调整,适配所有背景图;
  3. 保留所有原有功能(上传背景、随机对联、生成、保存),并完善了异常处理,保证程序稳定;
  4. 界面布局紧凑合理,新增控件不杂乱,新手也能快速上手操作。

最新文章

随机文章

基本 文件 流程 错误 SQL 调试
  1. 请求信息 : 2026-02-07 12:17:55 HTTP/2.0 GET : https://f.mffb.com.cn/a/473910.html
  2. 运行时间 : 0.126203s [ 吞吐率:7.92req/s ] 内存消耗:4,949.81kb 文件加载:140
  3. 缓存信息 : 0 reads,0 writes
  4. 会话信息 : SESSION_ID=8f625b02bc747597fe11ae895599971d
  1. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/public/index.php ( 0.79 KB )
  2. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/autoload.php ( 0.17 KB )
  3. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/autoload_real.php ( 2.49 KB )
  4. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/platform_check.php ( 0.90 KB )
  5. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/ClassLoader.php ( 14.03 KB )
  6. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/composer/autoload_static.php ( 4.90 KB )
  7. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/helper.php ( 8.34 KB )
  8. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-validate/src/helper.php ( 2.19 KB )
  9. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/helper.php ( 1.47 KB )
  10. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/stubs/load_stubs.php ( 0.16 KB )
  11. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Exception.php ( 1.69 KB )
  12. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-container/src/Facade.php ( 2.71 KB )
  13. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/deprecation-contracts/function.php ( 0.99 KB )
  14. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/polyfill-mbstring/bootstrap.php ( 8.26 KB )
  15. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/polyfill-mbstring/bootstrap80.php ( 9.78 KB )
  16. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/var-dumper/Resources/functions/dump.php ( 1.49 KB )
  17. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-dumper/src/helper.php ( 0.18 KB )
  18. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/symfony/var-dumper/VarDumper.php ( 4.30 KB )
  19. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/App.php ( 15.30 KB )
  20. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-container/src/Container.php ( 15.76 KB )
  21. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/container/src/ContainerInterface.php ( 1.02 KB )
  22. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/provider.php ( 0.19 KB )
  23. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Http.php ( 6.04 KB )
  24. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/helper/Str.php ( 7.29 KB )
  25. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Env.php ( 4.68 KB )
  26. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/common.php ( 0.03 KB )
  27. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/helper.php ( 18.78 KB )
  28. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Config.php ( 5.54 KB )
  29. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/app.php ( 0.95 KB )
  30. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/cache.php ( 0.78 KB )
  31. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/console.php ( 0.23 KB )
  32. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/cookie.php ( 0.56 KB )
  33. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/database.php ( 2.48 KB )
  34. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/facade/Env.php ( 1.67 KB )
  35. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/filesystem.php ( 0.61 KB )
  36. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/lang.php ( 0.91 KB )
  37. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/log.php ( 1.35 KB )
  38. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/middleware.php ( 0.19 KB )
  39. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/route.php ( 1.89 KB )
  40. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/session.php ( 0.57 KB )
  41. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/trace.php ( 0.34 KB )
  42. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/config/view.php ( 0.82 KB )
  43. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/event.php ( 0.25 KB )
  44. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Event.php ( 7.67 KB )
  45. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/service.php ( 0.13 KB )
  46. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/AppService.php ( 0.26 KB )
  47. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Service.php ( 1.64 KB )
  48. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Lang.php ( 7.35 KB )
  49. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/lang/zh-cn.php ( 13.70 KB )
  50. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/initializer/Error.php ( 3.31 KB )
  51. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/initializer/RegisterService.php ( 1.33 KB )
  52. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/services.php ( 0.14 KB )
  53. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/service/PaginatorService.php ( 1.52 KB )
  54. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/service/ValidateService.php ( 0.99 KB )
  55. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/service/ModelService.php ( 2.04 KB )
  56. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-trace/src/Service.php ( 0.77 KB )
  57. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Middleware.php ( 6.72 KB )
  58. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/initializer/BootService.php ( 0.77 KB )
  59. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/Paginator.php ( 11.86 KB )
  60. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-validate/src/Validate.php ( 63.20 KB )
  61. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/Model.php ( 23.55 KB )
  62. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/Attribute.php ( 21.05 KB )
  63. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/AutoWriteData.php ( 4.21 KB )
  64. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/Conversion.php ( 6.44 KB )
  65. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/DbConnect.php ( 5.16 KB )
  66. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/ModelEvent.php ( 2.33 KB )
  67. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/concern/RelationShip.php ( 28.29 KB )
  68. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/contract/Arrayable.php ( 0.09 KB )
  69. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/contract/Jsonable.php ( 0.13 KB )
  70. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/model/contract/Modelable.php ( 0.09 KB )
  71. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Db.php ( 2.88 KB )
  72. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/DbManager.php ( 8.52 KB )
  73. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Log.php ( 6.28 KB )
  74. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Manager.php ( 3.92 KB )
  75. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/log/src/LoggerTrait.php ( 2.69 KB )
  76. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/log/src/LoggerInterface.php ( 2.71 KB )
  77. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Cache.php ( 4.92 KB )
  78. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/psr/simple-cache/src/CacheInterface.php ( 4.71 KB )
  79. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/helper/Arr.php ( 16.63 KB )
  80. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/cache/driver/File.php ( 7.84 KB )
  81. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/cache/Driver.php ( 9.03 KB )
  82. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/CacheHandlerInterface.php ( 1.99 KB )
  83. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/Request.php ( 0.09 KB )
  84. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Request.php ( 55.78 KB )
  85. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/middleware.php ( 0.25 KB )
  86. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Pipeline.php ( 2.61 KB )
  87. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-trace/src/TraceDebug.php ( 3.40 KB )
  88. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/middleware/SessionInit.php ( 1.94 KB )
  89. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Session.php ( 1.80 KB )
  90. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/session/driver/File.php ( 6.27 KB )
  91. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/SessionHandlerInterface.php ( 0.87 KB )
  92. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/session/Store.php ( 7.12 KB )
  93. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Route.php ( 23.73 KB )
  94. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/RuleName.php ( 5.75 KB )
  95. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/Domain.php ( 2.53 KB )
  96. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/RuleGroup.php ( 22.43 KB )
  97. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/Rule.php ( 26.95 KB )
  98. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/RuleItem.php ( 9.78 KB )
  99. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/route/app.php ( 1.72 KB )
  100. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/facade/Route.php ( 4.70 KB )
  101. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/dispatch/Controller.php ( 4.74 KB )
  102. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/route/Dispatch.php ( 10.44 KB )
  103. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/controller/Index.php ( 4.81 KB )
  104. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/app/BaseController.php ( 2.05 KB )
  105. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/facade/Db.php ( 0.93 KB )
  106. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/connector/Mysql.php ( 5.44 KB )
  107. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/PDOConnection.php ( 52.47 KB )
  108. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/Connection.php ( 8.39 KB )
  109. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/ConnectionInterface.php ( 4.57 KB )
  110. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/builder/Mysql.php ( 16.58 KB )
  111. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/Builder.php ( 24.06 KB )
  112. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/BaseBuilder.php ( 27.50 KB )
  113. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/Query.php ( 15.71 KB )
  114. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/BaseQuery.php ( 45.13 KB )
  115. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/TimeFieldQuery.php ( 7.43 KB )
  116. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/AggregateQuery.php ( 3.26 KB )
  117. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/ModelRelationQuery.php ( 20.07 KB )
  118. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/ParamsBind.php ( 3.66 KB )
  119. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/ResultOperation.php ( 7.01 KB )
  120. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/WhereQuery.php ( 19.37 KB )
  121. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/JoinAndViewQuery.php ( 7.11 KB )
  122. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/TableFieldInfo.php ( 2.63 KB )
  123. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-orm/src/db/concern/Transaction.php ( 2.77 KB )
  124. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/log/driver/File.php ( 5.96 KB )
  125. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/LogHandlerInterface.php ( 0.86 KB )
  126. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/log/Channel.php ( 3.89 KB )
  127. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/event/LogRecord.php ( 1.02 KB )
  128. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-helper/src/Collection.php ( 16.47 KB )
  129. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/facade/View.php ( 1.70 KB )
  130. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/View.php ( 4.39 KB )
  131. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Response.php ( 8.81 KB )
  132. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/response/View.php ( 3.29 KB )
  133. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/Cookie.php ( 6.06 KB )
  134. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-view/src/Think.php ( 8.38 KB )
  135. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/framework/src/think/contract/TemplateHandlerInterface.php ( 1.60 KB )
  136. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-template/src/Template.php ( 46.61 KB )
  137. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-template/src/template/driver/File.php ( 2.41 KB )
  138. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-template/src/template/contract/DriverInterface.php ( 0.86 KB )
  139. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/runtime/temp/067d451b9a0c665040f3f1bdd3293d68.php ( 11.98 KB )
  140. /yingpanguazai/ssd/ssd1/www/f.mffb.com.cn/vendor/topthink/think-trace/src/Html.php ( 4.42 KB )
  1. CONNECT:[ UseTime:0.000974s ] mysql:host=127.0.0.1;port=3306;dbname=f_mffb;charset=utf8mb4
  2. SHOW FULL COLUMNS FROM `fenlei` [ RunTime:0.001501s ]
  3. SELECT * FROM `fenlei` WHERE `fid` = 0 [ RunTime:0.000684s ]
  4. SELECT * FROM `fenlei` WHERE `fid` = 63 [ RunTime:0.009900s ]
  5. SHOW FULL COLUMNS FROM `set` [ RunTime:0.001498s ]
  6. SELECT * FROM `set` [ RunTime:0.000622s ]
  7. SHOW FULL COLUMNS FROM `article` [ RunTime:0.001498s ]
  8. SELECT * FROM `article` WHERE `id` = 473910 LIMIT 1 [ RunTime:0.005408s ]
  9. UPDATE `article` SET `lasttime` = 1770437875 WHERE `id` = 473910 [ RunTime:0.005926s ]
  10. SELECT * FROM `fenlei` WHERE `id` = 66 LIMIT 1 [ RunTime:0.000571s ]
  11. SELECT * FROM `article` WHERE `id` < 473910 ORDER BY `id` DESC LIMIT 1 [ RunTime:0.001059s ]
  12. SELECT * FROM `article` WHERE `id` > 473910 ORDER BY `id` ASC LIMIT 1 [ RunTime:0.001290s ]
  13. SELECT * FROM `article` WHERE `id` < 473910 ORDER BY `id` DESC LIMIT 10 [ RunTime:0.007069s ]
  14. SELECT * FROM `article` WHERE `id` < 473910 ORDER BY `id` DESC LIMIT 10,10 [ RunTime:0.003847s ]
  15. SELECT * FROM `article` WHERE `id` < 473910 ORDER BY `id` DESC LIMIT 20,10 [ RunTime:0.004180s ]
0.127822s