news 2026/4/23 9:46:29

C# 实战:利用PrintDocument类高效实现自定义打印功能

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C# 实战:利用PrintDocument类高效实现自定义打印功能

1. 初识PrintDocument类:打印功能的核心引擎

第一次接触C#打印功能时,我完全被各种打印对话框和设置搞晕了。直到发现了PrintDocument这个神器,才发现原来实现打印功能可以如此简单。PrintDocument就像是打印功能的中央控制器,它负责协调整个打印流程,从页面设置到内容绘制,全都由它一手包办。

记得当时我需要给一个餐饮系统添加小票打印功能,用PrintDocument只花了不到半天就搞定了。这个类位于System.Drawing.Printing命名空间,使用时需要先添加引用。最让我惊喜的是,它不仅能打印文本,还能处理图像、表格等各种复杂内容。

// 最基本的打印示例 PrintDocument pd = new PrintDocument(); pd.PrintPage += (sender, e) => { e.Graphics.DrawString("Hello World", new Font("Arial", 12), Brushes.Black, 100, 100); }; pd.Print();

这段代码虽然简单,但包含了打印的三个核心要素:创建打印文档对象、定义打印内容、执行打印命令。在实际项目中,我们通常会把它封装成一个专门的打印服务类,方便各个模块调用。

2. 打印机设置:避开那些年我踩过的坑

刚开始用PrintDocument时,我最头疼的就是打印机设置问题。明明代码写对了,但打印机就是没反应。后来才发现,90%的打印问题都出在打印机配置上,而不是代码本身。

2.1 获取可用打印机列表

在打印之前,我们首先需要知道系统中有哪些打印机可用。通过PrinterSettings.InstalledPrinters可以获取所有已安装的打印机名称。这里有个小技巧:通常我们会把打印机列表显示在下拉框中,让用户自己选择。

// 获取所有打印机名称 foreach(string printer in PrinterSettings.InstalledPrinters) { Console.WriteLine(printer); } // 设置默认打印机 PrintDocument pd = new PrintDocument(); string defaultPrinter = pd.PrinterSettings.PrinterName;

2.2 处理打印机状态问题

在实际使用中,经常会遇到打印机脱机、端口错误等问题。我建议在打印前先检查打印机状态:

  1. 检查打印机是否就绪
  2. 检查是否有未完成的打印任务阻塞队列
  3. 验证打印机端口设置是否正确
// 检查打印机状态 if(!pd.PrinterSettings.IsValid) { MessageBox.Show("打印机设置无效"); return; } if(pd.PrinterSettings.IsDefaultPrinter) { // 处理默认打印机逻辑 }

3. 页面布局设计:让打印内容完美呈现

打印内容布局是另一个需要重点关注的领域。与屏幕显示不同,打印布局需要考虑纸张大小、边距、分页等实际问题。

3.1 设置页面属性

通过DefaultPageSettings属性,我们可以控制纸张大小、方向、边距等:

// 设置页面属性 pd.DefaultPageSettings.PaperSize = new PaperSize("A4", 827, 1169); // A4纸 pd.DefaultPageSettings.Margins = new Margins(50, 50, 50, 50); // 四边距50 pd.DefaultPageSettings.Landscape = true; // 横向打印

3.2 精确计算打印位置

打印内容的位置需要精确计算。我习惯使用PrintPageEventArgs提供的MarginBounds属性来确定可打印区域:

