DLL(動(dòng)態(tài)鏈接庫(kù))文件是Dynamic Link Library的縮寫(xiě)形式,是一種允許程序共享執(zhí)行特殊任務(wù)所必需的代碼和其他資源的可執(zhí)行文件。其多數(shù)情況下是帶有DLL擴(kuò)展名的文件,但也可能是EXE或其他擴(kuò)展名。Windows提供的DLL文件中包含了允許基于Windows的程序在Windows環(huán)境下操作的許多函數(shù)和資源。動(dòng)態(tài)鏈接提供了一種方法,使進(jìn)程可以調(diào)用不屬于其可執(zhí)行代碼的函數(shù)。這些函數(shù)的可執(zhí)行代碼位于一個(gè)DLL中,該DLL包含一個(gè)或多個(gè)已被編譯、鏈接并與使用它們的進(jìn)程分開(kāi)存儲(chǔ)的函數(shù)。DLL還有助于共享數(shù)據(jù)和資源,多個(gè)應(yīng)用程序可同時(shí)訪問(wèn)內(nèi)存中單DLL副本的內(nèi)容?傊,DLL是一個(gè)包含可由多個(gè)程序同時(shí)使用的代碼和數(shù)據(jù)的庫(kù)。動(dòng)態(tài)鏈接是相對(duì)于靜態(tài)鏈接而言的。所謂靜態(tài)鏈接是指把要調(diào)用的函數(shù)或者過(guò)程鏈接到可執(zhí)行文件中,成為可執(zhí)行文件的一部分。換句話說(shuō),函數(shù)和過(guò)程的代碼就在程序的exe文件中,該文件包含了運(yùn)行時(shí)所需的全部代碼。當(dāng)多個(gè)程序都調(diào)用相同函數(shù)時(shí),內(nèi)存中就會(huì)存在這個(gè)函數(shù)的多個(gè)拷貝,這樣就浪費(fèi)了寶貴的內(nèi)存資源。而動(dòng)態(tài)鏈接所調(diào)用的函數(shù)代碼并沒(méi)有被拷貝到應(yīng)用程序的可執(zhí)行文件中去,而是僅僅在其中加入了所調(diào)用函數(shù)的描述信息(往往是一些重定位信息)。僅當(dāng)應(yīng)用程序被裝入內(nèi)存開(kāi)始運(yùn)行時(shí),在Windows的管理下,才在應(yīng)用程序與相應(yīng)的DLL之間建立鏈接關(guān)系。當(dāng)要執(zhí)行所調(diào)用DLL中的函數(shù)時(shí),根據(jù)鏈接產(chǎn)生的重定位信息,Windows才轉(zhuǎn)去執(zhí)行DLL中相應(yīng)的函數(shù)代碼。
LabVIEW 中是通過(guò)Call Library Function Node(CLN)節(jié)點(diǎn)來(lái)完成DLL文件調(diào)用的。創(chuàng)建一個(gè)新的VI,右擊程序框圖,在Functions Palette中依次選中Connectivity——Libraries&Executables工具欄即可找到該節(jié)點(diǎn)(圖1)。
LabVIEW與外部程序間DLL文件的調(diào)用
將節(jié)點(diǎn)放置在程序框圖中,雙擊會(huì)出現(xiàn)它的配置對(duì)話框,共有四頁(yè)。第一頁(yè)用于填寫(xiě)被調(diào)用函數(shù)的信息(圖2)。Library name or path需給出DLL文件名和路徑,操作系統(tǒng)路徑下的DLL文件,直接輸入文件名也可調(diào)用,否則必須輸入全路徑。在這里已經(jīng)給出名字的DLL是被靜態(tài)加載到程序中的,也就是說(shuō)當(dāng)調(diào)用了這個(gè)DLL的VI被裝入內(nèi)存時(shí),DLL同時(shí)被裝入內(nèi)存。LabVIEW也可動(dòng)態(tài)加載DLL,只要勾選上Specify path on diagram的選項(xiàng)即可。選擇了這個(gè)選項(xiàng),在 Library name or path中輸入的內(nèi)容就無(wú)效了,取而代之的是CLN 節(jié)點(diǎn)多出一對(duì)輸入輸出,用于指明所需要使用的DLL的路徑。這樣,當(dāng)VI被打開(kāi)時(shí),DLL不會(huì)被裝入內(nèi)存,只用程序運(yùn)行到需要使用這個(gè)DLL中的函數(shù)時(shí),才把其裝入內(nèi)存。Function name是需要調(diào)用的函數(shù)的名稱(chēng),LabVIEW會(huì)把DLL中所有的暴露出來(lái)的函數(shù)都列出,用戶只要在下拉框中選取即可。Thread欄用于設(shè)定哪個(gè)線程里運(yùn)行被調(diào)用的函數(shù)。用戶可以通過(guò) CLN 節(jié)點(diǎn)的配置面板來(lái)指定被調(diào)用函數(shù)運(yùn)行所在的線程。CLN 的線程選項(xiàng)非常簡(jiǎn)單,只有兩項(xiàng): Run in UI thread和Run in any thread。LabVIEW的程序框圖上直接可以看出一個(gè) CLN節(jié)點(diǎn)是選用
圖2 填寫(xiě)被調(diào)用函數(shù)信息的什么線程。如果Run in UI thread,節(jié)點(diǎn)顏色是橙色的;Run in any thread則是淺黃色的(圖3)。
圖3 CLN不同線程對(duì)比
通常情況下,除非使用的動(dòng)態(tài)鏈接庫(kù)是多線程安全的,CLN 中選擇Run in any thread方式;否則必須選擇Run in UI thread方式。判斷一個(gè)動(dòng)態(tài)鏈接庫(kù)是不是多線程安全的,需通過(guò)以下方法:如果一個(gè)動(dòng)態(tài)鏈接庫(kù)的文檔中沒(méi)有明確說(shuō)明它是多線程安全的,那么就要當(dāng)作是非多線程安全的;在可以看到動(dòng)態(tài)鏈接庫(kù)源代碼的條件下,如果代碼中存在全局變量、靜態(tài)變量或者代碼中看不到有l(wèi)ock一類(lèi)的操作,那么這個(gè)動(dòng)態(tài)鏈接庫(kù)也就肯定不是多線程安全的。
選擇了Run in any thread方式,LabVIEW會(huì)在最方便的線程內(nèi)運(yùn)行動(dòng)態(tài)鏈接庫(kù)函數(shù),且一般會(huì)與調(diào)用它的VI在同一個(gè)線程內(nèi)運(yùn)行。因?yàn)長(zhǎng)abVIEW是自動(dòng)多線程的語(yǔ)言,它也很可能會(huì)把動(dòng)態(tài)鏈接庫(kù)函數(shù)分配給一個(gè)單獨(dú)的線程運(yùn)行。如果程序中存在沒(méi)有直接或間接先后關(guān)系的兩個(gè)CLN節(jié)點(diǎn),LabVIEW很可能會(huì)同時(shí)在不同的線程內(nèi)運(yùn)行它們所調(diào)用的函數(shù),也許是同一函數(shù)。對(duì)于非多線程安全的動(dòng)態(tài)鏈接庫(kù),這是很危險(xiǎn)的操作。很容易引起數(shù)據(jù)混亂,甚至是程序崩潰。
選擇Run in UI thread方式,因?yàn)長(zhǎng)abVIEW只有一個(gè)界面線程,所以如果所有的CLN設(shè)置都是界面線程,那么就可以保證這些CLN調(diào)用的函數(shù)肯定全部都運(yùn)行在同一線程下,肯定不會(huì)被同時(shí)調(diào)用。對(duì)于非多線程安全的動(dòng)態(tài)鏈接庫(kù),這種方式就保證了它的安全。