2
0
mirror of https://github.com/opencultureconsulting/php-basics.git synced 2025-03-30 00:00:15 +01:00
php-basics/src/DataStructures/StrictQueue.php

152 lines
4.7 KiB
PHP
Raw Normal View History

<?php
/**
2024-01-23 17:58:33 +01:00
* PHP Basics
*
* Copyright (C) 2024 Sebastian Meyer <sebastian.meyer@opencultureconsulting.com>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
2024-01-23 17:58:33 +01:00
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
2024-01-23 17:58:33 +01:00
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
declare(strict_types=1);
namespace OCC\Basics\DataStructures;
2024-01-26 11:40:26 +01:00
use ArrayAccess;
use Countable;
2024-02-12 17:20:45 +01:00
use InvalidArgumentException;
2024-01-26 11:40:26 +01:00
use Iterator;
2024-02-12 17:20:45 +01:00
use RangeException;
use RuntimeException;
2024-01-26 11:40:26 +01:00
use Serializable;
2024-02-16 16:16:05 +01:00
use function sprintf;
/**
2024-02-12 17:20:45 +01:00
* A type-sensitive, taversable queue (FIFO).
*
2024-02-12 17:20:45 +01:00
* Extends [\SplDoublyLinkedList](https://www.php.net/spldoublylinkedlist) with
* an option to restrict the allowed data types for list items by providing the
* constructor with an array of atomic types or fully qualified class names. It
* also restricts the iterator direction to first-in, first-out (FIFO) exactly
* like [\SplQueue](https://www.php.net/splqueue).
2024-01-24 22:45:04 +01:00
*
* @author Sebastian Meyer <sebastian.meyer@opencultureconsulting.com>
2024-01-23 17:58:33 +01:00
* @package Basics\DataStructures
2023-11-19 20:53:44 +01:00
*
2024-01-26 11:40:26 +01:00
* @api
*
2024-01-23 17:58:33 +01:00
* @template AllowedType of mixed
2023-11-21 22:42:55 +01:00
* @extends StrictList<AllowedType>
2024-01-26 11:40:26 +01:00
* @implements ArrayAccess<int, AllowedType>
2024-01-30 22:24:10 +01:00
* @implements Iterator<AllowedType>
*/
2024-01-26 11:40:26 +01:00
class StrictQueue extends StrictList implements ArrayAccess, Countable, Iterator, Serializable
{
/**
* Dequeue an item from the queue.
*
2023-11-21 22:42:55 +01:00
* @return AllowedType The dequeued item
2024-01-26 16:37:41 +01:00
*
* @api
*/
public function dequeue(): mixed
{
return parent::shift();
}
/**
* Add an item to the queue.
*
2024-02-12 17:20:45 +01:00
* @param AllowedType $value The item to enqueue
*
* @return void
*
2024-02-12 17:20:45 +01:00
* @throws InvalidArgumentException if `$value` is not of allowed type
2024-01-26 16:37:41 +01:00
*
* @api
*/
2024-02-12 17:20:45 +01:00
public function enqueue(mixed $value): void
{
2024-02-12 17:20:45 +01:00
parent::push($value);
}
/**
* Set the mode of iteration.
*
* @param int $mode The new iterator mode (0 or 1)
*
2024-01-26 11:40:26 +01:00
* There are two orthogonal sets of modes that can be set.
*
* The direction of iteration (fixed for StrictQueue):
* - StrictQueue::IT_MODE_FIFO (queue style)
*
* The behavior of the iterator (either one or the other):
* - StrictQueue::IT_MODE_DELETE (delete items)
* - StrictQueue::IT_MODE_KEEP (keep items)
*
* The default mode is: IT_MODE_FIFO | IT_MODE_KEEP
*
* @return int The set of flags and modes of iteration
*
2024-02-12 17:20:45 +01:00
* @throws RangeException if an invalid `$mode` is given
* @throws RuntimeException if trying to change iterator direction
2024-01-26 16:37:41 +01:00
*
* @api
*/
final public function setIteratorMode(int $mode): int
{
if ($mode > 1) {
2023-11-18 18:50:14 +01:00
throw new RuntimeException(
sprintf(
'Changing the iterator direction of %s is prohibited.',
static::class
)
);
2023-11-10 22:36:25 +01:00
}
return parent::setIteratorMode($mode);
}
/**
* Create a type-sensitive, traversable queue of items.
*
2024-01-24 22:45:04 +01:00
* @param string[] $allowedTypes Allowed data types of items (optional)
2024-01-26 11:40:26 +01:00
*
2024-01-24 22:45:04 +01:00
* If empty, all types are allowed.
* Possible values are:
* - "array"
* - "bool"
* - "callable"
* - "countable"
* - "float" or "double"
* - "int" or "integer" or "long"
* - "iterable"
* - "null"
* - "numeric"
* - "object" or FQCN
* - "resource"
* - "scalar"
* - "string"
*
2024-01-26 11:40:26 +01:00
* @return void
*
2024-02-12 17:20:45 +01:00
* @throws InvalidArgumentException if any value of `$allowedTypes` is not a string
*/
public function __construct(array $allowedTypes = [])
{
parent::__construct($allowedTypes);
$this->setIteratorMode(0);
}
}