[SWIG] 多继承与接口(%interface、%interface_impl、%interface_custom)

2024-02-13 11:48

本文主要是介绍[SWIG] 多继承与接口(%interface、%interface_impl、%interface_custom),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原文链接:https://www.yuque.com/cpptd/swig/xvh0b6
SWIG系列笔记:https://www.yuque.com/cpptd/swig

文章目录

      • 引言
      • SWIG中的接口
      • 将模板类申明为接口
          • 1. idescription.hpp
          • 2. example.hpp
          • 3. example.i
      • 源码:swiginterface.i

引言

SWIG报错:

warning 833: Warning for ITask, base IParameter ignored. Multiple inheritance is not supported in C#.

与C#类似,许多语言都没有多继承,但是它们有接口的概念,一个类可以实现多个接口。因此,我们可以将C++多继承改为多接口的实现。

注:C++虽然支持多继承,但不推荐使用多继承,它会带来很多很多复杂的耦合问题。但C++推荐一种多继承的形式:一个是主的继承类,其他都是接口(or抽象基类)。这也是后来的语言所提倡的(它们也是借鉴C++的经验):单继承是父类,其他是实现接口。

SWIG中的接口

关于接口的申明,SWIG提供了三种方法,分别是:%interface%interface_implinterface_custom

请添加图片描述
说明:

  1. Cpp是C++中的类图
  2. A、B、C是包装后,目标语言中的类图(这里以C#举例)

将模板类申明为接口

在一些很奇怪的场景下,我们也要将模板类申明成接口。

1. idescription.hpp

这是一个接口,它是模板类。

#pragma once
#include<string>namespace example{template <typename DerivedClassName>class IDescription{public:IDescription(const std::string& name) :_mName(name) {}virtual ~IDescription() = default;public:std::string name() const{return _mName;}DerivedClassName& setName(const std::string& name){_mName = name;return *(DerivedClassName*)this;}std::string description() const{return _mDescription;}DerivedClassName& setDescription(const std::string& description){_mDescription = description;return *(DerivedClassName*)this;}protected:std::string		_mName; 		//<- 名称std::string		_mDescription;	//<- 详细说明};};

2. example.hpp

这里有一个基类和派生类。派生类DerivedClass继承了基类BaseClass,并实现了IDescription接口。

#pragma once
#include<string>#include"idescription.hpp"namespace example{class BaseClass{public:virtual ~BaseClass() = default;virtual std::string className(){return "BaseClass";}};class DerivedClass : public BaseClass, public IDescription<DerivedClass>{public:DerivedClass(const std::string& name) : IDescription(name){}std::string className() override{return "DerivedClass";}};};

3. example.i
%module example%{
#include "idescription.hpp"
#include "example.hpp"
%}%include"std_string.i"//自定义一个template_interface宏,完成对模板类的接口定义
%include <swiginterface.i>
%define %template_interface(CTYPE, TEMPLATE_IMPL, INTERFACE...)
%interface_custom(#TEMPLATE_IMPL, #INTERFACE, CTYPE);
%template(#TEMPLATE_IMPL) CTYPE;
%enddef%include"idescription.hpp"%template_interface(example::IDescription<example::DerivedClass>, DerivedClassDescription, IDerivedClassDescription); //这句话要在%include"example.hpp"之前,即在example::IDescription<example::DerivedClass>之前
%include"example.hpp"

源码:swiginterface.i

/* -----------------------------------------------------------------------------* swiginterface.i** SWIG interface feature and typemaps implementation providing:*   %interface*   %interface_impl*   %interface_custom* ----------------------------------------------------------------------------- */%define INTERFACE_TYPEMAPS(CTYPE...)
%typemap(cstype) CTYPE "$&csinterfacename"
%typemap(cstype) CTYPE *, CTYPE [], CTYPE & "$csinterfacename"
%typemap(cstype) CTYPE *const& "$*csinterfacename"
%typemap(csin) CTYPE, CTYPE & "$csinput.GetInterfaceCPtr()"
%typemap(csin) CTYPE *, CTYPE *const&, CTYPE [] "$csinput == null ? new global::System.Runtime.InteropServices.HandleRef(null, global::System.IntPtr.Zero) : $csinput.GetInterfaceCPtr()"
%typemap(csout, excode=SWIGEXCODE) CTYPE {$&csclassname ret = new $&csclassname($imcall, true);$excodereturn ($&csinterfacename)ret;}
%typemap(csout, excode=SWIGEXCODE) CTYPE & {$csclassname ret = new $csclassname($imcall, $owner);$excodereturn ($csinterfacename)ret;}
%typemap(csout, excode=SWIGEXCODE) CTYPE *, CTYPE [] {global::System.IntPtr cPtr = $imcall;$csclassname ret = (cPtr == global::System.IntPtr.Zero) ? null : new $csclassname(cPtr, $owner);$excodereturn ($csinterfacename)ret;}
%typemap(csout, excode=SWIGEXCODE) CTYPE *const& {global::System.IntPtr cPtr = $imcall;$*csclassname ret = (cPtr == global::System.IntPtr.Zero) ? null : new $*csclassname(cPtr, $owner);$excodereturn ($*csinterfacename)ret;}
%typemap(csdirectorin) CTYPE "($&csinterfacename)new $&csclassname($iminput, true)"
%typemap(csdirectorin) CTYPE & "($csinterfacename)new $csclassname($iminput, false)"
%typemap(csdirectorin) CTYPE *, CTYPE [] "($iminput == global::System.IntPtr.Zero) ? null : ($csinterfacename)new $csclassname($iminput, false)"
%typemap(csdirectorin) CTYPE *const& "($iminput == global::System.IntPtr.Zero) ? null : ($*csinterfacename)new $*csclassname($iminput, false)"
%typemap(csdirectorout) CTYPE, CTYPE *, CTYPE *const&, CTYPE [], CTYPE & "$cscall.GetInterfaceCPtr()"
%typemap(csinterfacecode, declaration="  [System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)]\n  global::System.Runtime.InteropServices.HandleRef GetInterfaceCPtr();\n", cptrmethod="$interfacename_GetInterfaceCPtr") CTYPE %{[System.ComponentModel.EditorBrowsable(System.ComponentModel.EditorBrowsableState.Never)]global::System.Runtime.InteropServices.HandleRef $interfacename.GetInterfaceCPtr() {return new global::System.Runtime.InteropServices.HandleRef(this, $imclassname.$csclazzname$interfacename_GetInterfaceCPtr(swigCPtr.Handle));}
%}
%enddef%define %interface(CTYPE...)
%feature("interface", name="%sSwigInterface") CTYPE;
INTERFACE_TYPEMAPS(CTYPE)
%enddef%define %interface_impl(CTYPE...)
%rename("%sSwigImpl") CTYPE;
%feature("interface", name="%(rstrip:[SwigImpl])s") CTYPE;
INTERFACE_TYPEMAPS(CTYPE)
%enddef%define %interface_custom(PROXY, INTERFACE, CTYPE...)
%rename(PROXY) CTYPE;
%feature("interface", name=INTERFACE) CTYPE;
INTERFACE_TYPEMAPS(CTYPE)
%enddef

这篇关于[SWIG] 多继承与接口(%interface、%interface_impl、%interface_custom)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/705450

相关文章

关于C++中的虚拟继承的一些总结(虚拟继承,覆盖,派生,隐藏)

1.为什么要引入虚拟继承 虚拟继承是多重继承中特有的概念。虚拟基类是为解决多重继承而出现的。如:类D继承自类B1、B2,而类B1、B2都继承自类A,因此在类D中两次出现类A中的变量和函数。为了节省内存空间,可以将B1、B2对A的继承定义为虚拟继承,而A就成了虚拟基类。实现的代码如下: class A class B1:public virtual A; class B2:pu

vue3项目将所有访问后端springboot的接口统一管理带跨域

vue3项目将所有访问后端springboot的接口统一管理带跨域 一、前言1.安装Axios2.创建Axios实例3.创建API服务文件4.在组件中使用API服务 二、跨域三、总结 一、前言 在Vue 3项目中,统一管理所有访问后端Spring Boot接口的最佳实践是创建一个专门的API服务层。这可以让你的代码更加模块化、可维护和集中管理。你可以使用Axios库作为HTT

BD错误集锦7——在集成Spring MVC + MyBtis时使用c3p0作为数据库时报错Method com/mchange/v2/c3p0/impl/NewProxyPreparedStatem

异常信息如下: Type Exception ReportMessage Handler dispatch failed; nested exception is java.lang.AbstractMethodError: Method com/mchange/v2/c3p0/impl/NewProxyPreparedStatement.isClosed()Z is abstractDescr

第三十七章 添加和使用自定义标题元素 - 自定义标头的继承

文章目录 第三十七章 添加和使用自定义标题元素 - 自定义标头的继承自定义标头的继承示例 在 `SOAPHEADERS` 参数中指定支持的标头元素自定义标头的继承 第三十七章 添加和使用自定义标题元素 - 自定义标头的继承 自定义标头的继承 如果创建此Web 服务的子类,该子类将继承不特定于方法的标头信息 — 包含在 <request> 或 <response> 元素中的标头信

linux常用API接口

linux常用API接口 文章目录 linux常用API接口1.应用层内存映射mmap取消内存映射munmap终端打印可用方式1.puts 函数2.文件操作函数 fprintf3.字符输出函数 putchar4.fwrite 函数 2.内核层 1.应用层 内存映射mmap mmap 是一个用于内存映射的系统调用,它可以将一个文件或设备中的内容映射到进程的地址空间中,允许程

js实现继承的多种方式

1:原型链方式,即子类通过prototype将所有在父类中通过prototype追加的属性和方法都追加到Child,从而实现了继承   function Person(){   }   Person.prototype.hello = "hello";   Person.prototype.sayHello = function(){     alert(this.h

上位机图像处理和嵌入式模块部署(mcu和swd接口)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing @163.com】         最近学习mcu的时候,接触了不少调试器,这里面有daplink、st-link v2、j-link v9。虽然模块的形状可能不太一样,但是硬件的连线都差不多,都是mcu上的3.3v、clk、dio和gnd四根连线。出于好奇,今天花了点时间了解了一下debug port、sw

【架构设计】模块化-面向接口编程

模块化的优势 结构清晰:业务独立,代码实现分离便于协作:耦合度低,不会影响团队其他成员的开发进度便于维护:各模块管理自身代码、布局、资源,不影响主工程 模块化的特点 高内聚,低耦合 常见的模块化方法 所有模块都放在一个工程里实现每个模块都放在一个独立工程中实现 模块的划分 从高到低 业务层平台能力层基础层 模块间的通信 直接依赖事件或广播通信路由通信面向接口通信 具体实现

使用ig507金融数据库的股票API接口经验有感:Java与Python

一、Java技术: 1. Java调用ig507金融数据库(ig507.com)股票API接口 引言: 随着金融科技的不断发展,数据驱动的投资策略变得越来越重要。本文将介绍如何使用Java语言调用ig507金融数据库的股票API接口,以获取实时股票数据,并展示基本的编程步骤和注意事项。 步骤一:引入依赖库 在Java项目中,首先需要引入用于处理HTTP请求和网络通信的库,如Apache

Creating custom and compound Views in Android - Tutorial(翻译)

Creating custom and compound Views in Android - Tutorial(翻译) 译前的: 之前做了三篇学习笔记,从知乎上面看到了这篇英文的推荐,总的来说可以是一篇导读,没有相关的学习,看这篇,可以作为一个学习脉络导向;有相关的学习底子,可以作为一个基础夯实、思维理清。没想到一翻译就是四个多小时…英语渣,很多词句都不太准确,幸好有之前的学习基础打底…