5

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

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 => "b", c => {d => "e"}}是产生输出:

key c has hash value
key d has scalar value

所以这是我的问题:

  1. 我们是否总是有从返回的引用,hv_iternextsv()或者有时它返回标量?
  2. 为什么我看不到 key 的标量值输出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
    }
  }
}
4

1 回答 1

3

我们是否总是有从返回的引用,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
于 2015-06-26T13:58:24.040 回答