當前位置:首頁 » 編程語言 » c語言動態庫怎麼創建
擴展閱讀
webinf下怎麼引入js 2023-08-31 21:54:13
堡壘機怎麼打開web 2023-08-31 21:54:11

c語言動態庫怎麼創建

發布時間: 2022-05-26 19:52:49

『壹』 關於c語言中如何創建自己的庫函數能不能再詳細的解釋一下,謝謝了!!!

哈哈,選我吧!庫分靜態庫和動態鏈接庫,靜態庫以lib結尾,被編譯器里的鏈接器識別。windows下動態庫以dll結尾,被操作系統載入以模塊方式映射到進程地址空間。一般初學者先學會創建的是靜態庫。靜態庫是一個無需重定位的函數集。怎麼做到無需重定位呢?這是編譯器做的編譯工作,例如它指定開頭的位置作為基址,剩下的代碼用到的都是相對偏移。這樣,這段二進制代碼可以被放在內存中的任何位置執行,被寫入了lib文件里。在lib文件里,包含了函數名與函數地址組成的結構體,通過它編譯器可以找到lib文件里需要的二進制代碼並以靜態聯編的方式寫入我們調用它的exe文件里。這種代碼是被塞進exe文件里而無需修改,並在程序執行時被用到。為了讓庫被別人調用,我們可以寫一個頭文件.h,包含函數原型及聲明。

『貳』 怎麼將多文件C程序創建為動態鏈接庫

如果用的ide,那麼工程選項設置為生成dll就行
如果用的命令行
那麼逐個把.c編譯成.obj之後
再用link連接成為dll
或者直接編譯成dll
比如(vs)
cl 1.c 2.c 3.c 4.c /LD

『叄』 C或C++語言如何創建適用所有編程語言的動態庫

不能調用的原因一般是在於聲明,windows里的api就是用C語言寫函數庫,用C++寫類庫,除了dll文件之外,不同編程語言在使用這個庫的時候需要先用那個語言生成一個類似於聲明的文件.
你平時編程時使用windowsapi可能覺得是直接用的,其實不是,工具鏈里其實提供了相應聲明,只不過那寫聲明只包含windowsapi的聲明而不包含你的庫的聲明,需要你自己去布置你寫的庫的聲明

『肆』 如何建立自己的C語言函數庫

想想你自己定義的函數,如過你定義了一些函數,就可以看成你製作了一個函數庫,只不過你的函數都在程序上,所以不需要包含頭文件,如過把你自己定義的函數做成.h文件,就可以在主程序中直接調用

『伍』 c語言中怎樣建立自己的lib庫呀請教

你編譯完DLL之後能到編譯好的Dll目錄中找到對應的lib文件。

『陸』 C語言怎麼使用動態鏈接庫,如何創建(高手進)

C程序編譯成dll文件只不過是在要公開的介面函數聲明前面加上幾個特定的修飾符而已。用dev-cpp建了個dll的默認文檔,一切都很明了。(我把源代碼貼在下面)

/*dll.h文件*/

#ifndef _DLL_H_
#define _DLL_H_

#if BUILDING_DLL
# define DLLIMPORT __declspec (dllexport)
#else /* Not BUILDING_DLL */
# define DLLIMPORT __declspec (dllimport)
#endif /* Not BUILDING_DLL */

DLLIMPORT void HelloWorld (void);

#endif /* _DLL_H_ */

/*dllmain.c文件*/

/* Replace "dll.h" with the name of your header */
#include "dll.h"
#include <windows.h>
#include <stdio.h>
#include <stdlib.h>

DLLIMPORT void HelloWorld ()
{
MessageBox (0, "Hello World from DLL!\n", "Hi", MB_ICONINFORMATION);
}

BOOL APIENTRY DllMain (HINSTANCE hInst /* Library instance handle. */ ,
DWORD reason /* Reason this function is being called. */ ,
LPVOID reserved /* Not used. */ )
{
switch (reason)
{
case DLL_PROCESS_ATTACH:
break;

case DLL_PROCESS_DETACH:
break;

case DLL_THREAD_ATTACH:
break;

case DLL_THREAD_DETACH:
break;
}

/* Returns TRUE on success, FALSE on failure */
return TRUE;
}

