Чи краще використовувати module_invoke_all (), або module_invoke () та module_implements ()?


10

Використовувати module_invoke_all('some_hook')це просто, але чи краще це для продуктивності?

foreach (module_implements('some_hook') as $module) {
  $data = module_invoke($module, 'some_hook');
}

Відповіді:


11

Різниця невелика; module_invoke_all()виконує наступний код:

function module_invoke_all() {
  $args = func_get_args();
  $hook = $args[0];
  unset($args[0]);
  $return = array();
  foreach (module_implements($hook) as $module) {
    $function = $module . '_' . $hook;
    if (function_exists($function)) {
      $result = call_user_func_array($function, $args);
      if (isset($result) && is_array($result)) {
        $return = array_merge_recursive($return, $result);
      }
      elseif (isset($result)) {
        $return[] = $result;
      }
    }
  }

  return $return;
}

Єдина відмінність полягає в тому, що з module_invoke_all(), наприклад, func_get_args()викликається лише один раз, тоді як при використанні module_invoke() func_get_args()викликається кожен раз module_invoke()викликається; але це гранична різниця.
Є кілька випадків, коли module_implementing()і module_invoke()використовуються, як правило, коли модулю необхідно знати, до якого модуля викликається, наприклад, у випадку, search_get_info()коли він створює масив інформації про модулі, що реалізують пошукові функції.

function search_get_info($all = FALSE) {
  $search_hooks = &drupal_static(__FUNCTION__);

  if (!isset($search_hooks)) {
    foreach (module_implements('search_info') as $module) {
      $search_hooks[$module] = call_user_func($module . '_search_info');
      // Use module name as the default value.
      $search_hooks[$module] += array(
        'title' => $module,
        'path' => $module,
      );
      // Include the module name itself in the array.
      $search_hooks[$module]['module'] = $module;
    }
  }

  if ($all) {
    return $search_hooks;
  }

  $active = variable_get('search_active_modules', array('node', 'user'));
  return array_intersect_key($search_hooks, array_flip($active));
}

Інший приклад - image_styles () , який отримує список усіх стилів зображення, реалізованих модулями, і в якому використовується наступний код:

  foreach (module_implements('image_default_styles') as $module) {
    $module_styles = module_invoke($module, 'image_default_styles');
    foreach ($module_styles as $style_name => $style) {
      $style['name'] = $style_name;
      $style['module'] = $module;
      $style['storage'] = IMAGE_STORAGE_DEFAULT;
      foreach ($style['effects'] as $key => $effect) {
        $definition = image_effect_definition_load($effect['name']);
        $effect = array_merge($definition, $effect);
        $style['effects'][$key] = $effect;
      }
      $styles[$style_name] = $style;
    }
  }

В обох випадках отримана інформація поміщається в масив, де індекс - це коротке ім'я модуля.



-1

Напевно, краще не використовувати жодне, а натомість використовувати drupal_alter () .

Як зазначено в документації module_invoke_all () ,

Усі аргументи передаються за значенням. Використовуйте drupal_alter (), якщо вам потрібно передати аргументи за посиланням.

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.