[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

相关文章

Java 后端接口入参 - 联合前端VUE 使用AES完成入参出参加密解密

加密效果: 解密后的数据就是正常数据: 后端:使用的是spring-cloud框架,在gateway模块进行操作 <dependency><groupId>com.google.guava</groupId><artifactId>guava</artifactId><version>30.0-jre</version></dependency> 编写一个AES加密

JavaSE——封装、继承和多态

1. 封装 1.1 概念      面向对象程序三大特性:封装、继承、多态 。而类和对象阶段,主要研究的就是封装特性。何为封装呢?简单来说就是套壳屏蔽细节 。     比如:对于电脑这样一个复杂的设备,提供给用户的就只是:开关机、通过键盘输入,显示器, USB 插孔等,让用户来和计算机进行交互,完成日常事务。但实际上:电脑真正工作的却是CPU 、显卡、内存等一些硬件元件。

java线程深度解析(一)——java new 接口?匿名内部类给你答案

http://blog.csdn.net/daybreak1209/article/details/51305477 一、内部类 1、内部类初识 一般,一个类里主要包含类的方法和属性,但在Java中还提出在类中继续定义类(内部类)的概念。 内部类的定义:类的内部定义类 先来看一个实例 [html]  view plain copy pu

模拟实现vector中的常见接口

insert void insert(iterator pos, const T& x){if (_finish == _endofstorage){int n = pos - _start;size_t newcapacity = capacity() == 0 ? 2 : capacity() * 2;reserve(newcapacity);pos = _start + n;//防止迭代

京东物流查询|开发者调用API接口实现

快递聚合查询的优势 1、高效整合多种快递信息。2、实时动态更新。3、自动化管理流程。 聚合国内外1500家快递公司的物流信息查询服务,使用API接口查询京东物流的便捷步骤,首先选择专业的数据平台的快递API接口:物流快递查询API接口-单号查询API - 探数数据 以下示例是参考的示例代码: import requestsurl = "http://api.tanshuapi.com/a

股票数据接口-陈科肇

陈科肇 新浪财经 sz-深圳sh-上海历史分价表:http://market.finance.sina.com.cn/pricehis.php?symbol=sz000506&startdate=2016-12-27&enddate=2016-12-27历史成交明细(当日成交明细):http://vip.stock.finance.sina.com.cn/quotes_service/v

实例demo理解面向接口思想

浅显的理解面向接口编程 Android开发的语言是java,至少目前是,所以理解面向接口的思想是有必要的。下面通过一个简单的例子来理解。具体的概括我也不知道怎么说。 例子: 现在我们要开发一个应用,模拟移动存储设备的读写,即计算机与U盘、MP3、移动硬盘等设备进行数据交换。已知要实现U盘、MP3播放器、移动硬盘三种移动存储设备,要求计算机能同这三种设备进行数据交换,并且以后可能会有新的第三方的

七、Maven继承和聚合关系、及Maven的仓库及查找顺序

1.继承   2.聚合   3.Maven的仓库及查找顺序

对接话费充值API接口的开发步骤以及各种优势

对接话费充值API接口通常涉及以下步骤: 1.选择API提供商: 研究并选择一个可靠的话费充值API提供商。考虑因素包括覆盖范围、费率、交易限额、客户支持和用户评价。 2.注册和获取API密钥: 在选定的API提供商平台上注册账户,并获取API密钥或访问令牌,这是调用API时进行身份验证的必要信息。 3.阅读API文档: 仔细阅读API文档,了解如何构建请求、需要哪些参数、API的

java类中定义接口的有哪些好处

第一步:首先是是定义一个类,同时里面定义接口 public class Util { public interface Worker { void work(int a); } } 第二步:定义一个类去实现第一步类中定义的接口 public class Demo implements Worker { @Override public void work(int a) { System