關於以上代碼的幾點解釋:
一、__declspec (dllexport):這是關鍵,它標志著這個這個函數將成為對外的介面。(以下是我在網上下載的dllexport、dllimport、_declspec的一些說明):
使用包含在DLL的函數,必須將其導入。導入操作時通過dllimport來完成的,dllexport和dllimport都是vc(visual C++)和bc(Borland C++)所支持的擴展的關鍵字。但是dllexport和dllimport關鍵字不能被自身所使用,因此它的前面必須有另一個擴展關鍵字__declspec。通用格式如下:__declspec(specifier)其中specifier是存儲類標示符。對於DLL,specifier將是dllexport和dllimport。而且為了簡化說明導入和導出函數的語句,用一個宏名來代替__declspec.在此程序中,使用的是DllExport。如果用戶的DLL被編譯成一個C++程序,而且希望C程序也能使用它,就需要增加「C」的連接說明。#define DllExport extern "C"__declspec(dllexport),這樣就避免了標准C++命名損壞。(當然,如果讀者正在編譯的是C程序,就不要加入extern 「C」,因為不需要它,而且編譯器也不接受它)。

二、BOOL APIENTRY DllMain ()說明:(以下是我在網上收集的資料)

1、每一個DLL必須有一個入口點,DllMain是一個預設的入口函數。DllMain負責初始化(Initialization)和結束(Termination)工作,每當一個新的進程或者該進程的新的線程訪問DLL時,或者訪問DLL的每一個進程或者線程不再使用DLL或者結束時,都會調用DllMain。但是,使用TerminateProcess或TerminateThread結束進程或者線程,不會調用DllMain。
DllMain的函數原型:
BOOL APIENTRY DllMain(HANDLE hMole,DWORD ul_reason_for_call,LPVOID lpReserved)
{
switch(ul_reason_for_call)
{
case DLL_PROCESS_ATTACH:
.......
case DLL_THREAD_ATTACH:
.......
case DLL_THREAD_DETACH:
.......
case DLL_PROCESS_DETACH:
.......
return TRUE;
}
}

參數:
hMoudle:是動態庫被調用時所傳遞來的一個指向自己的句柄(實際上,它是指向_DGROUP段的一個選擇符);
ul_reason_for_call:是一個說明動態庫被調原因的標志。當進程或線程裝入或卸載動態連接庫的時候,操作系統調用入口函數,並說明動態連接庫被調用的原因。它所有的可能值為:
DLL_PROCESS_ATTACH: 進程被調用;
DLL_THREAD_ATTACH: 線程被調用;
DLL_PROCESS_DETACH: 進程被停止;
DLL_THREAD_DETACH: 線程被停止;
lpReserved:是一個被系統所保留的參數。

看到這里,我想大家應該會對將c程序編譯成dll文件有了個大體的概念。

關於對於dll文件的使用,我在vb.net里做了以下測試:
首先用vs.net 2003新建一個vb.net應用程序。
然後在工程屬性中引用System.Runtime.InteropServices命名空間。
然後在默認的窗體文件中添加如下代碼:
Public Class Form1
Inherits System.Windows.Forms.Form

#Region " Windows 窗體設計器生成的代碼 "

Private Sub Form1_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles MyBase.Load

End Sub

Private Sub Button1_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Button1.Click
Hello()
End Sub
End Class

Mole test
Sub main()
Dim frm As New Form1
Application.Run(frm)

End Sub
<DllImport("test.dll", EntryPoint:="HelloWorld", setlasterror:=True)> Public Sub Hello()

End Sub
End Mole

然後把上面用devcpp生成的test.dll放入工程bin目錄下,測試成功。

關於dll文件的一點設想:
關於多語言創建dll文件和動態使用dll文件,我感覺應該是插件技術plugin技術最直接的實現方式。特別是現在的.net平台,為動態導入dll文件中的函數提供了更簡易的方法。一個實現插件的基本思想可以是,在主程序和插件程序內做出一個規定的通訊方式,比如將一個可以代表使用插件功能的對象,由主程序創建對應插件程序的對象,然後由插件程序傳址調用,調用修改後的對象中保存了插件功能信息(比如插件名稱、功能函數指針等),然後再由主程序進行處理。

