青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

posts - 17,  comments - 2,  trackbacks - 0

Calling Managed Code from Unmanaged Code and vice-versa

By TarunNeo

This article shows you how to call managed code from unmanaged code and also the other way round.
C++/CLI, VB, VC7.1, C++, Windows, .NET, .NET 1.1VS.NET2003, Visual Studio, Dev

Posted21 Mar 2005
Updated21 Mar 2005 
Views78,822
Bookmarked50 times
22 votes for this Article.
Popularity: 5.47 Rating: 4.07 out of 5
0 votes, 0.0%
1
1 vote, 4.5%
2
3 votes, 13.6%
3
8 votes, 36.4%
4
10 votes, 45.5%
5

Introduction

.NET framework is one of the better development and execution environments for software these days. But there is a very huge amount of software components already developed and being developed out of unmanaged code. So, there needs to be an easy way for managed code to call the unmanaged code and the other way round.

I have seen some articles on this, but I did not find them giving a complete solution of what I was looking for. So here is one.

Background

Microsoft lets you call COM code from .NET code using RCW (Runtime Callable Wrappers). The RCW, a managed wrapper code, wraps the COM component. The .NET code then interacts with the RCW which in turn interacts with the COM component inside it. The reverse communication can be done using CCW (COM callable wrapper).

This article shows a way of manually creating a wrapper. It was fairly easy to call the unmanaged code from the managed code but not the other way around.

Code

The code that I have specified below consists of:

  • Unmanaged class: UnManaged_Class
  • Managed Wrapper class: Managed_Wrapper_Class

This class wraps the unmanaged class. This means that it “contains” an object of the unmanaged type which it uses to call the exposed methods in the unmanaged type.

  • Managed code: Managed_Class

This is how the managed code calls the unmanaged code:

For every exposed method in the unmanaged class, there should be a corresponding method in the Managed_Wrapper_Class. The managed code instantiates an object of theManaged_Wrapper_Class and calls the exposed methods in that class using this instance. These methods in the Managed_Wrapper_Class then call the corresponding methods in the unmanaged code. This is done using pInner as shown in the code:

//

/*//////////////////////////////////////////////////
//Unmanaged_Class.cpp
//////////////////////////////////////////////////*/


#ifndef UNMANAGED
#define UNMANAGED

class Unmanaged_Class
{
public:

    Unmanaged_Class();
    
    /*This is the method that is to be called from Managed code*/
    void methodToBeCalledInUnmanaged(int data);
};

#endif


*//////////////////////////////////////////////////

//Unmanaged_Class.cpp

//////////////////////////////////////////////////*/


#include "StdAfx.h"

#using <mscorlib.dll>
#include "Unmanaged.h"


Unmanaged_Class::Unmanaged_Class()
{
}

void Unmanaged_Class::methodToBeCalledInUnmanaged(int data)
{
    /*Here is the place where the Managed Wrapper code is called. */
    scallback(data+1);
}


/*//////////////////////////////////////////////////
//Managed_Wrapper.h 
//////////////////////////////////////////////////*/

#pragma once

#include "stdafx.h"

#using <mscorlib.dll>
#include "Unmanaged.h"


using namespace System::Runtime::InteropServices;
using namespace System;

namespace Managed_Wrapper
{

    /*Managed Wrapper Class */
    public __gc class Managed_Wrapper_Class
    {
    public: 
    
        //constructor

        Managed_Wrapper_Class();

        /* pInner is used to invoke the exposed 
        methods in the unmanaged class. */
        Unmanaged_Class * pInner; 


        /* An exposed function corresponding 
        to the exopsed function in Unmanaged*/
        void CallUnmanaged(int data);

    };
}

/*//////////////////////////////////////////////////
//Managed_Wrapper.cpp
//////////////////////////////////////////////////*/

#include "stdafx.h"

#include "Managed_Wrapper.h"

#using <mscorlib.dll>

namespace Managed_Wrapper
{
    Managed_Wrapper_Class::Managed_Wrapper_Class(void)
    {
        /* define the pInner object */
        pInner = new Unmanaged_Class();
    }


    void Managed_Wrapper_Class::CallUnmanaged(int data)
    {
        pInner->methodToBeCalledInUnmanaged (data);
    }

}
'/*//////////////////////////////////////////////////

'//Managed Code

'//VB.NET code

'//////////////////////////////////////////////////*/



'Import the Managed Wrapper Namespace

Imports Managed_Wrapper


'Create an instance of the Managed Wrapper class.

Dim forwardCaller As Managed_Wrapper_Class = New Managed_Wrapper_Class


'To call a method in the Managed_Wrapper_Class. This method in 

'turn will call the method in the unmanaged code

forwardCaller.CallUnmanaged(nudNumber.Value)

This was the easy part. Now is the hard part. Calling managed code from the unmanaged code.

