DLL - คู่มือฉบับย่อ
การลิงก์แบบไดนามิกเป็นกลไกที่เชื่อมโยงแอปพลิเคชันกับไลบรารีในขณะรันไทม์ ไลบรารียังคงอยู่ในไฟล์ของตนเองและจะไม่ถูกคัดลอกลงในไฟล์ปฏิบัติการของแอปพลิเคชัน DLL จะเชื่อมโยงไปยังแอปพลิเคชันเมื่อเรียกใช้แอปพลิเคชันแทนที่จะเป็นเมื่อสร้าง DLL อาจมีลิงก์ไปยัง DLL อื่น ๆ
หลายครั้ง DLL จะอยู่ในไฟล์ที่มีนามสกุลต่างกันเช่น. EXE, .DRV หรือ. DLL
ข้อดีของ DLL
ด้านล่างนี้เป็นข้อดีบางประการของการมีไฟล์ DLL
ใช้ทรัพยากรน้อยลง
ไฟล์ DLL ไม่ได้รับการโหลดลงใน RAM พร้อมกับโปรแกรมหลัก พวกเขาไม่ใช้พื้นที่เว้นแต่จำเป็น เมื่อต้องการไฟล์ DLL ไฟล์จะถูกโหลดและเรียกใช้ ตัวอย่างเช่นตราบใดที่ผู้ใช้ Microsoft Word กำลังแก้ไขเอกสารไฟล์ DLL ของเครื่องพิมพ์ไม่จำเป็นต้องใช้ใน RAM หากผู้ใช้ตัดสินใจที่จะพิมพ์เอกสารแอปพลิเคชัน Word จะทำให้ไฟล์ DLL ของเครื่องพิมพ์ถูกโหลดและเรียกใช้
ส่งเสริมสถาปัตยกรรมแบบแยกส่วน
DLL ช่วยส่งเสริมการพัฒนาโปรแกรมแบบแยกส่วน ช่วยให้คุณพัฒนาโปรแกรมขนาดใหญ่ที่ต้องใช้หลายภาษาหรือโปรแกรมที่ต้องใช้สถาปัตยกรรมโมดูลาร์ ตัวอย่างของโปรแกรมโมดูลาร์คือโปรแกรมบัญชีที่มีโมดูลจำนวนมากที่สามารถโหลดแบบไดนามิกได้ในขณะรันไทม์
ช่วยให้ใช้งานและติดตั้งได้ง่าย
เมื่อฟังก์ชันภายใน DLL ต้องการการอัปเดตหรือการแก้ไขการปรับใช้และการติดตั้ง DLL ไม่จำเป็นต้องเชื่อมโยงโปรแกรมกับ DLL อีกครั้ง นอกจากนี้หากหลายโปรแกรมใช้ DLL เดียวกันโปรแกรมทั้งหมดจะได้รับประโยชน์จากการอัปเดตหรือการแก้ไข ปัญหานี้อาจเกิดขึ้นบ่อยขึ้นเมื่อคุณใช้ DLL ของ บริษัท อื่นที่มีการอัปเดตหรือแก้ไขเป็นประจำ
แอปพลิเคชันและ DLL สามารถเชื่อมโยงไปยัง DLL อื่น ๆ โดยอัตโนมัติหากระบุการเชื่อมโยง DLL ในส่วน IMPORTS ของไฟล์ข้อกำหนดโมดูลเป็นส่วนหนึ่งของคอมไพล์ มิฉะนั้นคุณสามารถโหลดได้อย่างชัดเจนโดยใช้ฟังก์ชัน Windows LoadLibrary
ไฟล์ DLL ที่สำคัญ
COMDLG32.DLL - ควบคุมกล่องโต้ตอบ
GDI32.DLL - มีฟังก์ชันมากมายสำหรับการวาดภาพกราฟิกการแสดงข้อความและการจัดการแบบอักษร
KERNEL32.DLL - มีหลายร้อยฟังก์ชันสำหรับการจัดการหน่วยความจำและกระบวนการต่างๆ
USER32.DLL- มีฟังก์ชั่นส่วนต่อประสานผู้ใช้มากมาย เกี่ยวข้องกับการสร้างหน้าต่างโปรแกรมและปฏิสัมพันธ์ระหว่างกัน
ขั้นแรกเราจะพูดถึงปัญหาและข้อกำหนดที่คุณควรพิจารณาในขณะที่พัฒนา DLL ของคุณเอง
ประเภทของ DLL
เมื่อคุณโหลด DLL ในแอปพลิเคชันวิธีการเชื่อมโยงสองวิธีจะช่วยให้คุณสามารถเรียกใช้ฟังก์ชัน DLL ที่ส่งออกได้ การเชื่อมโยงสองวิธีคือ:
- การเชื่อมโยงแบบไดนามิกในเวลาโหลดและ
- การเชื่อมโยงแบบไดนามิกรันไทม์
การเชื่อมโยงแบบไดนามิกในเวลาโหลด
ในการเชื่อมโยงแบบไดนามิกในเวลาโหลดแอปพลิเคชันทำการเรียกอย่างชัดเจนไปยังฟังก์ชัน DLL ที่ส่งออกเช่นฟังก์ชันภายในเครื่อง ในการใช้การลิงก์แบบไดนามิกในเวลาโหลดให้จัดเตรียมไฟล์ส่วนหัว (.h) และไฟล์อิมพอร์ตไลบรารี (.lib) เมื่อคุณคอมไพล์และลิงก์แอ็พพลิเคชัน เมื่อคุณทำสิ่งนี้ตัวเชื่อมโยงจะให้ข้อมูลที่จำเป็นกับระบบในการโหลด DLL และแก้ไขตำแหน่งฟังก์ชัน DLL ที่ส่งออกในเวลาโหลด
การเชื่อมโยงแบบไดนามิกรันไทม์
ในการลิงก์แบบไดนามิกของรันไทม์แอ็พพลิเคชันเรียกใช้ฟังก์ชัน LoadLibrary หรือฟังก์ชัน LoadLibraryEx เพื่อโหลด DLL ที่รันไทม์ หลังจากโหลด DLL สำเร็จแล้วคุณใช้ฟังก์ชัน GetProcAddress เพื่อรับที่อยู่ของฟังก์ชัน DLL ที่ส่งออกที่คุณต้องการเรียกใช้ เมื่อคุณใช้การลิงก์แบบไดนามิกของรันไทม์คุณไม่จำเป็นต้องมีไฟล์ไลบรารีอิมพอร์ต
รายการต่อไปนี้อธิบายถึงเกณฑ์ของแอปพลิเคชันสำหรับการเลือกระหว่างการลิงก์แบบไดนามิกในเวลาโหลดและการลิงก์ไดนามิกรันไทม์
Startup performance : หากประสิทธิภาพการเริ่มต้นเริ่มต้นของแอปพลิเคชันมีความสำคัญคุณควรใช้การเชื่อมโยงแบบไดนามิกรันไทม์
Ease of use: ในการเชื่อมโยงแบบไดนามิกในเวลาโหลดฟังก์ชัน DLL ที่ส่งออกจะเหมือนกับฟังก์ชันภายในเครื่อง ช่วยให้คุณเรียกใช้ฟังก์ชันเหล่านี้ได้อย่างง่ายดาย
Application logic: ในการเชื่อมโยงแบบไดนามิกรันไทม์แอปพลิเคชันสามารถแตกแขนงเพื่อโหลดโมดูลต่างๆได้ตามต้องการ นี่เป็นสิ่งสำคัญเมื่อคุณพัฒนาเวอร์ชันหลายภาษา
จุดเข้าใช้งาน DLL
เมื่อคุณสร้าง DLL คุณสามารถเลือกระบุฟังก์ชันจุดเข้าได้ ฟังก์ชันจุดเริ่มต้นถูกเรียกใช้เมื่อกระบวนการหรือเธรดแนบตัวเองเข้ากับ DLL หรือแยกตัวเองออกจาก DLL คุณสามารถใช้ฟังก์ชันจุดเข้าเริ่มต้นหรือทำลายโครงสร้างข้อมูลตามที่ DLL ต้องการ
นอกจากนี้หากแอปพลิเคชันเป็นแบบมัลติเธรดคุณสามารถใช้ที่เก็บข้อมูลภายในเธรด (TLS) เพื่อจัดสรรหน่วยความจำที่เป็นส่วนตัวให้กับแต่ละเธรดในฟังก์ชันจุดเริ่มต้น โค้ดต่อไปนี้เป็นตัวอย่างของฟังก์ชันจุดเข้า DLL
BOOL APIENTRY DllMain(
HANDLE hModule, // Handle to DLL module DWORD ul_reason_for_call, LPVOID lpReserved ) // Reserved
{
switch ( ul_reason_for_call )
{
case DLL_PROCESS_ATTACHED:
// A process is loading the DLL.
break;
case DLL_THREAD_ATTACHED:
// A process is creating a new thread.
break;
case DLL_THREAD_DETACH:
// A thread exits normally.
break;
case DLL_PROCESS_DETACH:
// A process unloads the DLL.
break;
}
return TRUE;
}
เมื่อฟังก์ชันจุดเข้าใช้งานส่งคืนค่า FALSE แอปพลิเคชันจะไม่เริ่มทำงานหากคุณใช้การเชื่อมโยงแบบไดนามิกในเวลาโหลด หากคุณใช้การเชื่อมโยงแบบไดนามิกรันไทม์เฉพาะ DLL แต่ละรายการเท่านั้นที่จะไม่โหลด
ฟังก์ชันจุดเข้าใช้งานควรดำเนินการเริ่มต้นอย่างง่ายเท่านั้นและไม่ควรเรียกใช้ฟังก์ชันการโหลดหรือการยกเลิก DLL อื่น ๆ ตัวอย่างเช่นในฟังก์ชันจุดเข้าใช้งานคุณไม่ควรเรียกใช้ไฟล์LoadLibrary ฟังก์ชันหรือ LoadLibraryExฟังก์ชัน นอกจากนี้คุณไม่ควรโทรไปที่ไฟล์FreeLibrary ฟังก์ชันเมื่อกระบวนการกำลังยุติ
WARNING: ในแอพพลิเคชั่นมัลติเธรดตรวจสอบให้แน่ใจว่าการเข้าถึงข้อมูลส่วนกลางของ DLL ถูกซิงโครไนซ์ (เธรดปลอดภัย) เพื่อหลีกเลี่ยงความเสียหายของข้อมูล ในการดำเนินการนี้ให้ใช้ TLS เพื่อให้ข้อมูลเฉพาะสำหรับแต่ละเธรด
การส่งออกฟังก์ชัน DLL
ในการส่งออกฟังก์ชัน DLL คุณสามารถเพิ่มคีย์เวิร์ดของฟังก์ชันให้กับฟังก์ชัน DLL ที่ส่งออกหรือสร้างไฟล์ข้อกำหนดโมดูล (.def) ที่แสดงรายการฟังก์ชัน DLL ที่ส่งออก
ในการใช้คำสำคัญของฟังก์ชันคุณต้องประกาศแต่ละฟังก์ชันที่คุณต้องการส่งออกด้วยคำสำคัญต่อไปนี้:
__declspec(dllexport)
ในการใช้ฟังก์ชัน DLL ที่ส่งออกในแอปพลิเคชันคุณต้องประกาศแต่ละฟังก์ชันที่คุณต้องการนำเข้าด้วยคำสำคัญต่อไปนี้:
__declspec(dllimport)
โดยทั่วไปคุณจะใช้ไฟล์ส่วนหัวหนึ่งไฟล์ที่มีไฟล์ define คำสั่งและ ifdef คำสั่งเพื่อแยกคำสั่งการส่งออกและคำสั่งนำเข้า
คุณยังสามารถใช้ไฟล์ข้อกำหนดโมดูลเพื่อประกาศฟังก์ชัน DLL ที่ส่งออก เมื่อคุณใช้ไฟล์ข้อกำหนดของโมดูลคุณไม่จำเป็นต้องเพิ่มคีย์เวิร์ดของฟังก์ชันลงในฟังก์ชัน DLL ที่ส่งออก ในไฟล์ข้อกำหนดโมดูลคุณประกาศไฟล์LIBRARY คำสั่งและ EXPORTSคำสั่งสำหรับ DLL โค้ดต่อไปนี้เป็นตัวอย่างของไฟล์นิยาม
// SampleDLL.def
//
LIBRARY "sampleDLL"
EXPORTS
HelloWorld
เขียน DLL ตัวอย่าง
ใน Microsoft Visual C ++ 6.0 คุณสามารถสร้าง DLL ได้โดยเลือกไฟล์ Win32 Dynamic-Link Library ประเภทโครงการหรือ MFC AppWizard (dll) ประเภทโครงการ.
รหัสต่อไปนี้เป็นตัวอย่างของ DLL ที่สร้างขึ้นใน Visual C ++ โดยใช้ประเภทโครงการ Win32 Dynamic-Link Library
// SampleDLL.cpp
#include "stdafx.h"
#define EXPORTING_DLL
#include "sampleDLL.h"
BOOL APIENTRY DllMain( HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved )
{
return TRUE;
}
void HelloWorld()
{
MessageBox( NULL, TEXT("Hello World"),
TEXT("In a DLL"), MB_OK);
}
// File: SampleDLL.h
//
#ifndef INDLL_H
#define INDLL_H
#ifdef EXPORTING_DLL
extern __declspec(dllexport) void HelloWorld() ;
#else
extern __declspec(dllimport) void HelloWorld() ;
#endif
#endif
เรียก DLL ตัวอย่าง
รหัสต่อไปนี้เป็นตัวอย่างของโปรเจ็กต์ Win32 Application ที่เรียกใช้ฟังก์ชัน DLL ที่ส่งออกใน SampleDLL DLL
// SampleApp.cpp
#include "stdafx.h"
#include "sampleDLL.h"
int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
HelloWorld();
return 0;
}
NOTE : ในการลิงก์แบบไดนามิกเวลาโหลดคุณต้องลิงก์ไลบรารีอิมพอร์ต SampleDLL.lib ที่สร้างขึ้นเมื่อคุณสร้างโปรเจ็กต์ SampleDLL
ในการเชื่อมโยงแบบไดนามิกรันไทม์คุณใช้รหัสที่คล้ายกับรหัสต่อไปนี้เพื่อเรียกฟังก์ชัน DLL ที่ส่งออกของ SampleDLL.dll
...
typedef VOID (*DLLPROC) (LPTSTR);
...
HINSTANCE hinstDLL;
DLLPROC HelloWorld;
BOOL fFreeDLL;
hinstDLL = LoadLibrary("sampleDLL.dll");
if (hinstDLL != NULL)
{
HelloWorld = (DLLPROC) GetProcAddress(hinstDLL, "HelloWorld");
if (HelloWorld != NULL)
(HelloWorld);
fFreeDLL = FreeLibrary(hinstDLL);
}
...
เมื่อคุณคอมไพล์และเชื่อมโยงแอ็พพลิเคชัน SampleDLL ระบบปฏิบัติการ Windows จะค้นหา SampleDLL DLL ในตำแหน่งต่อไปนี้ตามลำดับนี้:
โฟลเดอร์แอปพลิเคชัน
โฟลเดอร์ปัจจุบัน
โฟลเดอร์ระบบ Windows (ไฟล์ GetSystemDirectory ฟังก์ชันส่งคืนเส้นทางของโฟลเดอร์ระบบ Windows)
โฟลเดอร์ Windows (ไฟล์ GetWindowsDirectory ฟังก์ชันส่งคืนเส้นทางของโฟลเดอร์ Windows)
ในการใช้ DLL จะต้องมีการลงทะเบียนโดยมีการอ้างอิงที่เหมาะสมใน Registry บางครั้งอาจเกิดขึ้นที่การอ้างอิง Registry เสียหายและไม่สามารถใช้ฟังก์ชันของ DLL ได้อีกต่อไป DLL สามารถลงทะเบียนใหม่ได้โดยเปิด Start-Run และป้อนคำสั่งต่อไปนี้:
regsvr32 somefile.dll
คำสั่งนี้ถือว่า somefile.dll อยู่ในไดเร็กทอรีหรือโฟลเดอร์ที่อยู่ใน PATH มิฉะนั้นจะต้องใช้เส้นทางแบบเต็มสำหรับ DLL ไฟล์ DLL สามารถยกเลิกการลงทะเบียนได้โดยใช้สวิตช์ "/ u" ดังที่แสดงด้านล่าง
regsvr32 /u somefile.dll
สามารถใช้เพื่อสลับเปิดและปิดบริการ
มีเครื่องมือมากมายเพื่อช่วยคุณแก้ไขปัญหา DLL บางส่วนของพวกเขาจะกล่าวถึงด้านล่าง
วอล์คเกอร์พึ่งพา
เครื่องมือ Dependency Walker (depends.exe) สามารถสแกนซ้ำสำหรับ DLL ที่ขึ้นอยู่กับที่ใช้โดยโปรแกรม เมื่อคุณเปิดโปรแกรมใน Dependency Walker Dependency Walker จะทำการตรวจสอบดังต่อไปนี้:
- ตรวจสอบ DLL ที่หายไป
- ตรวจสอบไฟล์โปรแกรมหรือ DLL ที่ไม่ถูกต้อง
- ตรวจสอบว่าฟังก์ชันการนำเข้าและฟังก์ชันการส่งออกตรงกัน
- ตรวจสอบข้อผิดพลาดการอ้างอิงแบบวงกลม
- ตรวจสอบโมดูลที่ไม่ถูกต้องเนื่องจากโมดูลมีไว้สำหรับระบบปฏิบัติการอื่น
เมื่อใช้ Dependency Walker คุณสามารถจัดทำเอกสาร DLL ทั้งหมดที่โปรแกรมใช้ อาจช่วยป้องกันและแก้ไขปัญหา DLL ที่อาจเกิดขึ้นในอนาคต Dependency Walker อยู่ในไดเร็กทอรีต่อไปนี้เมื่อคุณติดตั้ง Microsoft Visual Studio 6.0:
drive\Program Files\Microsoft Visual Studio\Common\Tools
DLL Universal Problem Solver
เครื่องมือ DLL Universal Problem Solver (DUPS) ใช้เพื่อตรวจสอบเปรียบเทียบจัดทำเอกสารและแสดงข้อมูล DLL รายการต่อไปนี้อธิบายถึงยูทิลิตี้ที่ประกอบเป็นเครื่องมือ DUPS:
Dlister.exe - ยูทิลิตี้นี้จะระบุ DLL ทั้งหมดในคอมพิวเตอร์และบันทึกข้อมูลลงในไฟล์ข้อความหรือไฟล์ฐานข้อมูล
Dcomp.exe - ยูทิลิตี้นี้เปรียบเทียบ DLL ที่แสดงอยู่ในไฟล์ข้อความสองไฟล์และสร้างไฟล์ข้อความที่สามที่มีความแตกต่าง
Dtxt2DB.exe - ยูทิลิตี้นี้โหลดไฟล์ข้อความที่สร้างขึ้นโดยใช้ยูทิลิตี้ Dlister.exe และยูทิลิตี้ Dcomp.exe ลงในฐานข้อมูล dllHell
DlgDtxt2DB.exe - ยูทิลิตี้นี้มียูทิลิตี้ Dtxt2DB.exe สำหรับผู้ใช้แบบกราฟิก (GUI)
โปรดคำนึงถึงเคล็ดลับต่อไปนี้ในขณะเขียน DLL:
ใช้รูปแบบการโทรที่เหมาะสม (C หรือ stdcall)
ระวังลำดับที่ถูกต้องของอาร์กิวเมนต์ที่ส่งไปยังฟังก์ชัน
อย่าปรับขนาดอาร์เรย์หรือต่อสตริงโดยใช้อาร์กิวเมนต์ที่ส่งผ่านไปยังฟังก์ชันโดยตรง โปรดจำไว้ว่าพารามิเตอร์ที่คุณส่งผ่านคือข้อมูล LabVIEW การเปลี่ยนขนาดอาร์เรย์หรือสตริงอาจทำให้เกิดความผิดพลาดโดยการเขียนทับข้อมูลอื่นที่จัดเก็บไว้ในหน่วยความจำ LabVIEW คุณอาจปรับขนาดอาร์เรย์หรือเชื่อมต่อสตริงได้หากคุณผ่าน LabVIEW Array Handle หรือ LabVIEW String Handle และกำลังใช้คอมไพเลอร์ Visual C ++ หรือคอมไพเลอร์ Symantec เพื่อคอมไพล์ DLL ของคุณ
ในขณะที่ส่งสตริงไปยังฟังก์ชันให้เลือกประเภทของสตริงที่ถูกต้องเพื่อส่งผ่าน ที่จับสตริง C หรือ Pascal หรือ LabVIEW
สตริงปาสคาลมีความยาว 255 อักขระ
สตริง C ถูกยกเลิกเป็น NULL หากฟังก์ชัน DLL ของคุณส่งคืนข้อมูลตัวเลขในรูปแบบสตริงไบนารี (ตัวอย่างเช่นผ่าน GPIB หรือพอร์ตอนุกรม) อาจส่งคืนค่า NULL เป็นส่วนหนึ่งของสตริงข้อมูล ในกรณีเช่นนี้การส่งผ่านอาร์เรย์ของจำนวนเต็มสั้น (8 บิต) มีความน่าเชื่อถือมากที่สุด
หากคุณกำลังทำงานกับอาร์เรย์หรือสตริงของข้อมูลมักจะส่งบัฟเฟอร์หรืออาร์เรย์ที่มีขนาดใหญ่พอที่จะเก็บผลลัพธ์ใด ๆ ที่วางไว้ในบัฟเฟอร์โดยฟังก์ชันเว้นแต่คุณจะส่งผ่านเป็นที่จับ LabVIEW ซึ่งในกรณีนี้คุณสามารถปรับขนาดได้โดยใช้ CIN ฟังก์ชันภายใต้คอมไพเลอร์ Visual C ++ หรือ Symantec
แสดงรายการฟังก์ชัน DLL ในส่วน EXPORTS ของไฟล์นิยามโมดูลหากคุณใช้ _stdcall
แสดงรายการฟังก์ชัน DLL ที่แอปพลิเคชันอื่นเรียกใช้ในส่วนไฟล์นิยามโมดูล EXPORTS หรือรวมคีย์เวิร์ด _declspec (dllexport) ในการประกาศฟังก์ชัน
หากคุณใช้คอมไพเลอร์ C ++ ให้เอ็กซ์พอร์ตฟังก์ชันด้วยคำสั่ง extern .C. {} ในไฟล์ส่วนหัวของคุณเพื่อป้องกันไม่ให้ชื่อยุ่งเหยิง
หากคุณกำลังเขียน DLL ของคุณเองคุณไม่ควรคอมไพล์ DLL ซ้ำในขณะที่แอพพลิเคชั่นอื่นโหลด DLL ลงในหน่วยความจำ ก่อนที่จะคอมไพล์ DLL ใหม่ตรวจสอบให้แน่ใจว่าแอปพลิเคชันทั้งหมดที่ใช้ DLL นั้นถูกยกเลิกการโหลดจากหน่วยความจำ เพื่อให้แน่ใจว่าไม่ได้โหลด DLL ลงในหน่วยความจำ คุณอาจล้มเหลวในการสร้างใหม่อย่างถูกต้องหากคุณลืมสิ่งนี้และคอมไพเลอร์ของคุณไม่เตือนคุณ
ทดสอบ DLL ของคุณด้วยโปรแกรมอื่นเพื่อให้แน่ใจว่าฟังก์ชัน (และ DLL) ทำงานได้อย่างถูกต้อง การทดสอบด้วยดีบักเกอร์ของคอมไพเลอร์ของคุณหรือโปรแกรม C แบบง่ายที่คุณสามารถเรียกใช้ฟังก์ชันใน DLL จะช่วยให้คุณระบุได้ว่าปัญหาที่อาจเกิดขึ้นนั้นเกี่ยวข้องกับ DLL หรือ LabVIEW หรือไม่
เราได้เห็นวิธีการเขียน DLL และวิธีสร้างโปรแกรม "Hello World" ตัวอย่างนั้นต้องให้แนวคิดเกี่ยวกับแนวคิดพื้นฐานในการสร้าง DLL
ในที่นี้เราจะให้คำอธิบายเกี่ยวกับการสร้าง DLL โดยใช้ Delphi, Borland C ++ และอีกครั้ง VC ++
ให้เรานำตัวอย่างเหล่านี้ทีละตัวอย่าง
วิธีเขียนและเรียก DLL ภายใน Delphi
การสร้าง DLL จาก Borland C ++ Builder IDE
การสร้าง DLL ใน Microsoft Visual C ++ 6.0