DLL - कैसे लिखें

सबसे पहले, हम उन मुद्दों और आवश्यकताओं पर चर्चा करेंगे, जिन्हें आपको अपने DLL को विकसित करते समय विचार करना चाहिए।

DLL के प्रकार

जब आप किसी एप्लिकेशन में DLL लोड करते हैं, तो लिंक करने के दो तरीके आपको निर्यात किए गए DLL फ़ंक्शन कहते हैं। लिंक करने के दो तरीके हैं -

  • लोड-टाइम डायनेमिक लिंकिंग, और
  • रन-टाइम डायनेमिक लिंकिंग।

लोड-टाइम डायनेमिक लिंकिंग

लोड-टाइम डायनेमिक लिंकिंग में, एक एप्लिकेशन स्थानीय कार्यों जैसे निर्यात किए गए DLL कार्यों के लिए स्पष्ट कॉल करता है। लोड-टाइम डायनेमिक लिंकिंग का उपयोग करने के लिए, एक हेडर (.h) फाइल और एक इंपोर्ट लाइब्रेरी (.lib) फाइल प्रदान करें, जब आप एप्लिकेशन को कंपाइल और लिंक करते हैं। जब आप ऐसा करते हैं, तो लिंकर डीएलएल को लोड करने के लिए आवश्यक जानकारी प्रदान करता है और लोड समय पर निर्यात किए गए डीएलएल फ़ंक्शन स्थानों को हल करने के लिए सिस्टम प्रदान करेगा।

रनटाइम गतिशील लिंकिंग

रनटाइम डायनेमिक लिंकिंग में, एक एप्लिकेशन रनटाइम पर DLL लोड करने के लिए या तो LoadLibrary फ़ंक्शन या LoadLibraryEx फ़ंक्शन को कॉल करता है। DLL सफलतापूर्वक लोड होने के बाद, आप निर्यात किए गए DLL फ़ंक्शन का पता प्राप्त करने के लिए, GetProcAddress फ़ंक्शन का उपयोग करते हैं, जिसे आप कॉल करना चाहते हैं। जब आप रनटाइम डायनेमिक लिंकिंग का उपयोग करते हैं, तो आपको एक आयात लाइब्रेरी फ़ाइल की आवश्यकता नहीं होती है।

निम्न सूची लोड-टाइम डायनेमिक लिंकिंग और रनटाइम डायनेमिक लिंकिंग के बीच चयन के लिए आवेदन मानदंड का वर्णन करती है -

  • 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 वैश्विक डेटा तक पहुँच सिंक्रनाइज़ (थ्रेड सुरक्षित) है। ऐसा करने के लिए, प्रत्येक थ्रेड के लिए अद्वितीय डेटा प्रदान करने के लिए टीएलएस का उपयोग करें।

निर्यात DLL कार्य

DLL फ़ंक्शन को निर्यात करने के लिए, आप निर्यात DLL फ़ंक्शन में फ़ंक्शन कीवर्ड जोड़ सकते हैं या निर्यात DLL फ़ंक्शन को सूचीबद्ध करने वाली एक मॉड्यूल परिभाषा (.def) फ़ाइल बना सकते हैं।

फ़ंक्शन कीवर्ड का उपयोग करने के लिए, आपको प्रत्येक फ़ंक्शन को घोषित करना होगा जिसे आप निम्नलिखित कीवर्ड के साथ निर्यात करना चाहते हैं -

__declspec(dllexport)

निर्यात किए गए DLL फ़ंक्शन का उपयोग करने के लिए, आपको प्रत्येक फ़ंक्शन को घोषित करना होगा जिसे आप निम्नलिखित कीवर्ड के साथ आयात करना चाहते हैं -

__declspec(dllimport)

आमतौर पर, आप एक हेडर फ़ाइल का उपयोग करेंगे define कथन और ifdef निर्यात विवरण और आयात विवरण को अलग करने के लिए बयान।

आप निर्यात DLL फ़ंक्शन की घोषणा करने के लिए एक मॉड्यूल परिभाषा फ़ाइल का उपयोग कर सकते हैं। जब आप मॉड्यूल परिभाषा फ़ाइल का उपयोग करते हैं, तो आपको फ़ंक्शन कीवर्ड को निर्यात किए गए DLL फ़ंक्शन में जोड़ना नहीं पड़ता है। मॉड्यूल परिभाषा फ़ाइल में, आप की घोषणा करते हैंLIBRARY बयान और EXPORTSडीएलएल के लिए बयान। निम्न कोड एक परिभाषा फ़ाइल का एक उदाहरण है।

// SampleDLL.def
//
LIBRARY "sampleDLL"

EXPORTS
   HelloWorld

एक नमूना DLL लिखें

Microsoft Visual C ++ 6.0 में, आप किसी एक को चुनकर DLL बना सकते हैं Win32 Dynamic-Link Library परियोजना का प्रकार या MFC AppWizard (dll) प्रोजेक्ट का प्रकार।

निम्न कोड DLL का एक उदाहरण है जो Win32 डायनामिक-लिंक लाइब्रेरी प्रोजेक्ट प्रकार का उपयोग करके Visual C ++ में बनाया गया था।

// 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 अनुप्रयोग प्रोजेक्ट का एक उदाहरण है जो ExportD 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 - लोड-टाइम डायनेमिक लिंकिंग में, आपको नमूना डीएलएल.लिब आयात लाइब्रेरी को लिंक करना होगा जो कि आप नमूनाडेल प्रोजेक्ट बनाते समय बनाई गई है।

रनटाइम डायनेमिक लिंकिंग में, आप उस कोड का उपयोग करते हैं, जो नमूना कोड 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);
}
...

जब आप सैंपलडीएलएल एप्लिकेशन को संकलित और लिंक करते हैं, तो विंडोज ऑपरेटिंग सिस्टम इस क्रम में निम्नलिखित स्थानों में सैंपलएलएल डीएलएल के लिए खोज करता है -

  • एप्लिकेशन फ़ोल्डर

  • वर्तमान फ़ोल्डर

  • Windows सिस्टम फ़ोल्डर () GetSystemDirectory फ़ंक्शन विंडोज सिस्टम फ़ोल्डर का पथ देता है)।

  • Windows फ़ोल्डर () GetWindowsDirectory फ़ंक्शन विंडोज फ़ोल्डर का पथ देता है)।