Передать дополнительные параметры для обратного вызова usort - PullRequest
37 голосов
/ 22 ноября 2011

У меня есть следующие функции.WordPress работает, но это действительно вопрос PHP.Они сортируют мои $term объекты в соответствии со свойством artist_lastname в метаданных каждого объекта.

Я хочу передать строку в $meta в первой функции.Это позволило бы мне повторно использовать этот код, поскольку я мог бы применить его к различным свойствам метаданных.

Но я не понимаю, как я могу передать дополнительные параметры в обратный вызов usort.Я пытался создать анонимную функцию в стиле JS, но версия PHP на сервере слишком старая и вызвала синтаксическую ошибку.

Любая помощь - или толчок в правый угол руководства - с благодарностью признательна.Спасибо!

function sort_by_term_meta($terms, $meta) 
{
  usort($terms,"term_meta_cmp");
}

function term_meta_cmp( $a, $b ) 
{
    $name_a = get_term_meta($a->term_id, 'artist_lastname', true);
    $name_b = get_term_meta($b->term_id, 'artist_lastname', true);
    return strcmp($name_a, $name_b); 
} 

Ответы [ 5 ]

80 голосов
/ 24 марта 2014

Думаю, этот вопрос заслуживает обновления. Я знаю, что первоначальный вопрос касался PHP версии 5.2, но я пришел сюда в поисках решения и нашел решение для более новых версий PHP и подумал, что это может быть полезно и для других людей.

Для PHP 5.3 и выше вы можете использовать ключевое слово ' use ', чтобы ввести локальные переменные в локальную область действия анонимной функции. Поэтому должно работать следующее:

function sort_by_term_meta(&$terms, $meta) {
    usort($terms, function($a, $b) use ($meta) {
        $name_a = get_term_meta($a->term_id, 'artist_lastname', true);
        $name_b = get_term_meta($b->term_id, 'artist_lastname', true);
        return strcmp($name_a, $name_b);  
    });
}

Еще немного общего кода

Если вы хотите отсортировать массив только один раз и вам нужен дополнительный аргумент, вы можете использовать анонимную функцию, подобную этой:

usort($arrayToSort, function($a, $b) use ($myExtraArgument) {
    //$myExtraArgument is available in this scope
    //perform sorting, return -1, 0, 1
    return strcmp($a, $b);
});

Если вам нужна многократно используемая функция для сортировки массива, который нуждается в дополнительном аргументе, вы всегда можете обернуть анонимную функцию, как в исходном вопросе:

function mySortFunction(&$arrayToSort, $myExtraArgument1, $myExtraArgument2) {
    usort($arrayToSort, function($a, $b) use ($myExtraArgument1, $myExtraArgument2) {
        //$myExtraArgument1 and 2 are available in this scope
        //perform sorting, return -1, 0, 1
        return strcmp($a, $b);
    });
}
23 голосов
/ 22 ноября 2011

В PHP одна опция для обратного вызова заключается в передаче двухэлементного массива, содержащего дескриптор объекта и имя метода для вызова объекта.Например, если $obj был экземпляром класса MyCallable, и вы хотите вызвать MyCallable метод *1006* для $obj, тогда вы можете передать array($obj, "method1") в качестве обратного вызова.

Одним из решений, использующих этот поддерживаемый тип обратного вызова, является определение одноразового класса, который по существу действует как тип замыкания:

function sort_by_term_meta( $terms, $meta ) 
{
    usort($terms, array(new TermMetaCmpClosure($meta), "call"));
}

function term_meta_cmp( $a, $b, $meta )
{
    $name_a = get_term_meta($a->term_id, $meta, true);
    $name_b = get_term_meta($b->term_id, $meta, true);
    return strcmp($name_a, $name_b); 
} 

class TermMetaCmpClosure
{
    private $meta;

    function __construct( $meta ) {
        $this->meta = $meta;
    }

    function call( $a, $b ) {
        return term_meta_cmp($a, $b, $this->meta);
    }
}
6 голосов
/ 22 ноября 2011

Предполагая, что у вас есть доступ к объектам и статическим объектам (PHP 5 или более поздней версии), вы можете создать объект и передать непосредственно туда аргументы, например:

<?php
class SortWithMeta {
    private static $meta;

    static function sort(&$terms, $meta) {
       self::$meta = $meta;
       usort($terms, array("SortWithMeta", "cmp_method"));
    }

    static function cmp_method($a, $b) {
       $meta = self::$meta; //access meta data
       // do comparison here
    }

}

// then call it
SortWithMeta::sort($terms, array('hello'));

При условии, что у вас нет доступак объектам / статика;Вы можете просто сделать глобальный:

$meta = array('hello'); //define meta in global

function term_meta_cmp($a, $b) {
   global $meta; //access meta data
   // do comparison here
}

usort($terms, 'term_meta_cmp');
3 голосов
/ 22 ноября 2011

В документах сказано, что create_function() должно работать на PHP> = 4.0.1. Это работает?

function term_meta_cmp( $a, $b, $meta )  {
    echo "$a, $b, $meta<hr>"; // Debugging output
}
$terms = array("d","c","b","a");
usort($terms, create_function('$a, $b', 'return term_meta_cmp($a, $b, "some-meta");'));
1 голос
/ 22 ноября 2011

Это совсем не поможет вам с usort(), но, тем не менее, может быть полезным. Вы можете отсортировать массив, используя одну из других функций сортировки, array_multisort().

Идея состоит в том, чтобы создать массив значений, по которым вы будете сортировать (возвращаемые значения из get_term_meta()), и мультисортировать их в соответствии с вашим основным $terms массивом.

function sort_by_term_meta(&$terms, $meta) 
{
    $sort_on = array();
    foreach ($terms as $term) {
        $sort_on[] = get_term_meta($term->term_id, $meta, true);
    }
    array_multisort($sort_on, SORT_ASC, SORT_STRING, $terms);
}
...