The unmanaged code has a function pointer. The address of the function pointer is the address of a method in the managed wrapper class. Also, the function pointer is initialized by the wrapper class and not in the unmanaged code. So this way, when the function pointer is called, the method in the managed wrapper code is called. Half of the task is done. The way of assigning the function pointer in the unmanaged code is not easy because the function has to point to a method which is managed. So, we use the wrapper delegate struct as shown in the code. Then convert this delegate struct to a function pointer of unmanaged type using Marshal::StructureToPtr (_Instance_Of_Delegate_Wrapper, &type_unmanaged_functionptr, false);

The managed wrapper code declares a delegate (a .NET way of a function pointer). The delegate is instantiated by the managed code. So when the method in the managed wrapper class is called (by the unmanaged code), it in turn calls the delegate in the same class (which is initialized by the managed code). As the delegate points to a function in the managed code, the method in the managed code gets called. This was the hard part.

/*///////////////////////////////////////////////////
/*Unmanaged_Class.h */
///////////////////////////////////////////////////*/


#ifndef UNMANAGED
#define UNMANAGED

#using <mscorlib.dll>

typedef void (*w_CallBack) (int status);

class Unmanaged_Class
{
public:

 Unmanaged_Class();
 w_CallBack scallback;

 /* To set the callback function. The address in ptr2F will be the
 address of a method in the Managed Wrapper class and will be assigned
 there. In this case it will be address of ActualMethodInWrapper(int );*/
 void setCallBackInUnmanaged(w_CallBack ptr2F);

 /*This is the method that is to be called from Managed code*/
 void methodToBeCalledInUnmanaged(int data);
};

#endif
/*///////////////////////////////////////////////////
//Unmanaged_Class.cpp
///////////////////////////////////////////////////*/

#include "StdAfx.h"

#using <mscorlib.dll>
#include "Unmanaged.h"


Unmanaged_Class::Unmanaged_Class()
{
}

void Unmanaged_Class::setCallBackInUnmanaged(w_CallBack ptr2F)
{
 /*scallback now points to ActualMethodInWrapper(int) in
 Managed_Wrapper_Class*/
 scallback = ptr2F;
}

void Unmanaged_Class::methodToBeCalledInUnmanaged(int data)
{
 /*Here is the place where the Managed Wrapper code is called. */
 scallback(data+1);
}
/*///////////////////////////////////////////////////
//Managed_Wrapper.h
///////////////////////////////////////////////////*/

#pragma once

#include "stdafx.h"

#using <mscorlib.dll>
#include "Unmanaged.h"


using namespace System::Runtime::InteropServices;
using namespace System;

namespace Managed_Wrapper
{

 /*Declare a delegate. It is to be invoked from the unmanaged code*/
 public __delegate void CallbackDelegate(int data);

 /* Declare a wrapping struct that wraps an object of the above 
 declared delegate. The delegate that this struct contains will 
 point to a method that will be called when this delegate is 
 invoked from the unmanaged code*/
 [StructLayoutAttribute( LayoutKind::Sequential, CharSet = CharSet::Ansi )]
 public __gc struct Managed_Delegate_Wrapper
 {
 [MarshalAsAttribute(UnmanagedType::FunctionPtr)]
 CallbackDelegate* _Delegate;
 };

 /*Managed Wrapper Class */
 public __gc class Managed_Wrapper_Class
 {
 public:

 //constructor

 Managed_Wrapper_Class();

 /* pInner is used to invoke the exposed methods in the unmanaged class. */
 Unmanaged_Class * pInner;

 /* Declare an instance of the wrapping struct */
 Managed_Delegate_Wrapper * _Status_Delegate;

 /* A method that will be called when the callback function in the unmanaged
 code is called. It is this method who’s pointer is passed to the unmanaged
 code.*/
 void ActualMethodInWrapper(int );

 /*A delegate type. To be used for calling managed code from here.*/
 __delegate int statusDelegate(int status);

 /*An object of the above delagate type is declared.
 It will be initialized in the managed code.*/
 statusDelegate *statD;

 /* An exposed function corresponding to the exopsed function in Unmanaged*/
 void CallUnmanaged(int data);

 };
}
/*///////////////////////////////////////////////////
//Managed_Wrapper.cpp
///////////////////////////////////////////////////*/

#include "stdafx.h"

#include "Managed_Wrapper.h"

#using <mscorlib.dll>

