C# 使用 Windows API 發送文件到印表機

来源:http://www.cnblogs.com/Brambling/archive/2017/05/28/6916115.html
-Advertisement-
Play Games

最近需要做一個列印的功能,於是在網上找到了這麼一個方法。 以上就是全部代碼了,調用就很簡單了,方法如下: ...


最近需要做一個列印的功能,於是在網上找到了這麼一個方法。

  [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
    public class DOCINFOA
    {
        [MarshalAs(UnmanagedType.LPStr)]
        public string pDocName;
        [MarshalAs(UnmanagedType.LPStr)]
        public string pOutputFile;
        [MarshalAs(UnmanagedType.LPStr)]
        public string pDataType;
    }

    public class PrintCode
    {
        [DllImport("winspool.Drv", EntryPoint = "OpenPrinterA", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool OpenPrinter([MarshalAs(UnmanagedType.LPStr)] string szPrinter, out IntPtr hPrinter, IntPtr pd);


        [DllImport("winspool.Drv", EntryPoint = "ClosePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool ClosePrinter(IntPtr hPrinter);


        [DllImport("winspool.Drv", EntryPoint = "StartDocPrinterA", SetLastError = true, CharSet = CharSet.Ansi, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool StartDocPrinter(IntPtr hPrinter, Int32 level, [In, MarshalAs(UnmanagedType.LPStruct)] DOCINFOA di);


        [DllImport("winspool.Drv", EntryPoint = "EndDocPrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool EndDocPrinter(IntPtr hPrinter);


        [DllImport("winspool.Drv", EntryPoint = "StartPagePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool StartPagePrinter(IntPtr hPrinter);


        [DllImport("winspool.Drv", EntryPoint = "EndPagePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool EndPagePrinter(IntPtr hPrinter);


        [DllImport("winspool.Drv", EntryPoint = "WritePrinter", SetLastError = true, ExactSpelling = true, CallingConvention = CallingConvention.StdCall)]
        public static extern bool WritePrinter(IntPtr hPrinter, IntPtr pBytes, Int32 dwCount, out Int32 dwWritten);


        /// <summary>
        /// 該方法把非托管記憶體中的位元組數組發送到印表機的列印隊列
        /// </summary>
        /// <param name="szPrinterName">印表機名稱</param>
        /// <param name="pBytes">非托管記憶體指針</param>
        /// <param name="dwCount">位元組數</param>
        /// <returns>成功返回true,失敗時為false</returns>
        public static bool SendBytesToPrinter(string szPrinterName, IntPtr pBytes, Int32 dwCount)
        {
            Int32 dwError = 0, dwWritten = 0;
            IntPtr hPrinter = new IntPtr(0);
            DOCINFOA di = new DOCINFOA();
            bool bSuccess = false;

            di.pDocName = "My C#.NET RAW Document";
            di.pDataType = "RAW";

            try
            {
                // 打開印表機
                if (OpenPrinter(szPrinterName.Normalize(), out hPrinter, IntPtr.Zero))
                {
                    // 啟動文檔列印
                    if (StartDocPrinter(hPrinter, 1, di))
                    {
                        // 開始列印
                        if (StartPagePrinter(hPrinter))
                        {
                            // 向印表機輸出位元組  
                            bSuccess = WritePrinter(hPrinter, pBytes, dwCount, out dwWritten);
                            EndPagePrinter(hPrinter);
                        }
                        EndDocPrinter(hPrinter);
                    }
                    ClosePrinter(hPrinter);
                }
                if (bSuccess == false)
                {
                    dwError = Marshal.GetLastWin32Error();
                }
            }
            catch (Win32Exception ex)
            {
                WriteLog(ex.Message);
                bSuccess = false;
            }
            return bSuccess;
        }


        /// <summary>
        /// 發送文件到印表機方法
        /// </summary>
        /// <param name="szPrinterName">印表機名稱</param>
        /// <param name="szFileName">列印文件的路徑</param>
        /// <returns></returns>
        public static bool SendFileToPrinter(string szPrinterName, string szFileName)
        {
            bool bSuccess = false;
            try
            {
                // 打開文件  
                FileStream fs = new FileStream(szFileName, FileMode.Open);

                // 將文件內容讀作二進位
                BinaryReader br = new BinaryReader(fs);

                // 定義位元組數組
                Byte[] bytes = new Byte[fs.Length];

                // 非托管指針  
                IntPtr pUnmanagedBytes = new IntPtr(0);

                int nLength;

                nLength = Convert.ToInt32(fs.Length);

                // 讀取文件內容到位元組數組
                bytes = br.ReadBytes(nLength);

                // 為這些位元組分配一些非托管記憶體
                pUnmanagedBytes = Marshal.AllocCoTaskMem(nLength);

                // 將托管位元組數組複製到非托管記憶體指針
                Marshal.Copy(bytes, 0, pUnmanagedBytes, nLength);

                // 將非托管位元組發送到印表機
                bSuccess = SendBytesToPrinter(szPrinterName, pUnmanagedBytes, nLength);

                // 釋放先前分配的非托管記憶體
                Marshal.FreeCoTaskMem(pUnmanagedBytes);

                fs.Close();
                fs.Dispose();
            }
            catch (Win32Exception ex)
            {
                WriteLog(ex.Message);
                bSuccess = false;
            }
            return bSuccess;
        }

        /// <summary>
        /// 將字元串發送到印表機方法
        /// </summary>
        /// <param name="szPrinterName">印表機名稱</param>
        /// <param name="szString">列印的字元串</param>
        /// <returns></returns>
        public static bool SendStringToPrinter(string szPrinterName, string szString)
        {
            bool flag = false;
            try
            {
                IntPtr pBytes;
                Int32 dwCount;

                // 獲取字元串長度  
                dwCount = szString.Length;

                // 將字元串複製到非托管 COM 任務分配的記憶體非托管記憶體塊,並轉換為 ANSI 文本
                pBytes = Marshal.StringToCoTaskMemAnsi(szString);

                // 將已轉換的 ANSI 字元串發送到印表機
                flag = SendBytesToPrinter(szPrinterName, pBytes, dwCount);

                // 釋放先前分配的非托管記憶體
                Marshal.FreeCoTaskMem(pBytes);
            }
            catch (Win32Exception ex)
            {
                WriteLog(ex.Message);
                flag = false;
            }
            return flag;
        }

        /// <summary>
        /// 寫入日誌方法
        /// </summary>
        /// <param name="msg">記錄信息</param>
        public static void WriteLog(string msg)
        {
            string str = string.Empty;
            string path = AppDomain.CurrentDomain.BaseDirectory + "log\\" + DateTime.Now.ToString("yyyy-MM-dd") + ".txt";

            FileStream filestream = new FileStream(path, FileMode.OpenOrCreate);

            str += "************" + DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + "************\r\n";
            str += msg;
            str += "************" + DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + "************\r\n";

            FileStream fs = new FileStream(path, FileMode.Append);
            StreamWriter sw = new StreamWriter(fs, System.Text.Encoding.Default);
            sw.WriteLine(str);

            sw.Flush();

            sw.Close();
            sw.Dispose();

            fs.Close();
            fs.Dispose();
        }
    }

以上就是全部代碼了,調用就很簡單了,方法如下:

private void Print_Click(object sender, EventArgs e)
        {
            //列印文件的路徑,該方法獲取到的文件路徑在應用程式 bin\Debug\ 目錄下,所以 ../../ 向上兩級目錄
            string fileName = AppDomain.CurrentDomain.BaseDirectory + @"../../File/demo.txt";

            if (PrintCode.SendFileToPrinter("PrinterName", fileName))
            {
                MessageBox.Show("文件已成功發送至列印隊列!","提示信息");
            }
        }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、通過top命令找到高耗CPU的進程,記下PID 2、使用命令ps -mp PID -o THREAD,tid,time找到高耗CPU的那些線程 3、jstack PID 4、對每個高耗CPU的線程,把線程id,轉換為16進程數( 可以使用命令:printf "%x\n" tid ),然後從thr ...
  • 今天因為C盤的記憶體變得不夠多而再次選擇了重裝系統,重裝系統,win7的重裝系統的映像網址為:http://win.njbda.cn/win7.html 我選擇的是“雨林木風”的64位系統,那麼問題來了,每個人的電腦本機系統不同,那麼我們需要根據自己的系統進行對win7的下載,那麼,如果像我一樣不記得 ...
  • 簡介 sed 是一種線上編輯器,它一次處理一行內容。處理時,把當前處理的行存儲在臨時緩衝區中,稱為“模式空間”(pattern space),接著用sed命令處理緩衝區中的內容,處理完成後,把緩衝區的內容送往屏幕。接著處理下一行,這樣不斷重覆,直到文件末尾。文件內容並沒有 改變,除非你使用重定向存儲 ...
  • 方法有二: 1、臨時開啟(重啟即失效) echo 1 > /proc/sys/net/ipv4/ip_forward 或 vim /proc/sys/net/ipv4/ip_forward,將0修改為1,保存退出 2、永久開啟 sed -i 's/net.ipv4.ip_forward = 0/ne ...
  • 閱讀目錄 1. 介紹 2. 軟體準備 3. 建立SVN Server倉庫 4. 配置安裝PHP&IF.SVNadmin 5. 啟動服務 1.介紹 公司最近想把Windows server平臺的SVN遷移到Linux平臺;這邊經過測試成功,所以寫個隨筆記錄一下 今天寫的是CentOS7上搭建基於Apa ...
  • 1.Lazy<T>的使用 無意間看到一段代碼,在創建對象的時候使用了Lazy,顧名思義Lazy肯定是延遲載入,那麼它具體是如何創建對象,什麼時候創建對象了? 先看這段示列代碼: 使用非常簡單,把 OrderService 放到Lazy<T> 中,然後 _orderSrv.Value 的時候才真正創建 ...
  • 一般拿Timer和Quartz相比較的,簡直就是對Quartz的侮辱,兩者的功能根本就不在一個層級上,如本篇介紹的Quartz強大的集群機制,可以採用基於 sqlserver,mysql的集群方案,當然還可以在第三方插件的基礎上實現quartz序列化到熱炒的mongodb,redis,震撼力可想而知 ...
  • C# 7.0已經出來一段時間了,大家都知道新特性裡面有個對元組的優化:ValueTuple。這裡利用詳盡的例子詳解Tuple VS ValueTuple(元組類VS值元組),10分鐘讓你更瞭解ValueTuple的好處和用法。 如果您對Tuple足夠瞭解,可以直接跳過章節”回顧Tuple”,直達章節 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...