以下是網上摘抄的一點資料:

動態鏈接庫中定義有兩種函數:導出函數(export function)和內部函數(internal function)。導出函數可以被其它模塊調用,內部函數在定義它們的DLL程序內部使用。

輸出函數的方法有以下幾種:

1、傳統的方法

在模塊定義文件的EXPORT部分指定要輸入的函數或者變數。語法格式如下:
entryname[=internalname] [@ordinal[NONAME]] [DATA] [PRIVATE]

其中:

entryname是輸出的函數或者數據被引用的名稱;

internalname同entryname;

@ordinal表示在輸出表中的順序號(index);

NONAME僅僅在按順序號輸出時被使用(不使用entryname);

DATA表示輸出的是數據項,使用DLL輸出數據的程序必須聲明該數據項為_declspec(dllimport)。

上述各項中,只有entryname項是必須的,其他可以省略。

對於「C」函數來說,entryname可以等同於函數名;但是對「C++」函數(成員函數、非成員函數)來說,entryname是修飾名。可以從.map映像文件中得到要輸出函數的修飾名,或者使用DUMPBIN /SYMBOLS得到,然後把它們寫在.def文件的輸出模塊。DUMPBIN是VC提供的一個工具。

如果要輸出一個「C++」類,則把要輸出的數據和成員的修飾名都寫入.def模塊定義文件。

2、在命令行輸出

對鏈接程序LINK指定/EXPORT命令行參數,輸出有關函數。

3、使用MFC提供的修飾符號_declspec(dllexport)

在要輸出的函數、類、數據的聲明前加上_declspec(dllexport)的修飾符,表示輸出。__declspec(dllexport)在C調用約定、C編譯情況下可以去掉輸出函數名的下劃線前綴。extern "C"使得在C++中使用C編譯方式成為可能。在「C++」下定義「C」函數,需要加extern 「C」關鍵詞。用extern "C"來指明該函數使用C編譯方式。輸出的「C」函數可以從「C」代碼里調用。

例如,在一個C++文件中,有如下函數:
extern "C" {void __declspec(dllexport) __cdecl Test(int var);}
其輸出函數名為:Test

MFC提供了一些宏,就有這樣的作用。

AFX_CLASS_IMPORT:__declspec(dllexport)

AFX_API_IMPORT:__declspec(dllexport)

AFX_DATA_IMPORT:__declspec(dllexport)

AFX_CLASS_EXPORT:__declspec(dllexport)

AFX_API_EXPORT:__declspec(dllexport)

AFX_DATA_EXPORT:__declspec(dllexport)

AFX_EXT_CLASS: #ifdef _AFXEXT
AFX_CLASS_EXPORT
#else
AFX_CLASS_IMPORT

AFX_EXT_API:#ifdef _AFXEXT
AFX_API_EXPORT
#else
AFX_API_IMPORT

AFX_EXT_DATA:#ifdef _AFXEXT
AFX_DATA_EXPORT
#else
AFX_DATA_IMPORT

像AFX_EXT_CLASS這樣的宏,如果用於DLL應用程序的實現中,則表示輸出(因為_AFX_EXT被定義,通常是在編譯器的標識參數中指定該選項/D_AFX_EXT);如果用於使用DLL的應用程序中,則表示輸入(_AFX_EXT沒有定義)。

要輸出整個的類,對類使用_declspec(_dllexpot);要輸出類的成員函數,則對該函數使用_declspec(_dllexport)。如:

class AFX_EXT_CLASS CTextDoc : public CDocument
{

}

extern "C" AFX_EXT_API void WINAPI InitMYDLL();

這幾種方法中,最好採用第三種,方便好用;其次是第一種,如果按順序號輸出,調用效率會高些;最次是第二種。

『柒』 c語言怎麼編寫動態鏈接庫

編寫動態鏈接庫的過程
其實和編寫普通C程序差不多少
只不過
把一個整體的C程序,摘出來一部分, 放在獨立的一個或者幾個C文件中
再把這些C文件打包成dll或者so文件而已。

