在C++中反射调用.NET的方法(三)

2020-01-06 16:54:54于丽

注意,由于是反射调用静态方法,并且调用方法时候并不需要参数,所以Invoke方法的参数为空。
在C++/CLI中,用nullptr表示空引用,跟C#的null作用一样。

反射调用索引器

SOD实体类可以通过索引器来访问对象属性,例如下面的C#代码:


int id=(int)CurrEntity["ID"];
CurrEntity["Name"]="张三";
string name=(string)CurrEntity["Name"];//张三

下面,我们研究如何通过索引器来给实体类的属性赋值:

我们定义一个 EntityHelper的C++/CLI类,在中间添加下面的代码:


private:
 Type^ entityBuilderType;
 MethodInfo^ mset; 
 Object^ _CurrEntity;
 //Action<String^, Object^>^ idxAction;
 void SetPropertyValue(Object^ entity, MethodInfo^ propMethod, String^ propName, Object^ value)
 {
  array<Object^>^ paraArr = gcnew array<Object^>{propName, value};
  propMethod->Invoke(entity, paraArr);
 }
public:
void set(Object^ value)
{
 this->mset = _CurrEntity->GetType()->GetMethod("set_Item", BindingFlags::Public | BindingFlags::Instance);
 //this->idxAction= (Action<String^, Object^>^)Delegate::CreateDelegate(Action<String^, Object^>::typeid, _CurrEntity, this->mset);
}
void SetPropertyValue(String^ propName, Object^ value)
{
 this->SetPropertyValue(this->CurrEntity, this->mset, propName, value);
 //参数类型为 Object的委托,可能没有性能优势,反而更慢。
 //this->idxAction(propName, value);
}

对索引器的访问,实际上就是调用类的 set_Item 方法,VS编译器会给包含索引器的对象生成这个方法,一般来说我们会对要反射调用的方法创建一个委托,但是实验证明,对索引器使用委托方法调用,反而效率不如直接反射调用,即下面的代码:


void SetPropertyValue(Object^ entity, MethodInfo^ propMethod, String^ propName, Object^ value)
 {
  array<Object^>^ paraArr = gcnew array<Object^>{propName, value};
  propMethod->Invoke(entity, paraArr);
 }

注:C++/CLI 的数组,也可以通过{ } 进行初始化。

一切准备就绪,下面可以通过以下步骤提交集合数据给.NET方法了:

1,反射.NET方法,获取参数的泛型形参类型;

2,创建此泛型形参的泛型List对象实例;

3,遍历C++集合(列表list),将结构数据赋值给动态创建的实体类对象;

4,添加动态实体类到泛型List对象集合内;

5,反射调用.NET方法,提交数据。


//示例1:直接调用.NET强类型的参数方法
 //仅仅适用于有一个参数的情况并且要求是泛型类型参数
 bool SaveUsers(std::list<CppUserInfo> users)
 {
  MethodInfo^ method = dotnetObject->GetType()->GetMethod("SaveUsers", BindingFlags::Public | BindingFlags::Instance);
  array<ParameterInfo^>^ pars = method->GetParameters();
  Type^ paraType= pars[0]->ParameterType;
  Type^ interfaceType = paraType->GetGenericArguments()[0];
  IList^ realList = CreateGenericList(interfaceType);
  Object^ userObj = helper->CreateEntityFromInterface(interfaceType);
  for each (CppUserInfo user in users)
  {
  helper->CurrEntity = ((ICloneable^)userObj)->Clone();//使用克隆,避免每次反射
  helper->SetPropertyValue("ID", user.ID);
  helper->SetPropertyValue("Name", gcnew String(user.Name));
  helper->SetPropertyValue("Birthday", Covert2NetDateTime(user.Birthday));
  realList->Add(helper->CurrEntity);
  }
  Object^ result= method->Invoke(dotnetObject, gcnew array<Object^>{ realList});
  return (bool)result;
 }