namespace Managed_Wrapper
{
 Managed_Wrapper_Class::Managed_Wrapper_Class(void)
 {
 /* define the pInner object */
 pInner = new Unmanaged_Class();

 /* define the wrapping struct instance declared in Managed_Wrapper_Class */
 _Status_Delegate = new Managed_Delegate_Wrapper();

 /* This is the actual delegate that is contained in the wraping struct */
 _Status_Delegate->_Delegate = 
   new CallbackDelegate(this, &Managed_Wrapper_Class::ActualMethodInWrapper);

 /* declare a function pointer of the same type as in unmanaged code */
 w_CallBack callback;

 /*convert the wrapping struct to a function pointer of the above type. */
 Marshal::StructureToPtr (_Status_Delegate, &callback, false);

 /* set this function pointer in the unmanaged code using pInner.*/
 pInner->setCallBackInUnmanaged(callback);
 }

 /*This is the method in the Managed_Wrapper_Class that is called
 when the function pointer in the unmanaged code is called.*/
 void Managed_Wrapper_Class::ActualMethodInWrapper(int status)
 {
 /*This method in turn calls the delegate in the managed code.
 The method that statD is actually poiting is specified in the
 managed code itself.*/
 statD(status);
 }

 void Managed_Wrapper_Class::CallUnmanaged(int data)
 {
 pInner->methodToBeCalledInUnmanaged (data);
 }

}
'/*//////////////////////////////////////////////////

'//Managed Code

'//VB.NET code

'//////////////////////////////////////////////////*/


'Import the Managed Wrapper Namespace

Imports Managed_Wrapper

'Create an instance of the Managed Wrapper class.

Dim forwardCaller As Managed_Wrapper_Class = New Managed_Wrapper_Class

'Create an instance of the delegate declared in the Managed Wrapper

'class. Initialize it with the address of the method that is supposed

'to be called when the delegate in the Managed_Wrapper_Class is called

Dim statDelg As New Managed_Wrapper_Class.statusDelegate(AddressOf status_Recd)

'This function gets called when the unmanaged code calls the

'managed wrapper code and which in turn calls the the delegate

'in there

Public Function status_Recd(ByVal status As Integer) As Integer

 'use status for something now. It took so much effort to get it :)

 MessageBox.Show("Status received is " + status.ToString(), "Status" + 
                 " received from the Unmanaged code")

End Function

 'To call a method in the Managed_Wrapper_Class. This method in

 'turn will call the method in the unmanaged code

 forwardCaller.CallUnmanaged(nudNumber.Value)

 'statD is called from the managed code. And statD in turn

 'will call the method status_Recd

 forwardCaller.statD = statDelg

Compiling and Linking

Choose a C++ .NET Class Library project to wrap your unmanaged code and compile it to generate the DLL. Then in your VB.NET code, add a reference to this DLL using Add Reference->Projects (Browse to the DLL). Also you would need to have your Project properties as in the demo project on the top.

Demo

Open the Managed_VBdotNET.sln solution and start it. Bingo.

Summary

I found this technique particularly useful for one of my projects in which I had some code which was already written in C++ and was a good idea to have it written in C++. I needed to add a GUI to it, for which VB.NET was a very straightforward choice. Through this way I could invoke C++ methods through VB.NET and VB.NET methods through C++.

Any suggestions are welcome and will be appreciated. Please feel free to ask any questions.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

TarunNeo


Tarun is a Computer Science Grad. He believes in
".......In between and after is glorious coding"


