虽说现在已经有越来越多的人转向用C#来做界面的开发,但是对于一个习惯了C这种半面向对象语言的人来说,MFC依然是难割舍的存在。最近在做些数据库相关的事,因为是数据库,自然离不开打印报表之类的东西,所以顺带的也去研究了下如何用MFC来进行打印相关的的
虽说现在已经有越来越多的人转向用C#来做界面的开发,但是对于一个习惯了C++这种半面向对象语言的人来说,MFC依然是难割舍的存在。最近在做些数据库相关的事,因为是数据库,自然离不开打印报表之类的东西,所以顺带的也去研究了下如何用MFC来进行打印相关的的操作。
本篇文章参考的比较多的是这篇文章和MSDN的示例代码,另外这篇文章也提供了很多有用的参考,如果想看原文的话可以去看看。
直接用微软提供的CPrintDialog类进行打印的相关操作是非常简单的,本人将代码封装成为两个函数,除了实际往一页纸上绘制东西的部分在第二个函数中进行定义外,其他与打印相关的操作都在第一个函数上实现了。具体如下:
打印相关操作主函数:
void CPrinttToPrinterDlg::OnBnClickedButtonPrint() { CPrintDialog dlg(FALSE, PD_ALLPAGES, NULL);//打印对话框对象定义,参数一设置弹出对话框为打印对话框,参数二设置打印范围为全部页面,参数三可指定打印机,这里缺省即可 /*打印对话框的初始值设置,在实际应用中可根据需要进行设置,如通过预先计算得到要打印的页面范围等*/ dlg.m_pd.nMinPage = 1;//指定开始/结束页码编辑控件的页码范围的最小值,若nMinPage=nMaxPage,则“页码范围”单选钮和开始/结束页码编辑控件被灰化 dlg.m_pd.nMaxPage = 2;//指定开始/结束页码编辑控件的页码范围的最大值 dlg.m_pd.nCopies = 1;//指定拷贝份数编辑控件的初始值 dlg.m_pd.nFromPage = 1;//指定开始页码编辑控件的初始值 dlg.m_pd.nToPage = 2;//指定结束页码编辑控件的初始值 if (dlg.DoModal() == IDOK) { HDC hdcPrinter = dlg.GetPrinterDC(); if (hdcPrinter == NULL)//检取设备环境的句柄,检索不到则提醒并退出 { MessageBox(_T("Buy a printer!")); } else { /*创建图形设备环境并与打印机设备关联,在上面绘图就相当于往打印机要打出的纸上绘图*/ CDC dcPrinter; dcPrinter.Attach(hdcPrinter); /* *在打印设置过程中有些东西我们希望用户来选,所以使用了对话框的形式,但是有些 *东西比如纸张大小和打印方向等我们希望它是固定的,不能让用户随意设置造成麻烦 *因此,我们利用已经得到的打印机图形设备环境,通过修改它来实现固定设置的功能 */ LPDEVMODE pDevMode; pDevMode = (LPDEVMODE)GlobalLock(dlg.m_pd.hDevMode); pDevMode->dmOrientation = DMORIENT_LANDSCAPE;//设置打印方向为横向 pDevMode->dmPaperSize = DMPAPER_A4;//设置纸张大小为A4 dcPrinter.ResetDC(pDevMode); //通知打印机驱动程序接收打印文档并开始打印 DOCINFO docinfo; memset(&docinfo, 0, sizeof(docinfo)); docinfo.cbSize = sizeof(docinfo); docinfo.lpszDocName = _T("CDC::StartDoc() Code Fragment"); //如果初始化失败则提醒并退出 if (dcPrinter.StartDoc(&docinfo) < 0) { MessageBox(_T("Printer wouldn't initalize")); } else { /*打印的纸张大小我们是需要知道的,此外我们还需要设备像素点和实际长度单位的换算关系即DPI*/ int xDPI = GetDeviceCaps(hdcPrinter, LOGPIXELSX);//返回X方向上每英寸的设备点数,即DPI float mmdpix = xDPI / 25.4;//每毫米所占的设备点数 int yDPI = GetDeviceCaps(hdcPrinter, LOGPIXELSY);//返回Y方向上每英寸的设备点数,即DPI float mmdpiy = yDPI / 25.4;//每毫米所占的设备点数 float printerscreenx, printerscreeny;//这里得到打印机屏幕的宽高,也就是纸张的大小,会比实际小约6mm printerscreenx = GetDeviceCaps(hdcPrinter, HORZSIZE); printerscreeny = GetDeviceCaps(hdcPrinter, VERTSIZE); /*因为要打印的页码范围和份数是用户选择的,所以在此对用户选择的项做处理*/ int pagebegin, pageend;//打印范围 if (dlg.PrintAll())//看用户是否选择了全部打印这一项,是的话则全部打印 { pagebegin = dlg.m_pd.nMinPage; pageend = dlg.m_pd.nMaxPage; } else if (dlg.PrintRange())//用户选择了选定页码范围项 { pagebegin = dlg.m_pd.nFromPage; pageend = dlg.m_pd.nToPage; } else//用户选择了选定范围打印,因为我们在这里没有提供选定范围的功能,所以用户一旦选择这里当成是操作错误处理即可 { MessageBox(_T("Could not choise this one")); dcPrinter.AbortDoc();//错误退出区别于EndDoc } int ncopy = dlg.m_pd.nCopies;//用户选择打印分数 /*在这里进行打印工作*/ while (ncopy--)//逐份打印 { for (int page = pagebegin; page <= pageend; page++)//从选定范围开始打印 { if (dcPrinter.StartPage() < 0) { MessageBox(_T("Could not start page")); dcPrinter.AbortDoc();//错误退出区别于EndDoc } else//如果进入这里则绘出要打印内容并结束掉一页的打印 { doThePrint(dcPrinter, page, mmdpix, mmdpiy, printerscreenx, printerscreeny); dcPrinter.EndPage(); } } } } dcPrinter.EndDoc();//打印完成退出 dcPrinter.Detach();//释放DC } } }实际往一页纸上打印的内容在此函数中进行定义,下面函数内容实现在打印的纸上绘制一个大小稍小于页面的绿色矩形:
void CPrinttToPrinterDlg::doThePrint(CDC &dc,int page, float mdpix, float mdpiy, float mpagex, float mpagey)//打印实际绘图函数,往参数一传入的CDC上面绘图 { CPen pen, *pOldPen;// 定义笔对象和指针 // 创建10单位宽的绿色实心笔 pen.CreatePen(PS_SOLID, 100, RGB(0, 255, 0)); pOldPen = dc.SelectObject(&pen);// 选入绿色笔 dc.Re【本文来自鸿网互联 (http://www.68idc.cn)】ctangle(0 * mdpix, 0 * mdpiy, mpagex*mdpix, mpagey*mdpiy);// 画矩形 dc.SelectObject(pOldPen);// 选出绿色笔 pen.DeleteObject();// 删除绿色笔 }
到此文章就已经写得七七八八了,调用主函数就可以实现往打印机上打印动东西的功能。至于打印什么东西,如何去打,就是GDI绘图的事了。另外由于我们还没有实现打印预览的功能,所以可以用虚拟打印机来看效果,当然如果有钱,也可以直接用真的打印机来看啦哈哈,本人用的是finepring,简而言之就两个字形容,神器!
打印预览还有如何去打印数据表格这些功能本人后面如果研究实现了的话也会贴上来,或者发个链接上来,这篇文章就先到这里吧。

MySQL和SQLite的主要区别在于设计理念和使用场景:1.MySQL适用于大型应用和企业级解决方案,支持高性能和高并发;2.SQLite适合移动应用和桌面软件,轻量级且易于嵌入。

MySQL中的索引是数据库表中一列或多列的有序结构,用于加速数据检索。1)索引通过减少扫描数据量提升查询速度。2)B-Tree索引利用平衡树结构,适合范围查询和排序。3)创建索引使用CREATEINDEX语句,如CREATEINDEXidx_customer_idONorders(customer_id)。4)复合索引可优化多列查询,如CREATEINDEXidx_customer_orderONorders(customer_id,order_date)。5)使用EXPLAIN分析查询计划,避

