函數(shù)參數(shù)傳遞機(jī)制問題在本質(zhì)上是調(diào)用函數(shù)(過程)和被調(diào)用函數(shù)(過程)在調(diào)用發(fā)生時(shí)進(jìn)行通信的方法問題。基本的參數(shù)傳遞機(jī)制有兩
創(chuàng)新互聯(lián)為企業(yè)級(jí)客戶提高一站式互聯(lián)網(wǎng)+設(shè)計(jì)服務(wù),主要包括成都網(wǎng)站建設(shè)、成都網(wǎng)站制作、成都App定制開發(fā)、成都微信小程序、宣傳片制作、LOGO設(shè)計(jì)等,幫助客戶快速提升營(yíng)銷能力和企業(yè)形象,創(chuàng)新互聯(lián)各部門都有經(jīng)驗(yàn)豐富的經(jīng)驗(yàn),可以確保每一個(gè)作品的質(zhì)量和創(chuàng)作周期,同時(shí)每年都有很多新員工加入,為我們帶來大量新的創(chuàng)意。
種:值傳遞和引用傳遞。
推薦:Python教程
值傳遞(passl-by-value)過程中,被調(diào)函數(shù)的形式參數(shù)作為被調(diào)函數(shù)的局部變量處理,即在堆棧中開辟了內(nèi)存空間以存放由主調(diào)函數(shù)放
進(jìn)來的實(shí)參的值,從而成為了實(shí)參的一個(gè)副本。值傳遞的特點(diǎn)是被調(diào)函數(shù)對(duì)形式參數(shù)的任何操作都是作為局部變量進(jìn)行,不會(huì)影響主調(diào)函
數(shù)的實(shí)參變量的值。
引用傳遞(pass-by-reference)過程中,被調(diào)函數(shù)的形式參數(shù)雖然也作為局部變量在堆棧中開辟了內(nèi)存空間,但是這時(shí)存放的是由主調(diào)函
數(shù)放進(jìn)來的實(shí)參變量的地址。被調(diào)函數(shù)對(duì)形參的任何操作都被處理成間接尋址,即通過堆棧中存放的地址訪問主調(diào)函數(shù)中的實(shí)參變量。正
因?yàn)槿绱耍徽{(diào)函數(shù)對(duì)形參做的任何操作都影響了主調(diào)函數(shù)中的實(shí)參變量。
更多技術(shù)請(qǐng)關(guān)注Python視頻教程。
def a(): global q q=1+2 return q def b(): a() c=2+q print(c) b()5 q3 函數(shù)中的變量在外面調(diào)用,需要申明為全局變量
Python 的函數(shù)傳遞參數(shù):
Python 傳參數(shù)可以理解為 C 的 const 指針(your_type* const your_variable),它所指向的對(duì)象可以被修改產(chǎn)生副作用,但變量本身不能修改指向其他對(duì)象。這個(gè)和 C++ 的 reference 差不多。
所以如果一定要產(chǎn)生 C 的修改指針指向其他對(duì)象的效果,用 list、dict 或其他自定義的 mutable 對(duì)象包裝是一個(gè)辦法,但我認(rèn)為這樣是一種不良實(shí)踐。在 C 語言中用參數(shù)輸出結(jié)果有非常多的理由:
C 語言沒有 tuple,不能返回多值,除非聲明一個(gè) struct 類型。這種情況下劃分 in 參數(shù)和 out 參數(shù)成為一種慣例
C 語言沒有異常機(jī)制,返回值一般要保留給 errno
但這些情況在 Python 中都是不存在的
第一步,我先從簡(jiǎn)單的調(diào)用出發(fā),定義了一個(gè)簡(jiǎn)單的函數(shù),該函數(shù)僅僅實(shí)現(xiàn)一個(gè)整數(shù)加法求和:
LIBEXPORT_API int mySum(int a,int b){ return a+b;}
C# 導(dǎo)入定義:
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)]
public static extern int mySum (int a,int b);
}
在C#中調(diào)用測(cè)試:
int iSum = RefComm.mySum(,);
運(yùn)行查看結(jié)果iSum為5,調(diào)用正確。第一步試驗(yàn)完成,說明在C#中能夠調(diào)用自定義的動(dòng)態(tài)鏈接庫(kù)函數(shù)。
第二步,我定義了字符串操作的函數(shù)(簡(jiǎn)單起見,還是采用前面的函數(shù)名),返回結(jié)果為字符串:
LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,"%s",a); return a;}
C# 導(dǎo)入定義:
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Auto,
CallingConvention=CallingConvention.StdCall)]
public static extern string mySum (string a, string b);
}
在C#中調(diào)用測(cè)試:
string strDest="";
string strTmp= RefComm.mySum("45", strDest);
運(yùn)行查看結(jié)果 strTmp 為"45",但是strDest為空。我修改動(dòng)態(tài)鏈接庫(kù)實(shí)現(xiàn),返回結(jié)果為串b:
LIBEXPORT_API char *mySum(char *a,char *b){sprintf(b,"%s",a) return b;}
修改 C# 導(dǎo)入定義,將串b修改為ref方式:
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Auto,CallingConvention=CallingConvention.StdCall)]
public static extern string mySum (string a, ref string b);
}
在C#中再調(diào)用測(cè)試:
string strDest="";
string strTmp= RefComm.mySum("45", ref strDest);
運(yùn)行查看結(jié)果 strTmp 和 strDest 均不對(duì),含不可見字符。再修改 C# 導(dǎo)入定義,將CharSet從Auto修改為Ansi:
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
public static extern string mySum (string a, string b);
}
在C#中再調(diào)用測(cè)試:
string strDest="";
string strTmp= RefComm. mySum("45", ref strDest);
運(yùn)行查看結(jié)果 strTmp 為"45",但是串 strDest 沒有賦值。第二步實(shí)現(xiàn)函數(shù)返回串,但是在函數(shù)出口參數(shù)中沒能進(jìn)行輸出。再次修改 C# 導(dǎo)入定義,將串b修改為引用(ref):
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
public static extern string mySum (string a, ref string b);
}
運(yùn)行時(shí)調(diào)用失敗,不能繼續(xù)執(zhí)行。
第三步,修改動(dòng)態(tài)鏈接庫(kù)實(shí)現(xiàn),將b修改為雙重指針:
LIBEXPORT_API char *mySum(char *a,char **b){sprintf((*b),"%s",a); return *b;}
C#導(dǎo)入定義:
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
public static extern string mySum (string a, ref string b);
}
在C#中調(diào)用測(cè)試:
string strDest="";
string strTmp= RefComm. mySum("45", ref strDest);
運(yùn)行查看結(jié)果 strTmp 和 strDest 均為"45",調(diào)用正確。第三步實(shí)現(xiàn)了函數(shù)出口參數(shù)正確輸出結(jié)果。
第四步,修改動(dòng)態(tài)鏈接庫(kù)實(shí)現(xiàn),實(shí)現(xiàn)整數(shù)參數(shù)的輸出:
LIBEXPORT_API int mySum(int a,int b,int *c){ *c=a+b; return *c;}
C#導(dǎo)入的定義:
public class RefComm
{
[DllImport("LibEncrypt.dll",
EntryPoint=" mySum ",
CharSet=CharSet.Ansi,CallingConvention=CallingConvention.StdCall)]
public static extern int mySum (int a, int b,ref int c);
}
在C#中調(diào)用測(cè)試:
int c=0;
int iSum= RefComm. mySum(,, ref c);
運(yùn)行查看結(jié)果iSum 和c均為5,調(diào)用正確。
經(jīng)過以上幾個(gè)步驟的試驗(yàn),基本掌握了如何定義動(dòng)態(tài)庫(kù)函數(shù)以及如何在 C# 定義導(dǎo)入,有此基礎(chǔ),很快我實(shí)現(xiàn)了變長(zhǎng)加密函數(shù)在 C# 中的調(diào)用,至此目標(biāo)實(shí)現(xiàn)。
三、結(jié)論
在 C# 中調(diào)用 C++ 編寫的動(dòng)態(tài)鏈接庫(kù)函數(shù),如果需要出口參數(shù)輸出,則需要使用指針,對(duì)于字符串,則需要使用雙重指針,對(duì)于 C# 的導(dǎo)入定義,則需要使用引用(ref)定義。
對(duì)于函數(shù)返回值,C# 導(dǎo)入定義和 C++ 動(dòng)態(tài)庫(kù)函數(shù)聲明定義需要保持一致,否則會(huì)出現(xiàn)函數(shù)調(diào)用失敗。定義導(dǎo)入時(shí),一定注意 CharSet 和 CallingConvention 參數(shù),否則導(dǎo)致調(diào)用失敗或結(jié)果異常。運(yùn)行時(shí),動(dòng)態(tài)鏈接庫(kù)放在 C# 程序的目錄下即可,我這里是一個(gè) C# 的動(dòng)態(tài)鏈接庫(kù),兩個(gè)動(dòng)態(tài)鏈接庫(kù)就在同一個(gè)目錄下運(yùn)行。