首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >包含指针的NativeCall结构

包含指针的NativeCall结构
EN

Stack Overflow用户
提问于 2016-02-06 21:13:06
回答 1查看 414关注 0票数 7

我有以下结构:

代码语言:javascript
复制
typedef struct _info{
  DWORD myInfo;
  BYTE  *pInfo;
  LPWSTR ExtData;

} Info;

因此,我使用NativeCall表示这个结构:

代码语言:javascript
复制
class Info is repr('CStruct') {
    has int32 $.myInfo;
    has Pointer[int8] $.pInfo ; 
    has Pointer[int16] $.ExtData;
}

这个代表可以吗?如何访问和设置$.pInfo所指向的数据

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2016-02-22 11:22:45

在我看来这个表象是可以的..。鉴于这个被发明的C库:(请不要挑剔我的C语言-)

代码语言:javascript
复制
#include <stdlib.h>
#include <stdio.h>
#include <string.h>

typedef unsigned long int DWORD;
typedef unsigned char BYTE;
typedef char * LPWSTR;

typedef struct _info{
  DWORD   myInfo;
  BYTE   *pInfo;
  LPWSTR  ExtData;
} Info;


Info* create_info();
void  display_info();
void  double_info(Info*);

Info*
create_info() {
    DWORD init_myinfo    = 2016;
    BYTE  init_pinfo     = 37;
    char  init_ExtData[] = "Hello World";

    Info *new_info = malloc(sizeof(Info));
    if (new_info == NULL) {
        printf( "Memory alloc failed\n" );
        exit(1);
    }

    new_info->myInfo = init_myinfo;

    BYTE *pinfo = malloc(sizeof(BYTE));
    *pinfo = init_pinfo;
    new_info->pInfo = pinfo;

    char *ext_data = malloc(sizeof(init_ExtData));
    strcpy(ext_data, init_ExtData);
    new_info->ExtData = ext_data;

    return new_info;
}

void
display_info(Info *inf) {
    printf("myInfo: %lu\n", inf->myInfo);
    printf("pInfo:  %i\n", *inf->pInfo);
    printf("ExtData: %s\n", inf->ExtData);
}

void
double_info(Info *inf) {

    inf->myInfo *=  2;

    if ( *(inf->pInfo) < 128 )
        *(inf->pInfo) *= 2;

    int extdata_len = strlen(inf->ExtData);
    char *tmp_extdata = malloc(extdata_len * 2 + 2);
    strncpy(tmp_extdata, inf->ExtData, extdata_len);
    tmp_extdata[extdata_len] = '+';
    strcpy(&tmp_extdata[extdata_len+1], inf->ExtData);
    inf->ExtData = tmp_extdata;
}

然后,您使用NativeCall的Raku结构定义或多或少会工作:

代码语言:javascript
复制
#!/usr/bin/env raku
use NativeCall;

class Info is repr('CStruct') {
    has int32          $.myInfo;
    has Pointer[int8]  $.pInfo ;
    has Pointer[Str]   $.ExtData;

    method Str {
      qq:to/END HERE/;
      myInfo: $!myInfo
      pInfo: { $!pInfo.deref }
      ExtData: { $!ExtData.deref }
      END HERE
    }
}

our sub create_info()  returns Info is native('pinfo') { * }
our sub display_info(Info) is native('pinfo') { * }
our sub double_info(Info is rw) is native('pinfo') { * }

my Info $inf = create_info();
say 'Displaying $inf after calling create_info()';
display_info $inf;

double_info $inf;
say 'Displaying $inf after calling double_info()';
display_info $inf;

say 'Displaying $inf by calling attribute methods on Raku object';
say "myInfo: $inf.myInfo()";
say "pInfo: $inf.pInfo.deref()";
say "ExtData: $inf.ExtData.deref()";

say 'Displaying $inf by stringifying Raku object';
say "$inf";
exit 0;

这一连串的生产;

代码语言:javascript
复制
Displaying $inf after calling create_info()
myInfo: 2016
pInfo:  37
ExtData: Hello World
Displaying $inf after calling double_info()
myInfo: 4032
pInfo:  74
ExtData: Hello World+Hello World
Displaying $inf by calling attribute methods on Raku object
myInfo: 4032
pInfo: 74
ExtData: Hello World+Hello World
Displaying $inf by stringifying Raku object
myInfo: 4032
pInfo: 74
ExtData: Hello World+Hello World

或者,你可以在Raku类中隐藏更多的‘胶水’;

代码语言:javascript
复制
#!/usr/bin/env raku
use NativeCall;

class Info is repr('CStruct') {
    has int32          $.myInfo is rw ;
    has Pointer[int8]  $!pInfo ;
    has Pointer[Str]   $!ExtData ;

    my sub create_info() returns Info is native('pinfo') { * }
    my sub display_info(Info)         is native('pinfo') { * }
    my sub double_info(Info is rw)    is native('pinfo') { * }

    method new     {  create_info()      }
    method display {  display_info(self) }
    method double  {  double_info(self)  }

    method pInfo   {  $!pInfo.deref      }
    method ExtData {  $!ExtData.deref    }

    method Str {
        qq:to/END HERE/;
        myInfo: {  self.myInfo  }
        pInfo: {   self.pInfo   }
        ExtData: { self.ExtData }
        END HERE
    }
}

my Info $inf .= new;
say 'Displaying $inf after calling .new';
$inf.display ;

$inf.double ;
say 'Displaying $inf after calling .double';
$inf.display ;

say 'Displaying $inf by calling attribute methods on Raku object';
say "myInfo: $inf.myInfo()";
say "pInfo: $inf.pInfo()";
say "ExtData: $inf.ExtData()";

$inf.myInfo = 12046 ;
say 'Displaying $inf by stringifying Raku object';
say "$inf";
exit 0;

..。看上去干净多了。它也同样产生;

代码语言:javascript
复制
Displaying $inf after calling .new
myInfo: 2016
pInfo:  37
ExtData: Hello World
Displaying $inf after calling .double
myInfo: 4032
pInfo:  74
ExtData: Hello World+Hello World
Displaying $inf by calling attribute methods on Raku object
myInfo: 4032
pInfo: 74
ExtData: Hello World+Hello World
Displaying $inf by stringifying Raku object
myInfo: 12046
pInfo: 74
ExtData: Hello World+Hello World

备注:

(1)对类型指针的属性调用.deref()方法,以获取所指向的实际数据。

(2)必须多次读取NativeCall文档 ;-)

(3)无论我尝试了什么,我都无法修改通过指针引用的数据--我一直得到“无法修改不变的数据”。

(4) YMMV

票数 6
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/35246529

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档