在MySQL中使用事务可以确保数据一致性。1)通过STARTTRANSACTION开始事务,执行SQL操作后用COMMIT提交或ROLLBACK回滚。2)使用SAVEPOINT可以设置保存点,允许部分回滚。3)性能优化建议包括缩短事务时间、避免大规模查询和合理使用隔离级别。

选择PostgreSQL而非MySQL的场景包括:1)需要复杂查询和高级SQL功能,2)要求严格的数据完整性和ACID遵从性,3)需要高级空间功能,4)处理大数据集时需要高性能。PostgreSQL在这些方面表现出色,适合需要复杂数据处理和高数据完整性的项目。

MySQL数据库的安全可以通过以下措施实现:1.用户权限管理:通过CREATEUSER和GRANT命令严格控制访问权限。2.加密传输:配置SSL/TLS确保数据传输安全。3.数据库备份和恢复:使用mysqldump或mysqlpump定期备份数据。4.高级安全策略:使用防火墙限制访问,并启用审计日志记录操作。5.性能优化与最佳实践:通过索引和查询优化以及定期维护兼顾安全和性能。

如何有效监控MySQL性能?使用mysqladmin、SHOWGLOBALSTATUS、PerconaMonitoringandManagement(PMM)和MySQLEnterpriseMonitor等工具。1.使用mysqladmin查看连接数。2.用SHOWGLOBALSTATUS查看查询数。3.PMM提供详细性能数据和图形化界面。4.MySQLEnterpriseMonitor提供丰富的监控功能和报警机制。

MySQL和SQLServer的区别在于:1)MySQL是开源的,适用于Web和嵌入式系统,2)SQLServer是微软的商业产品,适用于企业级应用。两者在存储引擎、性能优化和应用场景上有显着差异,选择时需考虑项目规模和未来扩展性。

在需要高可用性、高级安全性和良好集成性的企业级应用场景下,应选择SQLServer而不是MySQL。1)SQLServer提供企业级功能,如高可用性和高级安全性。2)它与微软生态系统如VisualStudio和PowerBI紧密集成。3)SQLServer在性能优化方面表现出色,支持内存优化表和列存储索引。


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

适用于 Eclipse 的 SAP NetWeaver 服务器适配器
将Eclipse与SAP NetWeaver应用服务器集成。

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

DVWA
Damn Vulnerable Web App (DVWA) 是一个PHP/MySQL的Web应用程序,非常容易受到攻击。它的主要目标是成为安全专业人员在合法环境中测试自己的技能和工具的辅助工具,帮助Web开发人员更好地理解保护Web应用程序的过程,并帮助教师/学生在课堂环境中教授/学习Web应用程序安全。DVWA的目标是通过简单直接的界面练习一些最常见的Web漏洞,难度各不相同。请注意,该软件中

Atom编辑器mac版下载
最流行的的开源编辑器

禅工作室 13.0.1
功能强大的PHP集成开发环境