- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
TLDR:如何在 C# 中为 SWIG 访问模板类型“T”?
假设我在 C++ 中有以下带有 Validate
函数的模板化类:
template<typename T>
struct MyTemplateClass
{
bool Validate(T* _in)
{
//...
}
// ... other stuff... MyTemplateClass is also a container for T*
};
假设我为各种对象实例化类:
%template(MyTemplateClassFoo) MyTemplateClass<Foo>
%template(MyTemplateClassBar) MyTemplateClass<Bar>
// etc.
在 C# 中,我希望 Validate
函数也能验证内存所有权。那就是 _in.swigCMemOwn
是 true
还是 false
,所以我希望 C# 包装器看起来像这样(对于 MyTemplateClassFoo
)
public class MyTemplateClassFoo{
public bool Validate(Foo _in) {
bool ret = _in.swigCMemOwn &&
ModuleCLRPINVOKE.MyTemplateClassFoo_Validate(swigcPtr, Foo.getCPtr(_in));
// SWIGEXCODE stuff
return ret;
}
// ...
}
这里的问题是,如果我想编写自己的Validate
函数,我不知道_in
是什么类型。在 Python 中,我可以使用 feature("shadow")
或 pythonprepend
和 pythonappend
目前我已经做到了:
%csmethodmodifiers MyTemplateClass::Validate "private"; 将 Validate
设为私有(private);
%rename(InternalValidate, fullname=1) "MyTemplateClass::Validate";
Validate
重命名为
InternalValidate
%typemap(cscode)
添加将调用 InternalValidate
的新函数:代码:
%typemap(cscode) MyTemplateClass %{
public bool Validate(/*Type?*/ _in)
{
return _in.swigCMemOwn && InternalValidate(_in);
}
%}
但是我不知道应该为/*Type?*/
指定什么。我试过 T
, T*
, typemap(cstype, T)
但似乎没有一些特殊的 swig 变量,比如$csargtype
我可以用。
我已经尝试研究 SWIG 如何包装 std::vector
,看起来他们可能正在定义一个宏,然后以某种方式为 vector 的每个特化调用它?我想我可以接受,但我不喜欢。
最佳答案
为了完成这些示例,我创建了以下头文件:
template<typename T>
struct MyTemplateClass
{
bool Validate(T* _in)
{
return false;
}
// ... other stuff... MyTemplateClass is also a container for T*
};
struct Foo {};
好消息是,实际上可以生成您要求的代码,比您尝试过的要简单得多。我们可以使用仅匹配Validate
的 csout 类型映射,我们就可以开始了:
%module test
%{
#include "test.hh"
%}
%typemap(csout, excode=SWIGEXCODE) bool Validate {
// referring to _in by name is a bit of a hack here, but it works...
bool ret = _in.swigCMemOwn && $imcall;$excode
return ret;
}
%include "test.hh"
%template(MyTemplateClassInt) MyTemplateClass<int>;
%template(MyTemplateClassFoo) MyTemplateClass<Foo>;
为了完整起见,让我们看看提出的原始问题。首先让我们通过使 MyTemplateClass
实际上不是模板来简化事情(即注释掉 test.hh 的第 1 行并在某处添加 T 的 typedef)。
在那种情况下,您尝试执行的操作非常有效,使用 $typemap(cstype, T)
在 SWIG 编译时查找用于给定类型的 C# 类型:
%module test
%{
#include "test.hh"
%}
%typemap(cscode) MyTemplateClass %{
public bool Validate($typemap(cstype, T) in) {
return in.swigCMemOwn && InternalValidate(in);
}
%}
%rename(InternalValidate) Validate;
%include "test.hh"
然而,当我们再次将其恢复为模板时,生成的代码不正确,生成的 Validate
是:
public bool Validate(SWIGTYPE_p_T in)
发生这种情况是因为 SWIG(至少 3.0,来自 Ubuntu 14.04)在那个上下文中不知道任何关于 T 的信息——模板替换没有正确发生。我不太确定这是错误还是预期行为,但无论哪种方式对我们来说都是一个问题。
但有趣的是,如果您愿意在 SWIG 认为替换有效的模板定义中编写 cscode 类型映射:
%module test
%{
#include "test.hh"
%}
%rename(InternalValidate) Validate;
template<typename T>
struct MyTemplateClass
{
bool Validate(T* _in)
{
return false;
}
// ... other stuff... MyTemplateClass is also a container for T*
%typemap(cscode) MyTemplateClass %{
public bool Validate($typemap(cstype, T) in) {
return in.swigCMemOwn && InternalValidate(in);
}
%}
};
struct Foo {};
%template(MyTemplateClassInt) MyTemplateClass<int>;
%template(MyTemplateClassFoo) MyTemplateClass<Foo>;
在上面的接口(interface)中,T
的类型确实被正确地替换到输出中。因此,如果您愿意接受 .i 文件和您在库中使用的实际头文件之间的重复,那就足够了。您还可以编辑头文件本身并将 SWIG 和 C++ 混合到其中,以下修改后的 test.hh 实现相同的结果:
template<typename T>
struct MyTemplateClass
{
bool Validate(T* _in)
{
return false;
}
// ... other stuff... MyTemplateClass is also a container for T*
#ifdef SWIG
%typemap(cscode) MyTemplateClass %{
public bool Validate($typemap(cstype, T) in) {
return in.swigCMemOwn && InternalValidate(in);
}
%}
#endif
};
struct Foo {};
这是可行的,因为 SWIG 定义了预处理器宏 SWIG,但它不会在正常的 C++ 编译期间定义,所以一切都很好。就我个人而言,我不喜欢那样 - 我宁愿使用干净的边界在逻辑上将 C++ 和 SWIG 位分开。
但是,如果您不愿意像那样复制并且不能/不会简单地编辑头文件,那么所有内容都不会丢失。我们可以(滥用)使用 %extend
来做同样的事情:
%module test
%{
#include "test.hh"
%}
%rename(InternalValidate) Validate;
%include "test.hh"
%extend MyTemplateClass {
%typemap(cscode) MyTemplateClass %{
public bool Validate($typemap(cstype, T) in) {
return in.swigCMemOwn && InternalValidate(in);
}
%}
}
%template(MyTemplateClassInt) MyTemplateClass<int>;
%template(MyTemplateClassFoo) MyTemplateClass<Foo>;
这再次起作用。
最后一个解决方法是,如果您在模板中有一个只使用 T 的 typedef,例如:
template<typename T>
struct MyTemplateClass {
typedef T type;
//...
然后下面的工作,将 typedef 引用为 $1_basetype::type
:
%module test
%{
#include "test.hh"
%}
%rename(InternalValidate) Validate;
%typemap(cscode) MyTemplateClass %{
public bool Validate($typemap(cstype, $1_basetype::type) in) {
return in.swigCMemOwn && InternalValidate(in);
}
%}
%include "test.hh"
%template(MyTemplateClassInt) MyTemplateClass<int>;
%template(MyTemplateClassFoo) MyTemplateClass<Foo>;
因此,尽管看起来应该可行的简单方法似乎并不可行,但仍有很多选择可以实现我们需要的结果。
关于c# - 为 C++ 模板扩展 C# 代理类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39533997/
我是 magento 的新手,目前我在 magento 安装期间遇到“必须加载 PHP 扩展 curl ”错误。你能帮帮我吗? 最佳答案 如果您的服务器上没有安装 curl,您可以键入以下命令之一来安
我在 macOS Mojave/macOS Big Sur/macOS Monterey/macOS Ventura 上使用最新的 php 版本 7.2 并收到类似错误 $composer requ
这个问题已经有答案了: Why generic type is not applicable for argument extends super class for both? (5 个回答) 已关
我正在使用 NightWatch.js 并进行一些 UI 测试,我想用一些额外的 desiredCapabilities 启动默认浏览器实例(即启用扩展并应用一些特定值)。 p> 注意:我可以执行这些
有人知道为什么我在 java 8 中使用此代码时没有服务器扩展名称吗: try { URL url = new URL(urlString); URLC
扩展提供给我的类(class)。为现有的类提供新功能。或扩展现有的mixin s 或虚拟类,任何东西都可以工作。 也许是这样的: class FlatButton {} // maybe no
我有一个关于使用 c 代码和 mod_wsgi 扩展 python 的问题。 我在 apache 服务器中有一个 django 应用程序,它查询 postgresql 数据库以生成报告。在某些报告中,
testcafe支持在Chrome浏览器中加载crx扩展吗? 如果是这样,请告诉我需要尝试什么方法。 我尝试了下面的代码,但没有成功 await t.eval(new Function(fs.read
这个问题已经有答案了: What is a raw type and why shouldn't we use it? (16 个回答) 已关闭 3 年前。 有什么区别: // 1 class A c
我正在编写一个 chrome 扩展来记录单击开始按钮后触发的请求。 这是我的文件:1. list .json { "manifest_version": 2, "name": "recorde
我每天都在使用 vim 和 perforce 现在我的问题是,如果我想查看 perforce 文件修订版,则从命令模式下的 vim :!p4 打印文件#1 vim 试图让我获得缓冲区 #1。有没有办法
大家好,我有一个关于 NUnit 扩展(2.5.10)的问题。 我想做的是向 数据库。为此,我使用 Event 创建了 NUnit 扩展 听众。 我遇到的问题是公共(public)无效 TestFin
我有弹出窗口,而不是模态窗口。 如何通过单击页面的其他部分(不在窗口中)来关闭此窗口? 最佳答案 像这样的东西: function closeWin(e, t) { var el = win.
我通常非常谨慎地使用扩展方法。当我确实觉得有必要编写一个扩展方法时,有时我想重载该方法。我的问题是,您对调用其他扩展方法的扩展方法有何看法?不好的做法?感觉不对,但我无法真正定义原因。 例如,第二个
扩展 Ant Ant带有一组预定义的任务,但是你可以创建自己的任务,如下面的例子所示。 定制Ant 任务应扩展 org.apache.tools.ant.Task 类,同时也应该拓展 execut
我想要一个重定向所有请求的扩展: http://website.com/foo.js 到: http://localhost/myfoo.js 我无法使用主机文件将主机从 website.com 编辑
对于为什么 QChartView 放在 QTabWidget 中时会扩展,我有点迷惑。 这是 QChartView 未展开(因为它被隐藏)时应用程序的图片。 应用程序的黑色部分是 QOpenGLWid
如果在连接条件中使用 OR 运算符,如何优化以下查询以避免 SQL 调优方面的 OR 扩展? SELECT t1.A, t2.B, t1.C, t1.D, t2.E FROM t1 LEFT J
一旦加载插件的问题得到解决(在 .NET 中通过 MEF 的情况下),下一步要解决的是与它们的通信。简单的方法是实现一个接口(interface),使用插件实现,但有时插件只需要扩展应用程序的工作方式
在我的 Symfony2 包中,我需要检查是否定义了一个函数(一个扩展)。更具体地说,如果安装了 KnpMenuBundle,我会在我的包中使用那个,否则我将自己渲染插件。 我试过了,但这当然不起作用
我是一名优秀的程序员,十分优秀!