所以寫法上沒有特別的
需要注意的是編譯。

如果是windows,最好用IDE,創建的時候選動態鏈接庫
如果是gcc, 那麼編譯的時候 增加編譯選項 -fPIC -shared即可。

『捌』 在MATLAB中用C語言編寫S函數時,怎麼生成動態鏈接庫(DLL)文件

在編寫好S函數的實現代碼*.c文件後,還需要在Matlab的命令輸出窗口中進行命令行輸入,完成對源代碼的編譯,這樣才能被S函數模塊所調用。在命令窗口中輸入命令:
「mex

-g

RandomTimeDelay.c」
M文件S-函數在MATLAB環境下可以通過解釋器直接執行,而C文件或其它語言編寫的C-函數,則需要先編譯成可以在MATLAB內運行的二進制代碼:動態連接庫或靜態連接庫,然後才可以使用,這些經過編譯的二進制文件就稱作MEX文件。用MEX命令來對*.c文件進行編譯,然後會在相應的目錄下生成對應的*.mexw32文件。然後就可以在simulink模塊中引用這個s函數了。

『玖』 如何建立動態鏈接庫

動態連接庫的創建步驟:

一、創建Non-MFC DLL動態鏈接庫

1、打開File —> New —> Project選項,選擇Win32 Dynamic-Link Library —>sample project

—>工程名:DllDemo

2、新建一個.h文件DllDemo.h

以下是引用片段:
#ifdefDllDemo_EXPORTS
#defineDllAPI__declspec(dllexport)
#else
#defineDllAPI__declspec(dllimport)
extern"C"//原樣編譯
{
DllAPIint__stdcallMax(inta,intb);//__stdcall使非C/C++語言內能夠調用API
}
#endif

3、在DllDemo.cpp文件中導入DllDemo.h文件,並實現Max(int,int)函數

以下是引用片段:
#include"DllDemo.h"
DllAPIint__stdcallMax(inta,intb)
{
if(a==b)
returnNULL;
elseif(a>b)
returna;
else
returnb;
}

4、編譯程序生成動態連接庫

二、用.def文件創建動態連接庫DllDemo.dll。

1、刪除DllDemo工程中的DllDemo.h文件。

2、在DllDemo.cpp文件頭,刪除 #include DllDemo.h語句。

3、向該工程中加入一個文本文件,命名為DllDemo.def並寫入如下語句:

LIBRARY MyDll

EXPORTS

Max@1

4、編譯程序生成動態連接庫。

動態鏈接的調用步驟:

一、隱式調用

1、 建立DllCnslTest工程

2、 將文件DllDemo.dll、DllDemo.lib拷貝到DllCnslTest工程所在的目錄

3、 在DllCnslTest.h中添加如下語句:

以下是引用片段:
#defineDllAPI__declspec(dllimport)
#pragmacomment(lib,"DllDemo.lib")//在編輯器link時,鏈接到DllDemo.lib文件
extern"C"
{
DllAPIint__stdcallMax(inta,intb);
}

4、在DllCnslTest.cpp文件中添加如下語句:

