将 vector<std::string> 转换为本机 PHP 数组

Swig convert vector<std::string> to native PHP array

本文关键字:本机 PHP 数组 转换 gt vector std string lt      更新时间:2023-10-16

我开始使用swigphp,并且我想将std::vector<std::string>返回到我的php代码。我试过这个(所有的编译工作)。

我的swig文件:

%module example
%include <std_string.i>
%include <std_vector.i>
%{
    extern std::vector<std::string> testSwig(const char *path);
%}
//try here to convert vector to native php array
%typemap(out) const std::vector<std::string> & {
    array_init($result);
    Array::const_iterator i = $1->begin();
    Array::const_iterator e = $1->end();
    for(; i != e; ++i ) {
        zval * data;
        MAKE_STD_ZVAL(data);
        ZVAL_STRINGL(data, (char*)i->c_str(), i->size(), 1);
        zend_hash_next_index_insert( HASH_OF($result), &data, sizeof(zval *), NULL );
    }
}
extern std::vector<std::string> testSwig(const char *path);

my CPP file:

#include <iostream>
#include <vector>
std::vector<std::string> testSwig(const char *path)
{
    std::vector<std::string> v;
    v.push_back("test 1");
    v.push_back("test 2");
    v.push_back("try vector");
    return v;
}

最后是我的php文件:

<?php
    echo '<pre>';
    echo print_r(testSwig('try arg'));
    echo '</pre>';
?>

我的问题是,我的转换std::vectorphp数组不工作;当我制作print_r时,我得到了这个结果:resource(1) of type (_p_std__vectorT_std__string_t)

将Flexo提供的答案加起来。

在PHP7 zend API MAKE_STD_ZVAL()宏被删除(包括一些其他的东西:https://wiki.php.net/phpng-upgrading)

为了使这个typemap在PHP7中工作,我做了下面的工作:

%typemap(out) std::vector<std::string> {
    array_init($result);
    $1_type::const_iterator i = $1.begin();
    $1_type::const_iterator e = $1.end();
    for(; i != e; ++i ) {
        add_next_index_string($result, (char*) i->c_str() );
    }
}

testSwig函数从php打印的只是期望的:

Array
(
    [0] => test 1
    [1] => test 2
    [2] => try vector
)

您的代码非常接近于工作,但由于类型不匹配而无法正确应用。您的out类型映射被编写为const std::vector<std::string> &,而您使用的测试函数只是返回std::vector<std::string>,即不是const,按值返回,而不是按引用返回。

有了这个固定,很明显,在你的代码中没有定义称为Array的类型,并且$1实际上是std::vector<std::string>的实例,而不是指向on的指针。我使用$1_ltype使SWIG为typemap的输入生成正确的完全限定类型。因此,为了使你的代码工作,我们需要这样修改它:

%module test
%{
#include <string>
#include <vector>
%}
%typemap(out) std::vector<std::string> {
    array_init($result);
    $1_type::const_iterator i = $1.begin();
    $1_type::const_iterator e = $1.end();
    for(; i != e; ++i ) {
        zval * data;
        MAKE_STD_ZVAL(data);
        ZVAL_STRINGL(data, (char*)i->c_str(), i->size(), 1);
        zend_hash_next_index_insert( HASH_OF($result), &data, sizeof(zval *), NULL );
    }
}
%inline %{
std::vector<std::string> testSwig(const char *path)
{
    std::vector<std::string> v;
    v.push_back("test 1");
    v.push_back("test 2");
    v.push_back("try vector");
    return v;
}
%}

在Ubuntu 14.04和PHP 5上,这足以让下面的测试用例像你希望的那样工作:

<?php
include('test.php');
print_r(test::testSwig('hello'));
?>

我编译并运行为:

swig -php -c++ test.i
g++ test_wrap.cpp $(php-config --includes --libs) -shared -o test.so -fPIC
php5 run.php
Array
(
    [0] => test 1
    [1] => test 2
    [2] => try vector
)

作为参考,SWIG向量和字符串类是非常有用的,给定:

%module test
%include <std_string.i>
%include <std_vector.i>
%template(StringVec) std::vector<std::string>;
%inline %{
std::vector<std::string> testSwig(const char *path)
{
    std::vector<std::string> v;
    v.push_back("test 1");
    v.push_back("test 2");
    v.push_back("try vector");
    return v;
}
%}

类型StringVec是正常的,无需进一步操作即可使用。