pd.PrintPage += (sender, e) => { RectangleF bounds = e.MarginBounds; float x = bounds.Left; float y = bounds.Top; // 计算行高 float lineHeight = font.GetHeight(e.Graphics); // 打印文本 e.Graphics.DrawString("订单号: 20230001", font, Brushes.Black, x, y); y += lineHeight * 1.5f; e.Graphics.DrawString("日期: " + DateTime.Now.ToString(), font, Brushes.Black, x, y); };

4. 实战案例:打印餐饮小票

让我们通过一个完整的餐饮小票打印案例,把前面学到的知识串起来。这个例子包含了文本、表格线和汇总信息的打印。

4.1 构建打印内容

首先,我们需要准备打印内容。我通常会创建一个专门的方法来生成打印文本:

public string BuildReceiptContent() { StringBuilder sb = new StringBuilder(); sb.AppendLine(" 美味餐厅 "); sb.AppendLine("---------------------"); sb.AppendLine("订单号: " + orderNumber); sb.AppendLine("时间: " + DateTime.Now.ToString("yyyy-MM-dd HH:mm")); sb.AppendLine("---------------------"); sb.AppendLine("菜品 数量 单价 小计"); foreach(var item in orderItems) { sb.AppendLine($"{item.Name,-10} {item.Quantity,3} {item.Price,6:C} {item.Total,6:C}"); } sb.AppendLine("---------------------"); sb.AppendLine($"总计: {totalAmount:C}"); sb.AppendLine("谢谢惠顾,欢迎下次光临"); return sb.ToString(); }

4.2 实现PrintPage事件

接下来是核心的打印逻辑。这里我们不仅要打印文本,还要绘制表格线:

private void PrintReceipt(object sender, PrintPageEventArgs e) { Graphics g = e.Graphics; Font titleFont = new Font("黑体", 14, FontStyle.Bold); Font contentFont = new Font("宋体", 10); Font footerFont = new Font("宋体", 8); float x = e.MarginBounds.Left; float y = e.MarginBounds.Top; // 打印标题 g.DrawString("美味餐厅", titleFont, Brushes.Black, x + (e.MarginBounds.Width - g.MeasureString("美味餐厅", titleFont).Width)/2, y); y += titleFont.GetHeight(g) * 1.5f; // 打印分隔线 g.DrawLine(Pens.Black, x, y, x + e.MarginBounds.Width, y); y += 10; // 打印订单信息 string[] lines = BuildReceiptContent().Split('\n'); foreach(string line in lines) { g.DrawString(line, contentFont, Brushes.Black, x, y); y += contentFont.GetHeight(g); } // 处理分页 e.HasMorePages = false; // 本例只有一页 }

5. 高级技巧:打印预览与多页处理

对于复杂的打印需求,我们还需要考虑打印预览和多页打印的功能。

5.1 实现打印预览

打印预览可以大大提升用户体验,避免浪费纸张:

// 打印预览 PrintPreviewDialog preview = new PrintPreviewDialog(); preview.Document = printDocument; preview.ShowDialog();

5.2 处理多页打印

当内容超过一页时,需要使用HasMorePages属性来控制分页:

private List<string> printLines; private int currentLine; private void PrintMultiPage(object sender, PrintPageEventArgs e) { Graphics g = e.Graphics; Font font = new Font("宋体", 10); float y = e.MarginBounds.Top; while(currentLine < printLines.Count) { string line = printLines[currentLine]; g.DrawString(line, font, Brushes.Black, e.MarginBounds.Left, y); y += font.GetHeight(g); currentLine++; if(y >= e.MarginBounds.Bottom) { e.HasMorePages = currentLine < printLines.Count; return; } } e.HasMorePages = false; currentLine = 0; // 重置计数器 }

6. 性能优化与异常处理

在实际项目中,打印功能还需要考虑性能和稳定性问题。

6.1 资源释放

打印完成后,一定要记得释放资源:

try { printDocument.Print(); } catch(InvalidPrinterException ex) { MessageBox.Show("打印机不可用: " + ex.Message); } finally { printDocument.Dispose(); }

6.2 异步打印

大量打印任务可能会阻塞UI线程,可以考虑使用异步打印:

// 异步打印 printDocument.PrintController = new StandardPrintController(); printDocument.BeginPrint += (s, e) => { /* 打印前准备 */ }; printDocument.EndPrint += (s, e) => { /* 打印后清理 */ }; // 在新线程中打印 Task.Run(() => printDocument.Print());

7. 常见问题解决方案

在多年的开发中,我总结了一些常见问题的解决方法:

  1. 打印内容偏移:检查打印机驱动设置中的页边距,确保与代码设置一致
  2. 中文乱码:使用支持中文的字体,如"宋体"、"微软雅黑"
  3. 打印速度慢:减少不必要的绘图操作,使用PrintController优化
  4. 图片模糊:确保图片分辨率足够高,至少300dpi
// 高质量打印设置 printDocument.DefaultPageSettings.PrinterResolution = new PrinterResolution { Kind = PrinterResolutionKind.High };

掌握了这些技巧后,你会发现用C#实现打印功能其实并不复杂。关键是要理解PrintDocument的工作原理,并在实际项目中不断实践和优化。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/23 9:45:27

测试开机启动脚本真实体验:OpenWrt环境实操分享

测试开机启动脚本真实体验&#xff1a;OpenWrt环境实操分享 在嵌入式设备和家用路由器场景中&#xff0c;OpenWrt 是一个被广泛采用的轻量级 Linux 发行版。它灵活、可定制&#xff0c;但对刚接触的用户来说&#xff0c;有些基础功能反而容易踩坑——比如“让一段命令在设备每…

作者头像 李华
网站建设 2026/4/22 15:43:18

Flowise多终端适配:PC/移动端一致体验

Flowise多终端适配&#xff1a;PC/移动端一致体验 Flowise 是一个真正让 AI 工作流“看得见、摸得着、用得上”的平台。它不靠命令行堆砌参数&#xff0c;也不靠写几十行代码配置链路&#xff0c;而是把 LangChain 的复杂能力&#xff0c;变成画布上可拖拽的节点——就像搭积木…

作者头像 李华
网站建设 2026/4/23 9:46:25

三天搭建企业级Agent!大模型深度嵌入业务实战教程

大模型技术正从"泛化对话"向"深度业务嵌入"转变&#xff0c;企业级Agent成为核心战场。企业需要可本地部署、高度定制化的智能体架构&#xff0c;而非通用聊天机器人。作者分享三天搭建企业级Agent的实战经验&#xff0c;提供面向新手的教程。展望未来&…

作者头像 李华
网站建设 2026/4/23 9:46:41

HY-MT1.5-1.8B部署卡顿?算力优化实战让推理速度提升2倍

HY-MT1.5-1.8B部署卡顿&#xff1f;算力优化实战让推理速度提升2倍 你是不是也遇到过这样的情况&#xff1a;明明选了参数量更小的HY-MT1.5-1.8B模型&#xff0c;想在本地或边缘设备上跑得快一点&#xff0c;结果用vLLM部署完&#xff0c;一调用Chainlit前端就卡顿、响应慢、吞…

作者头像 李华
网站建设 2026/4/23 9:45:32

GLM-4v-9b入门必看:GLM-4v-9b与GLM-4-9B语言模型能力差异解析

GLM-4v-9b入门必看&#xff1a;GLM-4v-9b与GLM-4-9B语言模型能力差异解析 你是不是也遇到过这些情况&#xff1a; 想让AI看懂一张密密麻麻的财务报表截图&#xff0c;结果它把数字读错了&#xff1b; 上传一张带小字的手机界面截图问“这个按钮点开后跳转到哪”&#xff0c;模…

作者头像 李华
网站建设 2026/4/20 1:36:14

DASD-4B-Thinking入门指南:如何用curl命令绕过Chainlit直接测试vLLM API

DASD-4B-Thinking入门指南&#xff1a;如何用curl命令绕过Chainlit直接测试vLLM API 你刚部署好DASD-4B-Thinking模型&#xff0c;看着Chainlit界面里流畅的对话体验&#xff0c;心里可能已经冒出一个念头&#xff1a;能不能不走前端&#xff0c;直接跟后端API打交道&#xff…

作者头像 李华