【问题标题】:How to handle hash value type in Perl XS如何在 Perl XS 中处理哈希值类型
【发布时间】:2015-06-26 11:45:00
【问题描述】:

我需要处理哈希值取决于值类型。这是有问题的代码:

I32 keys = hv_iterinit(hash);
for (I32 i = 0; i < keys; i++)
{
  char *key = NULL;
  I32 key_length = 0;
  SV *value = hv_iternextsv(hash, &key, &key_length);
  // SvROK(value);
  if (SvTYPE(SvRV(value)) < SVt_PVAV)
  {
    // handle scalar
    printf("key %s has scalar value\n", key);
  }
  else if (SvTYPE(SvRV(value)) == SVt_PVAV)
  {
    // handle array
    printf("key %s has array value\n", key);
  }
  else if (SvTYPE(SvRV(value)) == SVt_PVHV)
  {
    // handle hash
    printf("key %s has hash value\n", key);
  }
}

如果我不使用注释行,我会遇到标量值问题。例如下面的哈希 {a =&gt; "b", c =&gt; {d =&gt; "e"}} 是产生输出:

key c has hash value
key d has scalar value

所以这是我的问题:

  1. 我们是否总是有从hv_iternextsv() 返回的引用,或者有时它返回标量?
  2. 为什么我看不到键 a 的标量值输出。

更新。

我的错误在于使用hv_iternextsv() 的结果。我一直在想这总是一个参考。以下是工作代码的样子:

I32 keys = hv_iterinit(hash);
for (I32 i = 0; i < keys; i++)
{
  char *key = NULL;
  I32 key_length = 0;
  SV *value = hv_iternextsv(hash, &key, &key_length);
  if (!SvROK(value))
  {
    // handle scalar
  }
  else
  {
    if (SvTYPE(SvRV(value)) == SVt_PVAV)
    {
      // handle array
    }
    else if (SvTYPE(SvRV(value)) == SVt_PVHV)
    {
      // handle hash
    }
  }
}

【问题讨论】:

  • 我希望您的粘贴缺少一些括号,因为那将无法编译。也许你的意思是:if (SvTYPE(SvRV(value)) == SVt_PVAV) ?
  • 代码已更新,谢谢。
  • @ikegami 我更新了我的问题。我没有看到键 a 的标量值输出的问题。
  • 您还错误地迭代了哈希。我的回答中还演示了如何迭代哈希。
  • 迭代有什么问题?如果此代码不正确,为什么它会起作用?

标签: perl hash perl-xs


【解决方案1】:

我们是否总是有从hv_iternextsv() 返回的引用,或者有时它返回标量?

它总是返回一个标量。哈希值只能是标量。这些标量可以是引用 ($h{x} = [];),但不一定是 ($h{y} = 123;)。

为什么我没有看到键 a 的标量值输出。

您不可能返回您所说的内容,因为您的哈希没有名为d 的键。对于您提供的哈希,您的代码会输出以下内容:

key a has scalar value
key c has hash value

但你得到正确答案更多的是巧合。 SvTYPE(SvRV(value))value 不是参考???这是没有意义的!固定代码如下:

use strict;
use warnings;

use Inline C => <<'__EOI__';

  void print_keys(HV* hash) {
    char *key;
    I32 key_length;
    SV *value;

    hv_iterinit(hash);
    while (value = hv_iternextsv(hash, &key, &key_length)) {
      if (SvROK(value)) {
        SV * const referenced = SvRV(value);
        if (SvTYPE(referenced) == SVt_PVAV) {
          printf("The value at key %s is reference to an array\n", key);
        }
        else if (SvTYPE(referenced) == SVt_PVHV) {
           printf("The value at key %s is a reference to a hash\n", key);
        }
        else {
           printf("The value at key %s is a reference\n", key);
        }
      } else {
        printf("The value at key %s is not a reference\n", key);
      }
    }
  }

__EOI__

print_keys({a => "b", c => {d => "e"}});

输出:

The value at key a is not a reference
The value at key c is a reference to a hash

【讨论】:

  • 我再次调用该代码进行哈希,因此它产生了该输出。我添加了参考检查,它显示键 a 的值不是参考。所以我的第二个问题仍然有效。
  • 不要撒谎。它已被回答,但你在一小时后更改了它。如果您有新问题,请在 cmets 中提出或酌情作为问题提出。 // 您可以使用SvTYPE 检查标量的类型,但您为什么要这样做。如果你想要一个字符串,请使用SvPVutf8。如果你想要一个整数,使用SvIV等。你不应该关心它是如何存储的。
  • 我不撒谎,问题出在第一个问题上,我认为哈希条目值始终是参考,但在某些情况下并非如此。所以谢谢你的回答。
  • 你说“所以我的第二个问题仍然有效。”在你完全替换了第二个问题之后。
猜你喜欢
  • 2014-07-17
  • 2011-05-23
  • 2013-01-04
  • 2016-09-19
  • 2011-05-15
  • 1970-01-01
  • 2023-04-07
  • 1970-01-01
  • 2013-08-26
相关资源
最近更新 更多