相逢情便深,恨不相逢早:
背景:
开始做中谷项目时我还不知道怎么用DLL。后来看别人用才试着去学习使用。虽然现在已经会用了,但DLL的强大之处我了解的还远远不够。下面先简单说下DLL。
DLL是神马?
Windows操作系统是非常依赖于动态链接库(DLL)中的函数和数据的,实际上操作系统中几乎所有的内容都由DLL以一种或另外一种形式代表着,例如显示的字体和图标存储在GDI DLL中、显示windows桌面和处理用户的输入所需要的代码被存储在一个User DLL中、Windows编程所需要的大量API函数也呗包含在Kernel DLL中。
使用DLL优点多多:
最主要的一点是多个应用程序、甚至不同语言编写的应用程序可以共享一个DLL文件,实现资源“共享”,这样就大大缩小了应用程序的执行代码,有效利用内存;另一个优点是DLL文件作为一个单独的程序模块,封装性、独立性好,在软件需要升级的时候,开发人员只需要修改相应的DLl文件就可以了,而且,当DLL中的函数改变后,只要不是参数的改变,程序代码并不需要重新编译,这在编程时大大提高了软件开发和维护的效率。
纸上得来终觉浅:
打印:
很多窗体都需要打印功能,为了调用方便,我把打印写成一个类封装成DLL,然后在程序里添加引用。Print类代码也附到这里,不是重点。
Imports System.Drawing
Imports System.Drawing.Printing
Imports System.Windows.Forms
Public Class Class1
Private PrintFont As New Font("宋体", 10)
Private PrintLines As Integer = 50
Private PrintRecordNumber As Integer = 45
Private DataGridSource As DataGridView
Private ev As PrintPageEventArgs
Private PrintDataGrid As PrintDocument
Private PrintPriview As PrintPreviewDialog
Private PageSetup As PageSetupDialog
Private PrintScale As Double = 1
Private DataGridColumn As DataColumn
Private DataGridRow As DataRow
Private DataGridTable As DataTable
Private Cols As Integer
Private Rows As Integer = 1
Private ColsCount As Integer
Private PrintingLineNumber As Integer = 0
Private PageRecordNumber As Integer
Dim X_unit As Integer
Dim Y_unit As Integer
Dim _strPrint As String
Private PrintingPageNumber As Integer = 0
Private PageNumber As Integer
Private PrintRecordLeave As Integer
Private PrintRecordComplete As Integer = 0
''' <summary>
''' 设置打印字体
''' </summary>
''' <value></value>
''' <remarks></remarks>
Public WriteOnly Property setPrintFont() As System.Drawing.Font
Set(ByVal Value As System.Drawing.Font)
PrintFont = Value
End Set
End Property
Public WriteOnly Property setPrintRecordNumber() As Integer
Set(ByVal Value As Integer)
PrintRecordNumber = Value
End Set
End Property
''' <summary>
''' 设置表头
''' </summary>
''' <value></value>
''' <returns></returns>
''' <remarks></remarks>
Public Property setstrPrint() As String
Set(ByVal value As String)
_strPrint = value
End Set
Get
Return _strPrint
End Get
End Property
Sub New(ByVal TableSource As DataGridView)
DataGridSource = TableSource
DataGridTable = New DataTable
DataGridTable = DataGridSource.DataSource()
ColsCount = DataGridTable.Columns.Count
End Sub
Public Sub Print()
Try
PrintDataGrid = New System.Drawing.Printing.PrintDocument
AddHandler PrintDataGrid.PrintPage, AddressOf Me.PrintDataGrid_PrintPage
'*
PageSetup = New PageSetupDialog
PageSetup.PageSettings = PrintDataGrid.DefaultPageSettings
If PageSetup.ShowDialog() = DialogResult.Cancel Then
Exit Sub
End If
'*
If PrintDataGrid.DefaultPageSettings.Landscape = False Then
PrintLines = PrintDataGrid.DefaultPageSettings.PaperSize.Height / (PrintFont.Height + 5)
Else
PrintLines = PrintDataGrid.DefaultPageSettings.PaperSize.Width / (PrintFont.Height + 5)
End If
'*
If PrintDataGrid.DefaultPageSettings.PaperSize.PaperName.ToString = "custom" Then
End If
'*
PrintPriview = New PrintPreviewDialog
PrintPriview.Document = PrintDataGrid
PrintPriview.ShowDialog()
Catch ex As Exception
MessageBox.Show("错误:" & ex.ToString)
End Try
End Sub
Private Sub PrintDataGrid_PrintPage(ByVal sender As Object, ByVal ev As System.Drawing.Printing.PrintPageEventArgs)
Dim strPrint As String = Nothing
Dim DrawBrush As New SolidBrush(System.Drawing.Color.Blue)
Dim X As Integer
Dim Y As Integer
Dim DrawPoint As New PointF(X, Y)
'Dim row_count As Integer
PrintRecordLeave = DataGridTable.Rows.Count - PrintRecordComplete
'*
PageNumber = PrintRecordLeave / PrintRecordNumber
PrintingPageNumber = 0
'*
If PrintDataGrid.DefaultPageSettings.Landscape = True Then
X_unit = PrintDataGrid.DefaultPageSettings.PaperSize.Height / (DataGridTable.Columns.Count + 2)
Y_unit = PrintDataGrid.DefaultPageSettings.PaperSize.Width / PrintLines
Else
X_unit = PrintDataGrid.DefaultPageSettings.PaperSize.Width / (DataGridTable.Columns.Count + 2)
Y_unit = PrintDataGrid.DefaultPageSettings.PaperSize.Height / PrintLines
End If
'*
If DataGridTable.Rows.Count - PrintingPageNumber * PrintRecordNumber >= PrintRecordNumber Then
PageRecordNumber = PrintRecordNumber
Else
PageRecordNumber = (DataGridTable.Rows.Count - PrintingPageNumber * PrintRecordNumber) Mod PrintRecordNumber
End If
While PrintingPageNumber <= PageNumber
'*
'strPrint = DataGridSource.CaptionText
DrawPoint = New PointF(X_unit, Y_unit)
ev.Graphics.DrawString(strPrint, PrintFont, DrawBrush, DrawPoint)
'*
Dim ColumnText(DataGridTable.Columns.Count) As String
Dim Cols As Integer
'Dim Table As Integer
For Cols = 0 To DataGridTable.Columns.Count - 1
ColumnText(Cols) = DataGridTable.Columns(Cols).ToString
'*
DrawPoint = New PointF(X_unit * (Cols + 1), Y_unit * 2)
ev.Graphics.DrawString(ColumnText(Cols), PrintFont, DrawBrush, DrawPoint)
Next
DrawPoint = New PointF(X_unit, Y_unit * 2)
Call DrawLine(DrawPoint, ev)
'*
Dim printingline As Integer = 0
'*
Dim strNonce As String = ""
Dim strUpData As String = ""
While printingline < PageRecordNumber
DataGridRow = DataGridTable.Rows(PrintRecordComplete)
'*
For Cols = 0 To DataGridTable.Columns.Count - 1
DrawPoint.X = X_unit * (Cols + 1)
DrawPoint.Y = Y_unit * (printingline + 1 + 2)
If Cols = 0 Then
If strUpData <> "" And strNonce <> "" Then
If strUpData <> DataGridRow(ColumnText(0)) Then
ev.HasMorePages = True
Exit Sub
End If
End If
End If
ev.Graphics.DrawString(Convert.ToString(DataGridRow(ColumnText(Cols))), PrintFont, DrawBrush, DrawPoint)
strUpData = DataGridRow(ColumnText(0))
Next
DrawPoint.X = X_unit * 1
DrawPoint.Y = Y_unit * (printingline + 1 + 2)
Call DrawLine(DrawPoint, ev)
printingline += 1
PrintRecordComplete += 1
'*
If PrintRecordComplete >= DataGridTable.Rows.Count Then
ev.HasMorePages = False
Exit Sub
End If
End While
PrintingPageNumber += 1
If PrintingPageNumber > PageNumber Then
ev.HasMorePages = False
Else
ev.HasMorePages = True
Exit While
End If
End While
End Sub
Private Sub DrawLine(ByVal point As PointF, ByVal ev As System.Drawing.Printing.PrintPageEventArgs)
Dim blackPen As New Pen(System.Drawing.Color.Black, 1)
ev.Graphics.DrawLine(blackPen, point.X, point.Y + PrintFont.Height, point.X * (ColsCount + 1), point.Y + PrintFont.Height)
End Sub
End Class
在事件中调用时代码如下:
Private Sub btnPrint_Click(sender As Object, e As EventArgs) Handles btnPrint.Click
If dgvWindWorkLog.DataSource Is Nothing Then
MsgBox("没有可打印的内容,请重新操作!", vbOKOnly, "系统提示")
Exit Sub
End If
'加载dll文件,利用反射
Dim assembly As Assembly = assembly.LoadFrom(Application.StartupPath & "\\Print.dll")
Dim Print As New Print.Class1(Me.dgvWindWorkLog) '实例化打印类
Print.setstrPrint = "工作记录卡" '设置打印标题
Print.Print()
End Sub
几报加载浑错:
以上操作均没有错,如果在同一个Demo中测试,一切正常,打印效果老好了。然后将dll文件拷贝到中谷项目目录下,在事件中调用,提示错误。
测试了好多次都觉得不应该有错,我觉得哪里都一样就是到了项目中就运行不起来。还有下面这个问题,这两个问题我整整解决了一天。唉,只怪咱水平有限、经验有限啊!
报表:
不知道这叫不叫无独有偶,竟然同一天遇到两个一样的问题。
这也是做好的一个报表.grf,添加到程序中,结果每次运行到
Report = NewgrproLib.GridppReport
Report.LoadFromFile(Application.StartupPath& "\\报表.grf")时就提示上面的错误。
拨开云雾见青天:
我测试Demo的目标框架是.NET Framework 4.5,而中谷项目是.NETFramework 3.5。
于是……
宝剑锋从磨砺出:
No.1 经历了一次下次就知道如何解决和预防此类问题,自己做测试一定要细心。
其实报表中最大的问题,是我开发用的Grid++Report5.6,,给同组人的是5.0,(因为我很久前在机房合作开发时装了Grid++,安装文件找不到了,就跟当时给我传文件的那个同学又要了一份,结果两次的文件并不是同一版本)。这下热闹了,我们各种调试、各种注册、各种。。。终于决定删除重做。
No.2 合作开发一定要统一版本,不管是开发环境(包括比如VS版本、framework)、SVN客户端、调试工具以及开发过程中用到的控件等。因为稍不注意就可能是推翻重来,那可是血的代价啊!