C# 打印PDF的常用方法
这里先提供一个helper类的模板
1.使用默认程序打印
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Drawing.Printing;
using System.IO;
using System.Runtime.InteropServices;namespace PDF
{public static class PrintHelper{#region 打印机操作核心方法/// <summary>/// // 设置默认打印机/// </summary>/// <param name="Name"></param>/// <returns></returns>[DllImport("winspool.drv")]public static extern bool SetDefaultPrinter(String Name);/// <summary>/// 获取系统默认打印机名称/// </summary>public static string DefaultPrinterName(){PrintDocument fPrintDocument = new PrintDocument();return fPrintDocument.PrinterSettings.PrinterName;}/// <summary>/// 获取本机所有打印机列表/// </summary>public static List<string> GetLocalPrinters(){List<string> fPrinters = new List<string>();fPrinters.Add(DefaultPrinterName());foreach (string fPrinterName in PrinterSettings.InstalledPrinters){if (!fPrinters.Contains(fPrinterName)){fPrinters.Add(fPrinterName);}}return fPrinters;}/// <summary>/// 设置指定打印机为默认打印机/// </summary>public static bool SetLocalPrinter(string defaultPrinter){foreach (string item in GetLocalPrinters()){if (item.Contains(defaultPrinter)){return SetDefaultPrinter(item);}}return false;}#endregion/// <summary>/// 打印/// 通过系统关联程序打印文件/// 需注意的是,这种方式依赖于文件类型的默认关联程序/// </summary>public static void PrintByDefaultProcess(string LocalPath){Process p = new Process();ProcessStartInfo startInfo = new ProcessStartInfo{CreateNoWindow = true,WindowStyle = ProcessWindowStyle.Hidden,UseShellExecute = true,FileName = LocalPath,Verb = "print", // 它会调用与文件类型关联的默认应用程序来执行打印文件,如Adobe Reader或者word或图片查看器或其他PDF阅读器进行打印Arguments = $"/p /h \"{LocalPath}\" \"{DefaultPrinterName()}\"",//WorkingDirectory = Path.GetDirectoryName(Application.ExecutablePath) // 设置工作目录为应用程序启动目录};p.StartInfo = startInfo;p.Start();p.WaitForExit(5); // 等待5秒钟,确保打印任务已提交}}
}
需要注意的是,它虽然是一种简单且常用的方法。但这种方法依赖于系统关联的默认应用程序来处理打印任务,适用于大多数情况,但存在一些限制和潜在问题。如pdf后缀没有关联默认程序会报错。
所以我一般不会使用这种方式。
下面会罗列其他的方法和优缺点
2.使用命令行工具(如 Adobe Reader 的 /t 参数)
public static void PrintWithAdobeCommandLine(string filePath, string printerName)
{string adobePath = @"C:\Program Files\Adobe\Acrobat Reader DC\Reader\AcroRd32.exe";Process.Start(adobePath, $"/t \"{filePath}\" \"{printerName}\"");
}
优点:绕过默认应用程序,强制使用 Adobe Reader。
缺点:需硬编码 Adobe 路径,可能因版本或安装位置不同失效。
3.直接调用 Adobe Acrobat 的 COM 接口
如果系统中安装了 Adobe Acrobat Pro,可以通过 COM 对象直接控制打印:
using Adobe.Acrobat;public static void PrintWithAdobePdf(string filePath)
{Acrobat.AcroApp app = new Acrobat.AcroApp();Acrobat.AcroAVDoc avDoc = new Acrobat.AcroAVDoc();if (avDoc.Open(filePath, "")){Acrobat.AcroPDDoc pdDoc = (Acrobat.AcroPDDoc)avDoc.GetPDDoc();pdDoc.PrintPages(0, 0, 0, 0, 0, 0, 0); // 参数需根据需求调整avDoc.Close(true);}app.Exit();
}
优点:直接控制 Adobe 软件,可靠性高。
缺点:依赖 Adobe Acrobat Pro,非免费版;需处理 COM 引用和资源释放。
4. 使用 .NET 的 PrintDocument 类
结合 System.Drawing 处理 PDF(需将 PDF 渲染为图像):
public static void PrintPdfAsImage(string filePath)
{using (var document = PdfDocument.Load(filePath)) // 需要 PdfiumViewer 或其他库{var printDoc = new System.Drawing.Printing.PrintDocument();printDoc.PrintPage += (sender, e) =>{e.Graphics.DrawImage(document.RenderToImage(), e.MarginBounds);e.HasMorePages = false;};printDoc.Print();}
}
**优点:纯 .NET 实现。
缺点:**渲染质量依赖库,可能丢失文本选中等特性。
5. 使用 DEVExpress的PdfView打印
基本打印代码
using DevExpress.XtraPrinting;
using DevExpress.XtraPrinting.Preview;private void PrintPdf()
{// 检查 PdfViewer 是否已加载文档if (pdfViewer1.Document == null){MessageBox.Show("请先加载 PDF 文件!");return;}// 调用 Print 方法(弹出系统打印对话框)pdfViewer1.Print();
}
高级打印配置(自定义打印机/页边距)
private void PrintPdfWithSettings()
{if (pdfViewer1.Document == null) return;// 创建打印参数对象var printParams = new PrinterSettings{PrinterName = "Your_Printer_Name", // 指定打印机名称(需与系统一致)Copies = 2, // 打印份数FromPage = 1, // 起始页ToPage = 3 // 结束页(设置为 0 表示全部)};// 创建打印文档对象var printDocument = pdfViewer1.CreatePrintDocument();// 配置打印事件(可选)printDocument.PrintPage += (s, e) => {// 可在此处自定义打印逻辑(例如添加水印)e.HasMorePages = false; // 标记是否还有后续页面};// 应用打印参数并执行打印printDocument.PrinterSettings = printParams;printDocument.Print(); // 静默打印(不弹出对话框)
}
打印预览
private void PreviewBeforePrint()
{if (pdfViewer1.Document == null) return;// 创建打印预览窗体var printPreviewForm = new PrintPreviewForm(pdfViewer1);printPreviewForm.ShowDialog();
}
完整示例
private void SafePrintPdf()
{try{if (pdfViewer1.Document == null){MessageBox.Show("未加载 PDF 文件!");return;}var printDialog = new PrintDialog{Document = pdfViewer1.CreatePrintDocument(),AllowCurrentPage = true, // 允许选择当前页AllowSomePages = true};if (printDialog.ShowDialog() == DialogResult.OK){printDialog.Document.Print();}}catch (Exception ex){MessageBox.Show($"打印失败: {ex.Message}");}
}
以下是常见的主流三方dll打印,推荐
一、iTextSharp(免费/商业)
特点:
免费版:仅限非商业用途,功能受限。
商业版:需购买许可证,支持完整功能。
需配合其他库:iText 本身不直接提供打印功能,需结合 System.Drawing 或 Ghostscript 渲染后打印。
示例代码(静默打印):
using iTextSharp.text.pdf;
using System.Diagnostics;public static void PrintWithITextSharp(string filePath)
{try{// 使用 Ghostscript 渲染 PDF 为图像(需安装 Ghostscript)Process process = new Process();process.StartInfo.FileName = "gswin64c.exe"; // Ghostscript 路径process.StartInfo.Arguments = $"-dBATCH -dNOPAUSE -sDEVICE=mswinpr2 -sOutputFile=\"%printer%{GetDefaultPrinter()}\" \"{filePath}\"";process.StartInfo.UseShellExecute = false;process.StartInfo.CreateNoWindow = true;process.Start();process.WaitForExit();}catch (Exception ex){throw new Exception($"打印失败: {ex.Message}");}
}// 获取默认打印机名称
private static string GetDefaultPrinter()
{return System.Drawing.Printing.PrinterSettings.InstalledPrinters.Cast<string>().FirstOrDefault(p => p.ToLower().Contains("pdf"));
}
需要注意的是:这里为啥不用System.Drawing,因为如下
System.Drawing 主要用于 生成图形内容(如绘制线条、文本、图像),而不是解析 PDF 文件。如果要打印 PDF,需要先将 PDF 渲染为位图或矢量图像,再通过 System.Drawing.Printing.PrintDocument 发送到打印机。但这种方式会导致:
分辨率问题:PDF 的矢量内容会被栅格化为位图,放大后可能模糊。
性能问题:大文件渲染耗时,内存占用高。
功能缺失:无法保留 PDF 的文本可编辑性、书签等元数据。
以下是使用System.Drawing.Printing.PrintDocument 的方式,如果 PDF 是单页且内容简单,可以用 System.Drawing 渲染第一页为图像后打印
using System.Drawing;
using System.Drawing.Imaging;
using System.Drawing.Printing;public static void PrintPdfAsImage(string filePath)
{using (var image = RenderPdfToImage(filePath)) // 需自行实现 PDF 到图像的渲染{var printDoc = new PrintDocument();printDoc.PrintPage += (s, e) =>{e.Graphics.DrawImage(image, e.MarginBounds);e.HasMorePages = false;};printDoc.Print();}
}// 注意:需使用第三方库(如 PdfiumViewer)实现 RenderPdfToImage
二、PdfiumViewer(免费开源)
特点:
基于 Google 的 Pdfium 引擎,渲染性能高。
直接支持 PDF 打印,无需额外依赖。
支持静默打印、自定义页码范围。
优点:
无需额外依赖,API 简洁。
支持 PDF 元数据、书签等高级操作。
示例代码
using PdfiumViewer;public static void PrintWithPdfiumViewer(string filePath, string printerName = null)
{using (var document = PdfDocument.Load(filePath)){var printDialog = new System.Windows.Forms.PrintDialog{PrinterSettings = { PrinterName = printerName } // 可选指定打印机};if (printDialog.ShowDialog() == System.Windows.Forms.DialogResult.OK){document.Print(printDialog.PrinterSettings);}}
}
高级用法(静默打印):
public static void SilentPrint(string filePath)
{using (var document = PdfDocument.Load(filePath)){var printerSettings = new System.Drawing.Printing.PrinterSettings{PrinterName = "Your_Printer_Name",Copies = 2};document.Print(printerSettings);}
}
三、Spire.PDF(商业授权)
特点:
商业库,需购买许可证(免费版有水印)。
提供完整的 PDF 操作 API,包括打印、转换、加密等。
支持静默打印、自定义页边距、多线程打印。
注意事项:
免费版会在输出文件添加评估水印。
需在项目中引用 Spire.PDF NuGet 包。
示例代码
using Spire.Pdf;
using Spire.Pdf.Graphics;public static void PrintWithSpirePdf(string filePath)
{PdfDocument doc = new PdfDocument();doc.LoadFromFile(filePath);// 配置打印参数PdfPrintOptions options = new PdfPrintOptions{PrinterName = "Your_Printer_Name",Copies = 1,Range = PdfPrintRange.AllPages};// 静默打印(无对话框)doc.Print(options);
}
高级功能(添加水印后打印):
public static void PrintWithWatermark(string filePath)
{PdfDocument doc = new PdfDocument();doc.LoadFromFile(filePath);foreach (PdfPageBase page in doc.Pages){PdfFont font = new PdfFont(PdfFontFamily.Helvetica, 50);PdfBrush brush = PdfBrushes.LightGray;page.Canvas.DrawString("CONFIDENTIAL", font, brush, new PointF(0, 0));}doc.SaveToFile("Watermarked.pdf");doc.Close();// 打印带水印的文件PrintWithSpirePdf("Watermarked.pdf");
}
推荐场景
推荐 PdfiumViewer:免费、高性能,适合大多数场景。
商业项目选 Spire.PDF:功能全面,省去外部依赖。
避免 System.Drawing 直接打印 PDF:除非是单页简单内容。
已有 Ghostscript 环境 → iTextSharp(需额外配置)
大文件打印时,优先使用 PdfiumViewer 或 Spire.PDF(内存占用更低)。
批量打印时,关闭自动刷新(如 Spire.Pdf 的 Options.NoSaveAfterPrint)。
相关文章:
C# 打印PDF的常用方法
这里先提供一个helper类的模板 1.使用默认程序打印 using System; using System.Collections.Generic; using System.Diagnostics; using System.Drawing.Printing; using System.IO; using System.Runtime.InteropServices;namespace PDF {public static class PrintHelper{#…...

若依微服务的定制化服务
复制依赖 复制依赖 复制system服务的bootstrap.yml文件,修改port和name 在nacos复制一个新的nacos配置,修改对应的nacos的配置 ,可能不需要修改,看情况。 网关修改 注意curd的事项,模块名称的修改...

Axios 如何通过配置实现通过接口请求下载文件
前言 今天,我写了 《Nodejs 实现 Mysql 数据库的全量备份的代码演示》 和 《NodeJS 基于 Koa, 开发一个读取文件,并返回给客户端文件下载》 两篇文章。在这两篇文章中,我实现了数据库的备份,和提供数据库下载等接口。 但是&…...
小表驱动大表更快吗,不是
背景 head头表(5000),line行表(15万),导出数据包含头和行,一对多。 以行表为维度导出15万数据。 sql 如下两个sql查询,有如下差异 驱动方式:第一个大表驱动小表&…...

20250529-C#知识:运算符重载
C#知识:运算符重载 运算符重载能够让我们像值类型数据那样使用运算符对类或结构体进行运算,并且能够自定义运算逻辑。 1、运算符重载及完整代码示例 作用是让自定义的类或者结构体能够使用运算符运算符重载一定是public static的可以把运算符看成一个函…...
【HW系列】—目录扫描、口令爆破、远程RCE流量特征
本文仅用于技术研究,禁止用于非法用途。 文章目录 目录扫描漏洞的流量特征及检测方法一、基础流量特征二、工具特征差异三、绕过行为特征四、关联行为特征五、检测与防御建议 口令爆破漏洞的流量特征及检测方法一、基础流量特征二、工具标识特征三、绕过行为特征四…...

如何在WordPress网站中添加相册/画廊
在 WordPress 网站上添加相册可以让您展示许多照片。无论您是在寻找标准的网格相册画廊还是独特的瀑布流相册画廊体验,学习如何在 WordPress 网站上添加相册总是一个好主意。在本教程中,我们将介绍两种为 WordPress 网站添加相册的方法:使用区…...
【NLP基础知识系列课程-Tokenizer的前世今生第一课】Tokenizer 是什么?为什么重要?
语言的“颗粒度”:我们到底在切什么? 我们都知道模型要处理文本,第一步是把一段段字符变成“token”。但这些 token 究竟应该是句子、单词,还是更小的片段,比如“un break able”? 这背后涉及的是一个非…...

Codeforces Round 1025 (Div. 2)
Problem - A - Codeforces 查有没有人说谎,有一个必错的情况: 两个人都说输了,必有人撒谎,还有就是所有人都赢了,也是撒谎 来看代码: #include <iostream> #include <vector> using namespa…...

Ubuntu20.04操作系统ssh开启oot账户登录
文章目录 1 前提2 设置root密码3 允许ssh登录root账户3.1 编辑配置文件3.2 重启ssh服务 4 安全注意事项 1 前提 ssh可以使用普通用户正常登录。 2 设置root密码 打开终端,设置密码 sudo passwd root # 设置root密码3 允许ssh登录root账户 3.1 编辑配置文件 su…...

类欧几里得算法(floor_sum)
文章目录 普通floor_sum洛谷P5170 【模板】类欧几里得算法 万能欧几里得算法求 ∑ i 1 n A i B ⌊ a i b c ⌋ \sum_{i1}^{n}A^iB^{\lfloor \frac{aib}{c} \rfloor} ∑i1nAiB⌊caib⌋求 ∑ i 0 n ⌊ a i b c ⌋ \sum_{i0}^n \lfloor\frac{aib}{c}\rfloor ∑i0n⌊caib…...

每日Prompt:卵石拼画
提示词 世界卵石拼画大师杰作,极简风格,贾斯汀.贝特曼的风格,彩色的鹅卵石,斑马头像,鹅卵石拼画,马卡龙浅紫色背景,自然与艺术的结合,新兴的艺术创作形式,石头拼贴画&am…...
湖北理元理律师事务所观察:债务优化如何成为民生安全网
据央行2023年报告,中国家庭债务收入比达137.8%。面对债务高压,湖北理元理律师事务所的实践揭示:专业债务规划的价值不仅是减负数字,更是构建社会稳定的微观防线。 一、从“催收恐惧”到“主动管理”的转变 该所服务数据显示&…...
AI时代新词-机器学习即服务(MLaaS)
一、什么是机器学习即服务(MLaaS)? 机器学习即服务(Machine Learning as a Service,简称MLaaS)是一种云计算服务模式,它将机器学习工具和平台作为服务提供给用户。用户可以通过云平台访问机器学…...
设计模式简述(二十)规格模式
规格模式 描述组件 使用 描述 规格模式并不在传统的23设计模式中,属于后面扩展的设计模式。 简单描述就是对一批数据进行多条件(包括逻辑组合、有点装饰器的感觉,可以不断套娃)匹配。 组件 实体 package dp.spec;/*** TODO** …...
符合Python风格的对象(覆盖类属性)
覆盖类属性 Python 有个很独特的特性:类属性可用于为实例属性提供默认 值。Vector2d 中有个 typecode 类属性,bytes 方法两次用到了 它,而且都故意使用 self.typecode 读取它的值。因为 Vector2d 实 例本身没有 typecode 属性,所…...
题目 3314: 蓝桥杯2025年第十六届省赛真题-魔法科考试
题目 3314: 蓝桥杯2025年第十六届省赛真题-魔法科考试 时间限制: 3s 内存限制: 512MB 提交: 245 解决: 49 题目描述 小明正在参加魔法科的期末考试,考生需要根据给定的口诀组合出有效的 魔法。其中,老师给定了 n 个上半部分口诀 a1, a2, . . . , an 和 m…...
Java八股-Java优缺点,跨平台,jdk、jre、jvm关系,解释和编译
java优势劣势? 优势:面向对象,平台无关,垃圾回收,强大的生态系统 劣势:运行速度慢(相比于c和rust这样的原生编译语言会比较慢),语法繁琐(相比于python&…...
linux 内核态和用户态定时器函数使用总结
1,场景总结 定时器类型精度范围适用场景注意事项用户态信号定时器秒级简单任务调度、心跳检测信号处理函数中不可调用非异步安全函数timerfdepoll纳秒级高精度事件循环、多媒体处理需要配合IO多路复用机制使用内核timer_list毫秒级设备驱动、硬件交互基于jiffies时…...
支持selenium的chrome driver更新到136.0.7103.113
最近chrome释放新版本:136.0.7103.113 如果运行selenium自动化测试出现以下问题,是需要升级chromedriver才可以解决的。 selenium.common.exceptions.SessionNotCreatedException: Message: session not created: This version of ChromeDriver only s…...

硬件服务器基础
1、硬件服务器基础 2、服务器后面板 3、组件 3.1 CPU 3.2 内存 3.3 硬盘 3.4 风扇 4、服务器品牌 4.1 配置 4.2 CPU 架构 4.2.1 CPU 命名规则 4.2.2 服务器 CPU 和家用 CPU 的区别 4.2.3 CPU 在主板的位置 4.2.4 常见 CPU 安装方式 4.3 内存中组件 4.3.1 内存的分类 4.3.1.1 …...
LVS的DR模式部署
目录 一、引言:高并发场景下的流量调度方案 二、LVS-DR 集群核心原理与架构设计 (一)工作原理与数据流向 数据包流向步骤3: (二)模式特性与53网络要求 三、实战配置:从9环境搭建到参数调整…...

TRS收益互换平台开发实践:从需求分析到系统实现
一、TRS业务概述 TRS(Total Return Swap)收益互换是一种金融衍生工具,允许投资者通过支付固定或浮动利息,换取标的资产(如股票、指数)的收益权。典型应用场景包括: 跨境投资ÿ…...

测试Bug篇
本节概要: 软件测试的生命周期 bug的概念 buh要素 bug等级 bug生命周期 对于bug的定级与开发发生冲突如何解决 一、 软件测试的⽣命周期 软件测试贯穿于软件的整个生命周期,针对这句话我们⼀起来看⼀下软件测试是如何贯穿软件的整个生命周期。 软…...

【Linux系统移植】Cortex-A8 Linux系统移植(超详细)
目录 前言 一、ARM开发板ARM简介RISC和CISCARM产品分布核心板S5pv210 SOC嵌入式系统开发方式 二、嵌入式系统组成为什么要系统移植内核移植框图 三、嵌入式开发环境搭建搭建开发环境总流程设置ubuntu与windows共享目录修改用户为root用户安装NFS服务器安装tftp服务器安装交叉编…...

第十五届蓝桥杯大赛软件赛国赛Python 大学 C 组试做【本期题单: 设置密码、栈】
早上好啊大伙,这一期依旧是蓝桥杯备赛刷题的记录。 本期题单:设置密码、栈 前言 前段时间准备省赛,运气好进国赛了。所以就开始准备6月份的国赛。但是近期还有别的比赛要准备,所以刷题的速度比较慢,可能每一期就会有一…...

报错SvelteKitError: Not found: /.well-known/appspecific/com.chrome.devtools.json
报错信息 SvelteKitError: Not found: /.well-known/appspecific/com.chrome.devtools.json 解决方案一 更新所有依赖 npm update解决方案二(不一定成功) src\lib\hooks.server.ts,每次请求服务器时执行 import type { Handle } from &…...

word添加页眉
问题一: 为word文档添加页眉。 方法: 1、在要添加页眉的第一页页面顶端双击页眉区域,如果添加页眉页上面还有其他页或者与上一页添加页眉内容不同,记得取消“链接到前一节”(点击使其上面没有灰色即可)&…...
JavaScript 中的 BigInt:当普通数字不够“大“时的救星
前言 大家好!今天我们要聊一聊 JavaScript 中那个"大"到不行的数据类型 —— BigInt。如果你曾经因为 JavaScript 的数字精度问题而抓狂,或者好奇为什么 9007199254740992 9007199254740993 会返回 true,那么这篇文章就是为你准备…...
通过mailto:实现web/html邮件模板唤起新建邮件并填写内容
一、背景 在实现网站、html邮件模板过程中,难免会遇到需要通过邮箱向服务提供方发起技术支持等需求,因此,我们需要通过一个功能,能新建邮件并提供模板,提高沟通效率 二、mailto协议配置说明 参数描述mailto:nameema…...