• <ruby id="5koa6"></ruby>
    <ruby id="5koa6"><option id="5koa6"><thead id="5koa6"></thead></option></ruby>

    <progress id="5koa6"></progress>

  • <strong id="5koa6"></strong>
    • 軟件測試技術
    • 軟件測試博客
    • 軟件測試視頻
    • 開源軟件測試技術
    • 軟件測試論壇
    • 軟件測試沙龍
    • 軟件測試資料下載
    • 軟件測試雜志
    • 軟件測試人才招聘
      暫時沒有公告

    字號: | 推薦給好友 上一篇 | 下一篇

    VB應用程序中打印條形碼的兩種方法

    發布: 2007-7-14 20:28 | 作者: 佚名    | 來源: 網絡轉載     | 查看: 17次 | 進入軟件測試論壇討論

    領測軟件測試網 作者:四川、李佑民

    條形碼作為一種機器可識別的圖形,它能快速、準確地標識某種產品或商品,在許多數據庫應用中起作很重要的作用,如超市收銀、車站售票等場合。當某件物品上帶有的條形碼被條碼掃描器正確解讀后,將會得到該物品的唯一標識字符串,通過檢索數據庫我們就可以很容易知道它的一些其它屬性并作相應處理。雖然在Inte.net上能找到許多免費和不免費的條形碼打印控件,但是這些控件除了使用不方便外,還有一個最大的缺點:它們的打印輸出不能和我們的程序共存在一個打印頁面上,比如說在一個過程中,我們先向系統 Printer 中輸出一些內容,然后再調用控件的條形碼打印方法,最后打印的結果為兩頁!,如果現在我們要處理一張車票,上面不僅要打印條形碼,還要有終點站和票價等信息,那么控件就變得不可用。對程序員來說,可能還是希望能了解條形碼打印的原理,本文提出兩種打印方法與同行們探討。

    一、直接利用有條形碼打印功能的打印機

      有許多打印機能夠直接打印條形碼,但在 VB 中,我們在DOS時代熟悉的LPRINT語句已經不能再使用了,打印操作被Windows的Spool系統完全接管,輸出是以“頁”為單位,所有的打印輸出都被Windows轉換為圖形發送給打印驅動程序。而要使打印機打印條形碼就必須將對應的ESC序列直接發送給它,因此我們就要想辦法避開Windows的Spool系統,也就是說再程序中不能使用Printer對象和Printers集合處理打印輸出,在VB中要將ESC指令直接發送給打印機至少有三種方法,前兩種方法是調用Windows API 函數:Escape()和SpoolFile(),第三種是最容易的方法:打開打印機端口進行二進制存取,我們主要考慮這種方法。

      即使在Windows時代,”LPT1:”和”PRN”仍然是可用的,下面我們先作一個試驗:打開一個DOS窗口,在提示符下輸入COPY CON LPT1:回車,然后隨便輸入一些字符,最后按F6鍵,打印機就開始工作了,它將打印出你輸入的那些字符!下面的代碼演示了直接將指令和字符發送給打印機:

    Private Sub Command1_Click()

      Dim strOut As String

      StrOut = “這是直接發送到打印機端口的字符串”

      ‘ 打開打印機端口,其中的”LPT1:”可能需要根據你的打印機設置而改變

      Open “LPT1:” For Binary Access Write As #1

      ‘ 發送給打印機,注意語句的最后一個參數必須是變量

        Put #1, ,strOut

        ‘ 關閉打印機端口

        Close #1

    End Sub

      各種打印機打印條形碼的指令可能不同,比如將上面的變量 strOut賦值為:

      strOut = Chr(28) & “P” & Chr(5) & Chr(2) & Chr(3) & Chr(3) & Chr(6) & “012345”

      將在 AR2400 打印機上打印出內容為”012345”的 CODE39 格式的條形碼。具體的打印控制指令請參考打印機手冊。

      用這種方法的缺點:一是過份依賴打印機本身,而有條形碼打印功能的打印機通常要比普通打印機昂貴,這會使構造應用系統不夠經濟;二是所有的打印輸出都必須你自己處理,比如打印定位就很浪費時間。

    二、利用畫圖方式輸出到普通打印機

      條形碼的編碼規則不外乎是通過線條和線條間間隙的寬窄不同來表示二進制的1和0,只要我們了解了條形碼的編碼規則,完全可以用畫圖的方式在普通打印機上得到可以接受的效果。下面我們就使用最普遍的CODE39碼進行討論。

      CODE39碼的編碼規則是:

    1、 每五條線表示一個字符;

    2、 粗線表示1,細線表示0;

    3、 線條間的間隙寬的表示1,窄的表示0;

    4、 五條線加上它們之間的四條間隙就是九位二進制編碼,而且這九位中必定有三位是1,所以稱為39碼;

    5、 條形碼的首尾各一個*標識開始和結束

      在我們的程序中,給常用的字符都進行編碼,解讀時先取線條粗細,再取間隙寬窄,如:


      上圖中的字符*就可以解讀為 001101000,字符3解讀為 110000100

      下面就是我們給出的子過程:

    注釋: 將字符串 strBarCode 對應的條形碼輸出到缺省打印機

    Private Sub PrintBarCode( _

      ByVal strBarCode As String, _

      Optional ByVal intXPos As Integer = 0, _

      Optional ByVal intYPos As Integer = 0, _

      Optional ByVal intPrintHeight As Integer = 10, _

      Optional ByVal bolPrintText As Boolean = True _

    )

    注釋: 參數說明:

    注釋: strBarCode    - 要打印的條形碼字符串

    注釋: intXPos, intYPos - 打印條形碼的左上角坐標(缺省為(0,0),坐標刻度為:毫米)

    注釋: intHeight     - 打印高度(缺省為一厘米,坐標刻度為:毫米)

    注釋: bolPrintText   - 是否打印人工識別字符(缺省為true)


    注釋: "0"-"9","A-Z","-","%","$"和"*" 的條碼編碼格式,總共 40 個字符

    Static strBarTable(39) As String

    注釋: 初始化條碼編碼格式表

      strBarTable(0) = "001100100"   注釋: 0

      strBarTable(1) = "100010100"   注釋: 1

      strBarTable(2) = "010010100"   注釋: 2

      strBarTable(3) = "110000100"   注釋: 3

      strBarTable(4) = "001010100"   注釋: 4

      strBarTable(5) = "101000100"   注釋: 5

      strBarTable(6) = "011000100"   注釋: 6

      strBarTable(7) = "000110100"   注釋: 7

      strBarTable(8) = "100100100"   注釋: 8

      strBarTable(9) = "010100100"   注釋: 9

      strBarTable(10) = "100010010"  注釋: A

      strBarTable(11) = "010010010"  注釋: B

      strBarTable(12) = "110000010"  注釋: C

      strBarTable(13) = "001010010"  注釋: D

      strBarTable(14) = "101000010"  注釋: E

      strBarTable(15) = "011000010"  注釋: F

      strBarTable(16) = "000110010"  注釋: G

      strBarTable(17) = "100100010"  注釋: H

      strBarTable(18) = "010100010"  注釋: I

      strBarTable(19) = "001100010"  注釋: J

      strBarTable(20) = "100010001"  注釋: K

      strBarTable(21) = "010010001"  注釋: L

      strBarTable(22) = "110000001"  注釋: M

      strBarTable(23) = "001010001"  注釋: N

      strBarTable(24) = "101000001"  注釋: O

      strBarTable(25) = "011000001"  注釋: P

      strBarTable(26) = "000110001"  注釋: Q

      strBarTable(27) = "100100001"  注釋: R

      strBarTable(28) = "010100001"  注釋: S

      strBarTable(29) = "001100001"  注釋: T

      strBarTable(30) = "100011000"  注釋: U

      strBarTable(31) = "010011000"  注釋: V

      strBarTable(32) = "110001000"  注釋: W

      strBarTable(33) = "001011000"  注釋: X

      strBarTable(34) = "101001000"  注釋: Y

      strBarTable(35) = "011001000"  注釋: Z

      strBarTable(36) = "000111000"  注釋: -

      strBarTable(37) = "100101000"  注釋: %

      strBarTable(38) = "010101000"  注釋: $

      strBarTable(39) = "001101000"  注釋: *

      If strBarCode = "" Then Exit Sub 注釋: 不打印空串

      注釋: 保存打印機 ScaleMode

      Dim intOldScaleMode As ScaleModeConstants

      intOldScaleMode = Printer.ScaleMode

      注釋: 保存打印機 DrawWidth

      Dim intOldDrawWidth As Integer

      intOldDrawWidth = Printer.DrawWidth

      注釋: 保存打印機 Font

      Dim fntOldFont As StdFont

      Set fntOldFont = Printer.Font

      Printer.ScaleMode = vbTwips 注釋: 設置打印用的坐標刻度為緹(twip=1)

      Printer.DrawWidth = 1   注釋: 線寬為 1

      Printer.FontName = "宋體" 注釋: 打印在條碼下方字符的字體和大小

      Printer.FontSize = 10

      Dim strBC As String     注釋: 要打印的條碼字符串

      strBC = Ucase(strBarCode)

      注釋: 將以毫米表示的 X 坐標轉換為以緹表示

      Dim x As Integer

      x = Printer.ScaleX(intXPos, vbMillimeters, vbTwips)

      注釋: 將以毫米表示的 Y 坐標轉換為以緹表示

      Dim y As Integer

      y = Printer.ScaleY(intYPos, vbMillimeters, vbTwips)

      注釋: 將以毫米表示的高度轉換為以緹表示

      Dim intHeight As Integer

      intHeight = Printer.ScaleY(intPrintHeight, vbMillimeters, vbTwips)


      注釋: 是否在條形碼下方打印人工識別字符

      If bolPrintText = True Then

        注釋: 條碼打印高度要減去下面的字符顯示高度

        intHeight = intHeight - Printer.TextHeight(strBC)

      End If

      Const intWidthCU As Integer = 30 注釋: 粗線和寬間隙寬度

      Const intWidthXI As Integer = 10 注釋: 細線和窄間隙寬度

      Dim intIndex As Integer      注釋: 當前處理的字符串索引

      Dim i As Integer, j As Integer, k As Integer  注釋: 循環控制變量



      注釋: 添加起始字符

      If Left(strBC, 1) <> "*" Then

        strBC = "*" & strBC

      End If

      注釋: 添加結束字符

      If Right(strBC, 1) <> "*" Then

        strBC = strBC & "*"

      End If

      

      注釋: 循環處理每個要顯示的條碼字符

      For i = 1 To Len(strBC)

        注釋: 確定當前字符在 strBarTable 中的索引

        Select Case Mid(strBC, i, 1)

        Case "*"

          intIndex = 39

        Case "$"

          intIndex = 38

        Case "%"

          intIndex = 37

        Case "-"

          intIndex = 36

        Case "0" To "9"

          intIndex = CInt(Mid(strBC, i, 1))

        Case "A" To "Z"

          intIndex = Asc(Mid(strBC, i, 1)) - Asc("A") + 10

        Case Else

          MsgBox "要打印的條形碼字符串中包含無效字符!當前版本只支持字符 注釋:0注釋:-注釋:9注釋:,注釋:A注釋:-注釋:Z注釋:,注釋:-注釋:,注釋:%注釋:,注釋:$注釋:和注釋:*注釋:"

        End Select

        注釋: 是否在條形碼下方打印人工識別字符

        If bolPrintText = True Then

          Printer.CurrentX = x

          Printer.CurrentY = y + intHeight

          Printer.Print Mid(strBC, i, 1)

        End If

        For j = 1 To 5

          注釋: 畫細線

          If Mid(strBarTable(intIndex), j, 1) = "0" Then

            For k = 0 To intWidthXI - 1

              Printer.Line (x + k, y)-Step(0, intHeight)

            Next k

            x = x + intWidthXI

          注釋: 畫寬線

          Else

            For k = 0 To intWidthCU - 1

              Printer.Line (x + k, y)-Step(0, intHeight)

            Next k

            x = x + intWidthCU

          End If

          注釋: 每個字符條碼之間為窄間隙

          If j = 5 Then

            x = x + intWidthXI * 3

            Exit For

          End If

          注釋: 窄間隙

          If Mid(strBarTable(intIndex), j + 5, 1) = "0" Then

            x = x + intWidthXI * 3

          注釋: 寬間隙

          Else

            x = x + intWidthCU * 2

          End If

        Next j

      Next i

      注釋: 恢復打印機 ScaleMode

      Printer.ScaleMode = intOldScaleMode

      注釋: 恢復打印機 DrawWidth

      Printer.DrawWidth = intOldDrawWidth

      注釋: 恢復打印機 Font

      Set Printer.Font = fntOldFont

    End Sub
      最理想的情況是將它做成一個控件,在控件中提供一個打印方法,該方法實現與上

      那個過程大致相同,只是不能在控件中直接使用VB的Printer對象,否則VB會將你在控件中的打印輸出處理為一個單獨的頁面,而是應該將Printer.hDc傳給它,通過調用那些需要指定 HDC 的Windows API函數實現與容器的打印輸出在一個頁面上,比如我們可以這樣定義這個控件的打印方法:

    注釋: PrintIt 方法將對應的條形碼輸出到缺省打印機

    Public Sub PrintIt(ByVal PrintDC As Long, _

       Optional ByVal intXPos As Integer = 0, _

      Optional ByVal intYPos As Integer = 0, _

       Optional ByVal intPrintHeight As Integer = 10)

      既然不能使用Printer對象,那么畫線和輸出文字也不能使用Printer對象的Line和Print方法,在我們的程序中至少要申明以下三個Windows API函數:

    ‘ 移動畫筆的位置

    Private Declare Function MoveToEx Lib "gdi32" (ByVal hdc As Long, ByVal x As Long, ByVal y As Long, lpPoint As POINTAPI) As Long

    ‘ 從畫筆的當前位置到(x,y)畫一條線

    Private Declare Function LineTo Lib "gdi32" (ByVal hdc As Long, ByVal x As Long, ByVal y As Long) As Long

    ‘ 在(x,y)處輸出一個字符串

    Private Declare Function TextOut Lib "gdi32" Alias "TextOutA" (ByVal hdc As Long, ByVal x As Long, ByVal y As Long, ByVal lpString As String, ByVal nCount As Long) As Long

    ‘ MoveToEx() 函數需要的參數

    Private Type POINTAPI

      xp As Long

      yp As Long

    End Type

    Dim papi As POINTAPI

    畫線操作為(原來的Printer.Line函數):

    MoveToEx PrintDC, x + k, y, papi

    LineTo PrintDC, x + k, y + intHeight + 1

    打印字符為(原來的Printer.Print函數):

    TextOut PrintDC, x, y + intHeight, Mid(strBC, i + 1, 1), 1 

    延伸閱讀

    文章來源于領測軟件測試網 http://www.kjueaiud.com/


    關于領測軟件測試網 | 領測軟件測試網合作伙伴 | 廣告服務 | 投稿指南 | 聯系我們 | 網站地圖 | 友情鏈接
    版權所有(C) 2003-2010 TestAge(領測軟件測試網)|領測國際科技(北京)有限公司|軟件測試工程師培訓網 All Rights Reserved
    北京市海淀區中關村南大街9號北京理工科技大廈1402室 京ICP備2023014753號-2
    技術支持和業務聯系:info@testage.com.cn 電話:010-51297073

    軟件測試 | 領測國際ISTQBISTQB官網TMMiTMMi認證國際軟件測試工程師認證領測軟件測試網

    老湿亚洲永久精品ww47香蕉图片_日韩欧美中文字幕北美法律_国产AV永久无码天堂影院_久久婷婷综合色丁香五月

  • <ruby id="5koa6"></ruby>
    <ruby id="5koa6"><option id="5koa6"><thead id="5koa6"></thead></option></ruby>

    <progress id="5koa6"></progress>

  • <strong id="5koa6"></strong>