[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

相关文章

SpringBoot实现接口数据加解密的三种实战方案

《SpringBoot实现接口数据加解密的三种实战方案》在金融支付、用户隐私信息传输等场景中,接口数据若以明文传输,极易被中间人攻击窃取,SpringBoot提供了多种优雅的加解密实现方案,本文将从原... 目录一、为什么需要接口数据加解密?二、核心加解密算法选择1. 对称加密(AES)2. 非对称加密(R

Java对接Dify API接口的完整流程

《Java对接DifyAPI接口的完整流程》Dify是一款AI应用开发平台,提供多种自然语言处理能力,通过调用Dify开放API,开发者可以快速集成智能对话、文本生成等功能到自己的Java应用中,本... 目录Java对接Dify API接口完整指南一、Dify API简介二、准备工作三、基础对接实现1.

Python多重继承慎用的地方

《Python多重继承慎用的地方》多重继承也可能导致一些问题,本文主要介绍了Python多重继承慎用的地方,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面... 目录前言多重继承要慎用Mixin模式最后前言在python中,多重继承是一种强大的功能,它允许一个

Java controller接口出入参时间序列化转换操作方法(两种)

《Javacontroller接口出入参时间序列化转换操作方法(两种)》:本文主要介绍Javacontroller接口出入参时间序列化转换操作方法,本文给大家列举两种简单方法,感兴趣的朋友一起看... 目录方式一、使用注解方式二、统一配置场景:在controller编写的接口,在前后端交互过程中一般都会涉及

usb接口驱动异常问题常用解决方案

《usb接口驱动异常问题常用解决方案》当遇到USB接口驱动异常时,可以通过多种方法来解决,其中主要就包括重装USB控制器、禁用USB选择性暂停设置、更新或安装新的主板驱动等... usb接口驱动异常怎么办,USB接口驱动异常是常见问题,通常由驱动损坏、系统更新冲突、硬件故障或电源管理设置导致。以下是常用解决

go中空接口的具体使用

《go中空接口的具体使用》空接口是一种特殊的接口类型,它不包含任何方法,本文主要介绍了go中空接口的具体使用,具有一定的参考价值,感兴趣的可以了解一下... 目录接口-空接口1. 什么是空接口?2. 如何使用空接口?第一,第二,第三,3. 空接口几个要注意的坑坑1:坑2:坑3:接口-空接口1. 什么是空接

如何用java对接微信小程序下单后的发货接口

《如何用java对接微信小程序下单后的发货接口》:本文主要介绍在微信小程序后台实现发货通知的步骤,包括获取Access_token、使用RestTemplate调用发货接口、处理AccessTok... 目录配置参数 调用代码获取Access_token调用发货的接口类注意点总结配置参数 首先需要获取Ac

讯飞webapi语音识别接口调用示例代码(python)

《讯飞webapi语音识别接口调用示例代码(python)》:本文主要介绍如何使用Python3调用讯飞WebAPI语音识别接口,重点解决了在处理语音识别结果时判断是否为最后一帧的问题,通过运行代... 目录前言一、环境二、引入库三、代码实例四、运行结果五、总结前言基于python3 讯飞webAPI语音

MyBatis-Plus中Service接口的lambdaUpdate用法及实例分析

《MyBatis-Plus中Service接口的lambdaUpdate用法及实例分析》本文将详细讲解MyBatis-Plus中的lambdaUpdate用法,并提供丰富的案例来帮助读者更好地理解和应... 目录深入探索MyBATis-Plus中Service接口的lambdaUpdate用法及示例案例背景

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链