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
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...