PHP:使用类似Java的Comparable对自定义类进行排序?

时间:2011-10-05 11:42:48

标签: php sorting

如何使用sort()对自己的自定义类进行排序?

我一直在扫描网页,找到任何制作类的方法,比如Java,但没有太多运气。我尝试实现__equals()但没有运气。我也试过__toString()。我的班级看起来像这样:

class Genre {
    private $genre;
    private $count;
    ...
}

我想按顺序对它们进行排序,这是一个整数,按降序排列......($ genre是一个字符串)

4 个答案:

答案 0 :(得分:11)

您可以创建自定义排序方法并使用http://www.php.net/manual/en/function.usort.php函数来调用它。

示例:

$Collection = array(..); // An array of Genre objects

// Either you must make count a public variable, or create
// an accessor function to access it
function CollectionSort($a, $b)
{
    if ($a->count == $b->count)
    {
        return 0;
    }
    return ($a->count < $b->count) ? -1 : 1;
}

usort($Collection, "CollectionSort");

如果您想制作更通用的收藏系统,可以试试这样的

interface Sortable
{
    public function GetSortField();
}

class Genre implements Sortable
{
    private $genre;
    private $count;

    public function GetSortField()
    {
        return $count;
    }
}

class Collection
{
    private $Collection = array();

    public function AddItem($Item)
    {
        $this->Collection[] = $Item;
    }

    public function GetItems()
    {
        return $this->Collection;
    }

    public function Sort()
    {
        usort($this->Collection, 'GenericCollectionSort');
    }
}

function GenericCollectionSort($a, $b)
{
    if ($a->GetSortField() == $b->GetSortField())
    {
        return 0;
    }
    return ($a->GetSortField() < $b->GetSortField()) ? -1 : 1;
}

$Collection = new Collection();
$Collection->AddItem(...); // Add as many Genre objects as you want
$Collection->Sort();
$SortedGenreArray = $Collection->GetItems();

答案 1 :(得分:3)

也许您可以使用“usort”功能:

class Genre {
    private $genre;
    private $count;
    ...

    public function __construct($g, $c)
    {
       $this->genre=g;
       $this->count=c;
    }


    public static function compare($a, $b)
    {
        if ($a->count < $b->count) return -1;
        else if($a->count == $b->count) return 0;
        else return 1;
    }

    ...
}



$genres= array(
  new Genre (1, 5),
  new Genre (2, 2),
  new Genre (3, 7)
);

usort($genres, array("Genre", "compare"));

关心托马斯

答案 2 :(得分:0)

最简单的方法是在类中的方法上使用usort,该方法接受匹配对象数组作为输入。这是与上面链接的文档中的示例3。然而,这有点笨拙和难看。

更好的方法是使用ArrayAccess创建特定于所需对象的新类型的数组类。这将使您能够像预期的那样使用自定义数组,但也可以在数组上运行任意排序方法。

在幕后,你可能仍然想要使用usort,但是你会把这个事实隐藏在一个更好的界面背后,可能看起来像这样:

$array = new GenreCollection();

$array[] = new Genre(1); // Genre's constructor is a factory that can load genres via an ID
$array[] = new Genre(2);
$array[] = new Genre(3);
$array[] = new Genre(4);
$array[] = new Genre(5);

// Example Sorts
$array->SortByName();
$array->SortByDateInvented();
$array->SortByID();
$array->SortBySubGenres(); // Arranges Genres into a hierarchy where 'Death Metal' comes with other metal after 'Heavy Metal' - Add in a nest-level value for making dropdowns and other nested lists.

答案 3 :(得分:-1)

__toString()适合我:

class Genre
{
    private $genre;
    private $count;

    public function __construct( $genre, $count = 0 )
    {
        $this->genre = $genre;
        $this->count = $count;
    }

    public function __toString()
    {
        return $this->count . ' ' . $this->genre;
    }    
}

$collection = array(
    new Genre( 'alternative', 3 ),
    new Genre( 'jazz', 2 ),
    new Genre( 'hiphop', 1 ),
    new Genre( 'heavy metal', 1 )
);

natsort( $collection );

foreach( $collection as $genre )
{
    echo $genre . "\n";
}

产地:

1 heavy metal
1 hiphop
2 jazz
3 alternative