以下是引用片段:
#include"DllCnslTest.h"//或者#include"DllDemo.h"
voidmain()
{
intvalue;
value=Max(2,9);
printf("TheMaxvalueis%d
",value);
}

5、編譯並生成應用程序DllCnslTest.exe

二、顯式調用

1、 建立DllWinTest工程

2、 將文件DllDemo.dll拷貝到DllWinTest工程所在的目錄或Windows系統目錄下。

3、 用vc/bin下的Dumpbin.exe的小程序,查看DLL文件(DllDemo.dll)中的函數結構。

4、 使用類型定義關鍵字typedef,定義指向和DLL中相同的函數原型指針。

例:

以下是引用片段:
typedefint(*lpMax)(inta,intb);//此語句可以放在.h文件中

5、 通過LoadLibray()將DLL載入到當前的應用程序中並返回當前DLL文件的句柄。

例:

以下是引用片段:
HINSTANCEhDll;//聲明一個Dll實例文件句柄
hDll=LoadLibrary("DllDemo.dll");//導入DllDemo.dll動態連接庫

6、 通過GetProcAddress()函數獲取導入到應用程序中的函數指針。

例:

以下是引用片段:
lpMaxMax;
Max=(lpMax)GetProcAddress(hDLL,"Max");
intvalue;
value=Max(2,9);
printf("TheMaxvalueis%d",value);

7、 函數調用完畢後,使用FreeLibrary()卸載DLL文件。

FreeLibrary(hDll);

8、 編譯並生成應用程序DllWinTest.exe

註:顯式鏈接應用程序編譯時不需要使用相應的Lib文件。

建議購買譚浩強教授編輯的C語言編輯叢書進行研究。

『拾』 C語言vs怎麼使用自己做的靜態庫與動態庫,本人小白,請求詳解

1.靜態鏈接庫

打開VS2010,新建一個項目,選擇win32項目,點擊確定,選擇靜態庫這個選項,預編譯頭文件可選可不選。

在這個空項目中,添加一個.h文件和一個.cpp文件。名字我們起為static.h和static.cpp

static.h文件:

[cpp]view plain

  • #ifndefLIB_H

  • #defineLIB_H

  • extern"C"intsum(inta,intb);

  • #endif


  • static.cpp文件:
  • [cpp]view plain

  • #include"static.h"

  • intsum(inta,intb)

  • {

  • returna+b;

  • }

  • 編譯這個項目之後,會在debug文件夾下生成static.lib文件,這個就是我們需要的靜態鏈接庫。
  • 下面說明如何調用靜態鏈接庫。

    首先需要新建一個空項目,起名為test。將之前static項目下的static.h和static.lib這個2個文件復制到test項目的目錄下,並在工程中加入static.h文件。

    新建一個test.cpp文件如下:

    [cpp]view plain

  • #include<stdio.h>

  • #include<stdlib.h>

  • #include"static.h"

  • #pragmacomment(lib,"static.lib")

  • intmain()

  • {

  • printf("%d ",sum(1,2));

  • system("pause");

  • return0;

  • }

  • 編譯運行可得結果:3
  • #pragma comment(lib,"static.lib"),這一句是顯式的導入靜態鏈接庫。除此之外,還有其他的方法,比如通過設置路徑等等,這里不做介紹。

    2.動態鏈接庫

    和創建靜態鏈接庫一樣,需要創建一個空的win32項目,選擇dll選項。創建dynamic.cpp和dynamic.h文件

    dynamic.h文件:

    [cpp]view plain

  • #ifndefDYNAMIC

  • #defineDYNAMIC

  • extern"C"__declspec(dllexport)intsum(inta,intb);

  • #endifDYNAMIC


  • dynamic.cpp文件:

    [cpp]view plain

  • #include"dynamic.h"

  • intsum(inta,intb)

  • {

  • returna+b;

  • }

  • 編譯這個項目,會在debug文件夾下生成dynamic.dll文件。

  • 下面介紹如何調用動態鏈接庫,這里講的是顯式的調用。

    在剛才的test項目下,把static.lib和static.h文件刪除,把dynamic.h和dynamic.dll復制到該目錄下,並在項目中添加dynamic.h文件,修改test.cpp文件為:

    [cpp]view plain

  • #include<stdio.h>

  • #include<stdlib.h>

  • #include<Windows.h>

  • #include"dynamic.h"

  • intmain()

  • {

  • HINSTANCEhDll=NULL;

  • typedefint(*PSUM)(inta,intb);

  • PSUMpSum;

  • hDll=LoadLibrary(L"dynamic.dll");

  • pSum=(PSUM)GetProcAddress(hDll,"sum");

  • printf("%d ",pSum(1,2));

  • system("pause");

  • FreeLibrary(hDll);

  • return0;

  • }


  • 編譯運行結果為:3
  • 特別提示:

    1.extern "C"中的C是大寫,不是小寫

    2.如果從VS2010中直接運行程序,lib和dll需要放到test項目的目錄下;如果想雙擊項目test下的debug文件中的exe文件直接運行的話,需把lib和dll放入debug文件夾下。