Occupation:Web Developer
Location:United States United States
posted on 2008-11-14 11:40 BeyondCN 閱讀(1048) 評(píng)論(0)  編輯 收藏 引用 所屬分類: .NET
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            91久久嫩草影院一区二区| 亚洲国产精品第一区二区三区| 欧美精品一区在线发布| 欧美在线啊v| 欧美一区二区三区免费在线看| 午夜精品久久久久久99热软件| 亚洲一区在线播放| 亚洲欧美日韩系列| 久久精品主播| 欧美11—12娇小xxxx| 欧美男人的天堂| 国产区在线观看成人精品| 国产欧美一区二区三区沐欲 | 国产在线不卡精品| 国内自拍一区| 一区二区三区三区在线| 亚洲主播在线| 久热这里只精品99re8久| 欧美高清在线一区| 亚洲一区在线直播| 麻豆国产精品va在线观看不卡| 欧美日韩成人网| 国产日韩一区二区三区在线播放| 在线播放亚洲一区| 亚洲性av在线| 亚洲国产精品va在线看黑人| 一个色综合av| 欧美不卡一区| 国外成人在线视频| 亚洲午夜精品久久久久久app| 久久一综合视频| 日韩视频在线永久播放| 欧美一区午夜精品| 欧美日韩伦理在线| 亚洲国产美女精品久久久久∴| 欧美伊人久久久久久久久影院| 亚洲国产美女久久久久| 午夜电影亚洲| 国产精品久久久久9999吃药| 亚洲精品在线三区| 欧美不卡福利| 久久久精品一区| 国产精品美女久久福利网站| 亚洲人成人99网站| 午夜国产不卡在线观看视频| 韩国av一区| 久久看片网站| 欧美国产高清| 在线看片一区| 久久国产精品一区二区三区| 最新高清无码专区| 美女视频黄免费的久久| 国产专区一区| 亚洲欧美一区二区三区在线| 亚洲精品免费看| 欧美不卡在线| 亚洲伦理在线| 亚洲国产视频直播| 老司机午夜精品| 亚洲第一区中文99精品| 久久精品中文字幕一区二区三区| 亚洲综合第一| 国产精品影片在线观看| 亚洲一区二区动漫| 中国av一区| 国产精品久久久久秋霞鲁丝 | 国产精品网红福利| 午夜精品久久| 久久国产精品久久精品国产| 国产一区日韩二区欧美三区| 久久乐国产精品| 久久精品一本久久99精品| 国产一区二区三区直播精品电影| 久久久不卡网国产精品一区| 久久精彩免费视频| 一区二区在线视频观看| 欧美电影在线播放| 欧美精品在线视频| 欧美一区二区日韩一区二区| 欧美一区在线直播| 亚洲国产高清高潮精品美女| 亚洲国产日韩一区二区| 欧美日在线观看| 欧美在线观看网址综合| 亚洲无亚洲人成网站77777| 国产精一区二区三区| 久久资源在线| 欧美美女操人视频| 午夜久久一区| 久久精品国产99精品国产亚洲性色 | 狠狠久久五月精品中文字幕| 欧美成人69av| 国产精品福利网| 噜噜爱69成人精品| 欧美国产视频在线观看| 亚洲欧美中文日韩v在线观看| 久久精品视频网| 亚洲视频第一页| 久热精品视频在线观看一区| 亚洲手机视频| 狼人社综合社区| 亚洲男人第一av网站| 亚洲日本国产| 亚洲东热激情| 久久久噜噜噜久久人人看| 亚洲精品久久久久久久久| 亚洲一区二区免费| 亚洲国产视频直播| 欧美一区1区三区3区公司| 亚洲毛片在线看| 久久久久久9999| 午夜精品久久久久久久99樱桃 | 性欧美激情精品| 亚洲精品免费在线| 欧美在线看片a免费观看| 亚洲专区一区二区三区| 欧美大片va欧美在线播放| 久久久久综合一区二区三区| 欧美亚洲第一页| 亚洲久久一区二区| 亚洲高清久久| 久久爱www.| 久久精品国产亚洲5555| 欧美亚洲第一区| avtt综合网| 中文有码久久| 欧美裸体一区二区三区| 欧美成人69| 精品51国产黑色丝袜高跟鞋| 亚洲欧美精品伊人久久| 亚洲影视九九影院在线观看| 欧美日韩国产一级| 亚洲精品日韩综合观看成人91| 亚洲国产视频直播| 猛男gaygay欧美视频| 欧美aa在线视频| 亚洲精品美女在线观看播放| 欧美91大片| 亚洲精品在线一区二区| 亚洲黄色小视频| 欧美激情在线观看| 亚洲精品男同| 亚洲一区视频| 国产精品夜夜嗨| 欧美一区影院| 欧美波霸影院| 99日韩精品| 国产精品久久一级| 欧美一进一出视频| 欧美黄色片免费观看| 99re6这里只有精品视频在线观看| 欧美精品免费观看二区| 夜夜嗨av一区二区三区免费区| 亚洲综合日本| 极品日韩av| 欧美精品一区二区三区久久久竹菊 | 亚洲精品日韩综合观看成人91| 欧美丰满高潮xxxx喷水动漫| 欧美激情一区二区| 中文av一区特黄| 国产美女精品视频| 久久在线精品| 一本久久综合| 日韩视频精品在线| 国产精品99久久久久久www| 欧美日韩免费观看一区三区| 亚洲网友自拍| 久久午夜国产精品| 亚洲福利在线看| 欧美视频在线播放| 欧美亚洲网站| 亚洲精品九九| 久久久国产精品亚洲一区| 亚洲国产精品一区二区尤物区 | 久热成人在线视频| 亚洲老板91色精品久久| 欧美一区国产二区| 亚洲欧洲一区二区三区在线观看| 国产精品国产三级国产普通话蜜臀| 午夜精品www| 91久久精品国产| 久久久久国产精品一区| 亚洲视频图片小说| 亚洲成色777777女色窝| 国产精品尤物福利片在线观看| 欧美激情1区2区3区| 欧美自拍丝袜亚洲| 一本色道婷婷久久欧美| 欧美二区不卡| 久久蜜桃精品| 欧美一区二区免费| 日韩午夜免费视频| 亚洲国产天堂久久综合网| 国产精品日韩欧美| 欧美日韩三级电影在线| 久久综合久久久| 久久精品夜色噜噜亚洲a∨| 亚洲午夜在线| 99精品视频免费| 日韩视频在线观看免费|