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)