As classes de coleção fornecem um conjunto de ferramentas para manipular matrizes
ou objetos Traversable
. Se você já usou underscore.js, tem uma idéia do que
pode esperar das classes de coleção.
Instâncias de coleção são imutáveis; modificar uma coleção irá gerar uma nova coleção. Isso torna o trabalho com objetos de coleção mais previsível, pois as operações são livres de efeitos colaterais.
Coleções podem ser criadas usando uma matriz ou um objeto Traversable
. Você
também interagirá com as coleções sempre que interagir com o ORM no CakePHP. Um
simples uso de uma coleção seria:
use Cake\Collection\Collection;
$items = ['a' => 1, 'b' => 2, 'c' => 3];
$collection = new Collection($items);
// Crie uma nova coleção contendo elementos
// com um valor maior que um.
$overOne = $collection->filter(function ($value, $key, $iterator) {
return $value > 1;
});
Você também pode usar a função auxiliar collection()
em vez de new Collection()
:
$items = ['a' => 1, 'b' => 2, 'c' => 3];
// Ambos formam uma instância de coleção.
$collectionA = new Collection($items);
$collectionB = collection($items);
O benefício do método auxiliar é que é mais fácil encadear do que (new Collection($items))
.
O Cake\Collection\CollectionTrait
permite integrar recursos semelhantes a
coleções em qualquer objeto Traversable
que você possui no seu aplicativo.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
As coleções podem ser iteradas e/ou transformadas em novas coleções com os
métodos each()
e map()
. O método each()
não criará uma
nova coleção, mas permitirá que você modifique quaisquer objetos dentro da
coleção:
$collection = new Collection($items);
$collection = $collection->each(function ($value, $key) {
echo "Element $key: $value";
});
O retorno de each()
será um objeto collection. Cada um iterará a coleção
imediatamente aplicando o retorno de chamada a cada valor na coleção.
O método map()
criará uma nova coleção com base no retorno
de chamada que está sendo aplicada a cada objeto na coleção original:
$items = ['a' => 1, 'b' => 2, 'c' => 3];
$collection = new Collection($items);
$new = $collection->map(function ($value, $key) {
return $value * 2;
});
// $result contém [2, 4, 6];
$result = $new->toList();
// $result contém ['a' => 2, 'b' => 4, 'c' => 6];
$result = $new->toArray();
O método map()
criará um novo iterador que cria preguiçosamente os
itens resultantes quando iterado.
Um dos usos mais comuns de uma função map()
é extrair uma única coluna
de uma coleção. Se você deseja criar uma lista de elementos contendo os valores
de uma propriedade específica, pode usar o método extract()
:
$collection = new Collection($people);
$names = $collection->extract('name');
// $result contém ['mark', 'jose', 'barbara'];
$result = $names->toList();
Como em muitas outras funções da classe de coleção, você pode especificar um caminho separado por pontos para extrair colunas. Este exemplo retornará uma coleção que contém os nomes dos autores de uma lista de artigos:
$collection = new Collection($articles);
$names = $collection->extract('author.name');
// $result contém ['Maria', 'Stacy', 'Larry'];
$result = $names->toList();
Por fim, se a propriedade que você está procurando não pode ser expressa como um caminho, você pode usar uma função de retorno de chamada para retorná-la:
$collection = new Collection($articles);
$names = $collection->extract(function ($article) {
return $article->author->name . ', ' . $article->author->last_name;
});
Freqüentemente, existem propriedades necessárias para extrair uma chave comum
presente em várias matrizes ou objetos profundamente aninhados em outras estruturas.
Para esses casos, você pode usar o combinador {*}
na chave do caminho.
Esse correspondente geralmente é útil ao combinar dados da associação HasMany e BelongsToMany:
$data = [
[
'name' => 'James',
'phone_numbers' => [
['number' => 'number-1'],
['number' => 'number-2'],
['number' => 'number-3'],
]
],
[
'name' => 'James',
'phone_numbers' => [
['number' => 'number-4'],
['number' => 'number-5'],
]
]
];
$numbers = (new Collection($data))->extract('phone_numbers.{*}.number');
$numbers->toList();
// Retorna ['number-1', 'number-2', 'number-3', 'number-4', 'number-5']
Este último exemplo usa toList()
diferente de outros exemplos, o que é
importante quando estamos obtendo resultados com chaves possivelmente duplicadas.
Ao usar toList()
, garantimos a obtenção de todos os valores, mesmo que haja
chaves duplicadas.
Ao contrário de Cake\Utility\Hash::extract()
este método suporta
apenas o curinga {*}
. Todos os outros correspondentes de curinga e atributos
não são suportados.
Coleções permitem que você crie uma nova coleção feita de chaves e valores em uma coleção existente. Os caminhos de chave e valor podem ser especificados com notação de caminhos com ponto:
$items = [
['id' => 1, 'name' => 'foo', 'parent' => 'a'],
['id' => 2, 'name' => 'bar', 'parent' => 'b'],
['id' => 3, 'name' => 'baz', 'parent' => 'a'],
];
$combined = (new Collection($items))->combine('id', 'name');
// O resultado ficará assim quando convertido em array
[
1 => 'foo',
2 => 'bar',
3 => 'baz',
];
Opcionalmente, você também pode usar um groupPath
para agrupar resultados com base em um caminho:
$combined = (new Collection($items))->combine('id', 'name', 'parent');
// O resultado ficará assim quando convertido em array
[
'a' => [1 => 'foo', 3 => 'baz'],
'b' => [2 => 'bar']
];
E por fim, você pode usar closures para criar caminhos de chaves/valores/grupos dinamicamente,
por exemplo, ao trabalhar com entidades e datas (convertidas em instâncias Cake/Time
pelo ORM),
você pode querer agrupar os resultados por data:
$combined = (new Collection($entities))->combine(
'id',
function ($entity) { return $entity; },
function ($entity) { return $entity->date->toDateString(); }
);
// O resultado ficará assim quando convertido em array
[
'date string like 2015-05-01' => ['entity1->id' => entity1, 'entity2->id' => entity2, ..., 'entityN->id' => entityN]
'date string like 2015-06-01' => ['entity1->id' => entity1, 'entity2->id' => entity2, ..., 'entityN->id' => entityN]
]
Você pode parar a iteração a qualquer momento usando o método stopWhen()
.
A chamada em uma coleção criará uma nova e irá interromper a execução de novos resultados
se a chamada passada retornar verdadeira para um dos elementos:
$items = [10, 20, 50, 1, 2];
$collection = new Collection($items);
$new = $collection->stopWhen(function ($value, $key) {
// Pare no primeiro valor maior que 30
return $value > 30;
});
// $result contém [10, 20];
$result = $new->toList();
Às vezes, os itens internos de uma coleção contêm matrizes ou iteradores com mais
itens. Se você deseja nivelar a estrutura interna para iterar uma vez todos os
elementos, pode usar o método unfold()
. Ele criará uma nova coleção que
produzirá todos os elementos aninhados na coleção:
$items = [[1, 2, 3], [4, 5]];
$collection = new Collection($items);
$new = $collection->unfold();
// $result contém [1, 2, 3, 4, 5];
$result = $new->toList();
Ao passar uma chamada para unfold()
, você pode controlar quais elementos
serão desdobrados de cada item da coleção original. Isso é útil para retornar
dados de serviços paginados:
$pages = [1, 2, 3, 4];
$collection = new Collection($pages);
$items = $collection->unfold(function ($page, $key) {
// Um serviço da web imaginário que retorna uma página de resultados
return MyService::fetchPage($page)->toList();
});
$allPagesItems = $items->toList();
Se você estiver usando o PHP 5.5+, você pode usar a palavra-chave yield
dentro de
unfold()
para retornar quantos elementos de cada item da coleção você precisará:
$oddNumbers = [1, 3, 5, 7];
$collection = new Collection($oddNumbers);
$new = $collection->unfold(function ($oddNumber) {
yield $oddNumber;
yield $oddNumber + 1;
});
// $result contém [1, 2, 3, 4, 5, 6, 7, 8];
$result = $new->toList();
Ao lidar com grandes quantidades de itens em uma coleção, pode fazer sentido
processar os elementos em lotes, em vez de um por um. Para dividir uma coleção
em várias matrizes de um determinado tamanho, você pode usar a função chunk()
:
$items = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11];
$collection = new Collection($items);
$chunked = $collection->chunk(2);
$chunked->toList(); // [[1, 2], [3, 4], [5, 6], [7, 8], [9, 10], [11]]
A função chunk
é particularmente útil ao realizar o processamento em lote, por
exemplo, com um resultado no banco de dados:
$collection = new Collection($articles);
$collection->map(function ($article) {
// Alterar uma propriedade no artigo
$article->property = 'changed';
})
->chunk(20)
->each(function ($batch) {
myBulkSave($batch); // Esta função será chamada para cada lote
});
Bem como chunk()
, chunkWithKeys()
permite dividir uma coleção
em lotes menores, mas com as chaves preservadas. Isso é útil ao agrupar matrizes associativas:
$collection = new Collection([
'a' => 1,
'b' => 2,
'c' => 3,
'd' => [4, 5]
]);
$chunked = $collection->chunkWithKeys(2)->toList();
// Cria
[
['a' => 1, 'b' => 2],
['c' => 3, 'd' => [4, 5]]
]
As coleções facilitam a filtragem e a criação de novas coleções com base no resultado
das funções de retorno de chamada. Você pode usar filter()
para criar uma nova
coleção de elementos que correspondem a um retorno de chamada de critério:
$collection = new Collection($people);
$ladies = $collection->filter(function ($person, $key) {
return $person->gender === 'female';
});
$guys = $collection->filter(function ($person, $key) {
return $person->gender === 'male';
});
O inverso de filter()
é reject()
. Este método cria um filtro negativo,
removendo elementos que correspondem à função de filtro:
$collection = new Collection($people);
$ladies = $collection->reject(function ($person, $key) {
return $person->gender === 'male';
});
Você pode fazer testes de verificação com funções de filtro. Para ver se todos
os elementos de uma coleção correspondem a um teste, você pode usar every()
:
$collection = new Collection($people);
$allYoungPeople = $collection->every(function ($person) {
return $person->age < 21;
});
Você pode ver se a coleção contém pelo menos um elemento que corresponde a
uma função de filtro usando o método some()
:
$collection = new Collection($people);
$hasYoungPeople = $collection->some(function ($person) {
return $person->age < 21;
});
Se você precisar extrair uma nova coleção contendo apenas os elementos que
contêm um determinado conjunto de propriedades, use o método match()
:
$collection = new Collection($comments);
$commentsFromMark = $collection->match(['user.name' => 'Mark']);
O nome da propriedade pode ser um caminho separado por pontos. Você pode
percorrer entidades aninhadas e verificar valores que elas contêm.
Quando você só precisa do primeiro elemento correspondente de uma coleção,
pode usar firstMatch()
:
$collection = new Collection($comments);
$comment = $collection->firstMatch([
'user.name' => 'Mark',
'active' => true
]);
Como você pode ver acima, match()
e firstMatch()
permitem fornecer
várias condições para a correspondência. Além disso, as condições
podem ser para caminhos diferentes, permitindo expressar condições complexas
para comparação.
A contraparte de uma operação map()
geralmente é uma reduce
.
Esta função ajudará você a criar um único resultado de todos os elementos
em uma coleção:
$totalPrice = $collection->reduce(function ($accumulated, $orderLine) {
return $accumulated + $orderLine->price;
}, 0);
No exemplo acima, $totalPrice
será a soma de todos os preços únicos contidos na
coleção. Observe que o segundo argumento para da função reduce()
assume o valor
inicial da operação de redução que você está executando:
$allTags = $collection->reduce(function ($accumulated, $article) {
return array_merge($accumulated, $article->tags);
}, []);
Para extrair o valor mínimo de uma coleção com base em uma propriedade,
basta usar a função min()
. Isso retornará o elemento completo da coleção
e não apenas o menor valor encontrado:
$collection = new Collection($people);
$youngest = $collection->min('age');
echo $youngest->name;
Você também pode expressar a propriedade para comparar, fornecendo um caminho ou uma função de retorno de chamada:
$collection = new Collection($people);
$personYoungestChild = $collection->min(function ($person) {
return $person->child->age;
});
$personWithYoungestDad = $collection->min('dad.age');
O mesmo pode ser aplicado à função max()
, que retornará um único elemento
da coleção com o maior valor de propriedade:
$collection = new Collection($people);
$oldest = $collection->max('age');
$personOldestChild = $collection->max(function ($person) {
return $person->child->age;
});
$personWithOldestDad = $collection->max('dad.age');
Finalmente, o método sumOf()
retornará a soma de uma propriedade de todos os elementos:
$collection = new Collection($people);
$sumOfAges = $collection->sumOf('age');
$sumOfChildrenAges = $collection->sumOf(function ($person) {
return $person->child->age;
});
$sumOfDadAges = $collection->sumOf('dad.age');
Calcule o valor médio dos elementos na coleção. Opcionalmente, forneça um caminho correspondente ou função para extrair valores e gerar a média:
$items = [
['invoice' => ['total' => 100]],
['invoice' => ['total' => 200]],
];
// Média: 150
$average = (new Collection($items))->avg('invoice.total');
Calcule o valor mediano de um conjunto de elementos. Opcionalmente, poderá fornecer um caminho correspondente ou função para extrair valores para gerar a mediana:
$items = [
['invoice' => ['total' => 400]],
['invoice' => ['total' => 500]],
['invoice' => ['total' => 100]],
['invoice' => ['total' => 333]],
['invoice' => ['total' => 200]],
];
// Média: 333
$median = (new Collection($items))->median('invoice.total');
Os valores da coleção podem ser agrupados por chaves diferentes em uma nova coleção quando eles compartilham o mesmo valor para uma propriedade:
$students = [
['name' => 'Mark', 'grade' => 9],
['name' => 'Andrew', 'grade' => 10],
['name' => 'Stacy', 'grade' => 10],
['name' => 'Barbara', 'grade' => 9]
];
$collection = new Collection($students);
$studentsByGrade = $collection->groupBy('grade');
// O resultado ficará assim quando convertido em array:
[
10 => [
['name' => 'Andrew', 'grade' => 10],
['name' => 'Stacy', 'grade' => 10]
],
9 => [
['name' => 'Mark', 'grade' => 9],
['name' => 'Barbara', 'grade' => 9]
]
]
Como de costume, é possível fornecer um caminho separado por pontos para propriedades aninhadas ou sua própria função de retorno de chamada para gerar os grupos dinamicamente:
$commentsByUserId = $comments->groupBy('user.id');
$classResults = $students->groupBy(function ($student) {
return $student->grade > 6 ? 'approved' : 'denied';
});
Se você deseja apenas saber o número de ocorrências por grupo, pode fazê-lo usando o
método countBy()
. Ele usa os mesmos argumentos de groupBy
, portanto já
deve ser familiar para você:
$classResults = $students->countBy(function ($student) {
return $student->grade > 6 ? 'approved' : 'denied';
});
// O resultado ficará assim quando convertido em array:
['approved' => 70, 'denied' => 20]
Em certos casos, você sabe que um elemento é exclusivo para a propriedade que
você deseja agrupar. Se você deseja um único resultado por grupo, pode usar a
função indexBy()
:
$usersById = $users->indexBy('id');
// Quando convertido em resultado da matriz, pode parecer
[
1 => 'markstory',
3 => 'jose_zap',
4 => 'jrbasso'
]
Assim como na função groupBy()
, você também pode usar um caminho de propriedade ou um retorno de chamada:
$articlesByAuthorId = $articles->indexBy('author.id');
$filesByHash = $files->indexBy(function ($file) {
return md5($file);
});
Os elementos de diferentes coleções podem ser agrupados usando o método zip()
.
Ele retornará uma nova coleção contendo uma matriz que agrupa os elementos de cada
coleção que são colocados na mesma posição:
$odds = new Collection([1, 3, 5]);
$pairs = new Collection([2, 4, 6]);
$combined = $odds->zip($pairs)->toList(); // [[1, 2], [3, 4], [5, 6]]
Você também pode compactar várias coleções de uma só vez:
$years = new Collection([2013, 2014, 2015, 2016]);
$salaries = [1000, 1500, 2000, 2300];
$increments = [0, 500, 500, 300];
$rows = $years->zip($salaries, $increments)->toList();
// Retorna:
[
[2013, 1000, 0],
[2014, 1500, 500],
[2015, 2000, 500],
[2016, 2300, 300]
]
Como você já pode ver, o método zip()
é muito útil para transpor
matrizes multidimensionais:
$data = [
2014 => ['jan' => 100, 'feb' => 200],
2015 => ['jan' => 300, 'feb' => 500],
2016 => ['jan' => 400, 'feb' => 600],
]
// Reunindo dados de janeiro e fevereiro
$firstYear = new Collection(array_shift($data));
$firstYear->zip($data[0], $data[1])->toList();
// Ou $firstYear->zip(...$data) em PHP >= 5.6
// Retorna
[
[100, 300, 400],
[200, 500, 600]
]
Os valores da coleção podem ser classificados em ordem crescente ou decrescente
com base em uma coluna ou função personalizada. Para criar uma nova coleção
classificada com os valores de outra, você pode usar sortBy
:
$collection = new Collection($people);
$sorted = $collection->sortBy('age');
Como visto acima, você pode classificar passando o nome de uma coluna ou propriedade presente nos valores da coleção. Você também pode especificar um caminho de propriedade usando a notação de ponto. O próximo exemplo classificará os artigos pelo nome do autor:
$collection = new Collection($articles);
$sorted = $collection->sortBy('author.name');
O método sortBy()
é flexível o suficiente para permitir que você
especifique uma função extratora que permitirá selecionar dinamicamente o
valor a ser usado para comparar dois valores diferentes na coleção:
$collection = new Collection($articles);
$sorted = $collection->sortBy(function ($article) {
return $article->author->name . '-' . $article->title;
});
Para especificar em qual direção a coleção deve ser classificada, é
necessário fornecer SORT_ASC
ou SORT_DESC
como o segundo
parâmetro para classificar na direção ascendente ou descendente,
respectivamente. Por padrão, as coleções são classificadas em direção descendente:
$collection = new Collection($people);
$sorted = $collection->sortBy('age', SORT_ASC);
Às vezes, você precisará especificar que tipo de dados você está tentando
comparar para obter resultados consistentes. Para esse fim, você deve fornecer
um terceiro argumento na função sortBy()
com uma das seguintes constantes:
SORT_NUMERIC: Para comparar números
SORT_STRING: Para comparar valores de sequência
SORT_NATURAL: Para classificar sequência contendo números e se você desejar que esses números sejam ordenados de maneira natural. Por exemplo: mostrando «10» depois de «2».
SORT_LOCALE_STRING: Para comparar seqüências de caracteres com base na localidade atual.
Por padrão, SORT_NUMERIC
é usado:
$collection = new Collection($articles);
$sorted = $collection->sortBy('title', SORT_ASC, SORT_NATURAL);
Warning
Muitas vezes, é caro iterar coleções classificadas mais de uma vez. Se você planeja fazer isso,
considere converter a coleção em uma matriz ou simplesmente use o método compile()
nela.
Nem todos os dados devem ser representados de maneira linear. As coleções facilitam
a construção e o nivelamento de estruturas hierárquicas ou aninhadas. Criar uma estrutura
aninhada em que os filhos são agrupados por uma propriedade de identificador pai é fácil
com o método nest()
.
Dois parâmetros são necessários para esta função. O primeiro é a propriedade que representa o identificador do item. O segundo parâmetro é o nome da propriedade que representa o identificador para o item pai:
$collection = new Collection([
['id' => 1, 'parent_id' => null, 'name' => 'Birds'],
['id' => 2, 'parent_id' => 1, 'name' => 'Land Birds'],
['id' => 3, 'parent_id' => 1, 'name' => 'Eagle'],
['id' => 4, 'parent_id' => 1, 'name' => 'Seagull'],
['id' => 5, 'parent_id' => 6, 'name' => 'Clown Fish'],
['id' => 6, 'parent_id' => null, 'name' => 'Fish'],
]);
$collection->nest('id', 'parent_id')->toList();
// Retorna
[
[
'id' => 1,
'parent_id' => null,
'name' => 'Birds',
'children' => [
['id' => 2, 'parent_id' => 1, 'name' => 'Land Birds', 'children' => []],
['id' => 3, 'parent_id' => 1, 'name' => 'Eagle', 'children' => []],
['id' => 4, 'parent_id' => 1, 'name' => 'Seagull', 'children' => []],
]
],
[
'id' => 6,
'parent_id' => null,
'name' => 'Fish',
'children' => [
['id' => 5, 'parent_id' => 6, 'name' => 'Clown Fish', 'children' => []],
]
]
];
Os elementos filhos são aninhados dentro da propriedade children
dentro de
cada um dos itens da coleção. Esse tipo de representação de dados é útil para
renderizar menus ou percorrer elementos até um determinado nível na árvore.
O inverso de nest()
é listNested()
. Este método permite nivelar
uma estrutura de árvore novamente em uma estrutura linear. São necessários dois
parâmetros; o primeiro é o modo de deslocamento (asc, desc ou leaves) e o segundo
é o nome da propriedade que contém os filhos de cada elemento da coleção.
Tomando a entrada da coleção aninhada criada no exemplo anterior, podemos deixar nivelado:
$nested->listNested()->toList();
// Retorna
[
['id' => 1, 'parent_id' => null, 'name' => 'Birds', 'children' => [...]],
['id' => 2, 'parent_id' => 1, 'name' => 'Land Birds'],
['id' => 3, 'parent_id' => 1, 'name' => 'Eagle'],
['id' => 4, 'parent_id' => 1, 'name' => 'Seagull'],
['id' => 6, 'parent_id' => null, 'name' => 'Fish', 'children' => [...]],
['id' => 5, 'parent_id' => 6, 'name' => 'Clown Fish']
]
Por padrão, a árvore é atravessada da raiz para as extremidades. Você também pode instruí-lo a retornar apenas os elementos filhos da árvore:
$nested->listNested()->toList();
// Retorna
[
['id' => 3, 'parent_id' => 1, 'name' => 'Eagle'],
['id' => 4, 'parent_id' => 1, 'name' => 'Seagull'],
['id' => 5, 'parent_id' => 6, 'name' => 'Clown Fish']
]
Depois de converter uma árvore em uma lista aninhada, você pode usar o método
printer()
para configurar como a saída da lista deve ser formatada:
$nested->listNested()->printer('name', 'id', '--')->toArray();
// Retorna
[
3 => 'Eagle',
4 => 'Seagull',
5 -> '--Clown Fish',
]
O método printer()
também permite usar um retorno de chamada para gerar as
chaves e/ou valores:
$nested->listNested()->printer(
function ($el) {
return $el->name;
},
function ($el) {
return $el->id;
}
);
Permite que você veja se uma coleção contém algum elemento:
$collection = new Collection([]);
// Retorna true
$collection->isEmpty();
$collection = new Collection([1]);
// Retorna false
$collection->isEmpty();
As coleções permitem que você verifique rapidamente se elas contêm um
valor específico usando o método contains()
:
$items = ['a' => 1, 'b' => 2, 'c' => 3];
$collection = new Collection($items);
$hasThree = $collection->contains(3);
As comparações são realizadas usando o operador ===
. Se você
deseja fazer tipos de comparação mais flexíveis, pode usar o método some()
.
Às vezes, você pode querer mostrar uma coleção de valores em uma ordem aleatória.
Para criar uma nova coleção que retornará cada valor em uma posição diferente,
use o shuffle
:
$collection = new Collection(['a' => 1, 'b' => 2, 'c' => 3]);
// Isso poderia retornar [2, 3, 1]
$collection->shuffle()->toList();
Ao transpor uma coleção, você obtém uma nova coleção contendo uma linha feita de cada uma das colunas originais:
$items = [
['Products', '2012', '2013', '2014'],
['Product A', '200', '100', '50'],
['Product B', '300', '200', '100'],
['Product C', '400', '300', '200'],
]
$transpose = (new Collection($items))->transpose()->toList();
// Retorna
[
['Products', 'Product A', 'Product B', 'Product C'],
['2012', '200', '300', '400'],
['2013', '100', '200', '300'],
['2014', '50', '100', '200'],
]
Baralhar uma coleção geralmente é útil ao fazer análises estatísticas rápidas.
Outra operação comum ao executar esse tipo de tarefa é retirar alguns valores
aleatórios de uma coleção, para que mais testes possam ser realizados. Por exemplo,
se você quiser selecionar 5 usuários aleatórios aos quais deseja aplicar alguns
testes A/B, poderá usar a função sample()
:
$collection = new Collection($people);
// Retire no máximo 20 usuários aleatórios desta coleção
$testSubjects = $collection->sample(20);
sample()
terá no máximo o número de valores que você especificar no primeiro
argumento. Se não houver elementos suficientes na coleção para satisfazer a amostra,
a coleção completa em uma ordem aleatória será retornada.
Sempre que você desejar obter uma fatia de uma coleção, use a função take()
,
ela criará uma nova coleção com, no máximo, o número de valores que você especificar
no primeiro argumento, iniciando na posição passada no segundo argumento:
$topFive = $collection->sortBy('age')->take(5);
// Leve 5 pessoas da coleção a partir da posição 4
$nextTopFive = $collection->sortBy('age')->take(5, 4);
As posições são baseadas em zero, portanto, o número da primeira posição é 0
.
Embora o segundo argumento de take()
possa ajudá-lo a pular alguns
elementos antes de obtê-los da coleção, você também pode usar skip()
para o mesmo objetivo que uma maneira de tirar o resto dos elementos depois
de uma certa posição:
$collection = new Collection([1, 2, 3, 4]);
$allExceptFirstTwo = $collection->skip(2)->toList(); // [3, 4]
Um dos usos mais comuns de take()
é obter o primeiro elemento da
coleção. Um método de atalho para alcançar o mesmo objetivo é usar o
método first()
:
$collection = new Collection([5, 4, 3, 2]);
$collection->first(); // Returns 5
Da mesma forma, você pode obter o último elemento de uma coleção usando o
método last()
:
$collection = new Collection([5, 4, 3, 2]);
$collection->last(); // Retorna 2
Você pode compor várias coleções em uma única. Isso permite coletar dados de
várias fontes, concatená-los e aplicar outras funções de coleta de maneira
muito suave. O método append()
retornará uma nova coleção contendo os
valores das duas fontes:
$cakephpTweets = new Collection($tweets);
$myTimeline = $cakephpTweets->append($phpTweets);
// Tweets contendo cakefest de ambas as fontes
$myTimeline->filter(function ($tweet) {
return strpos($tweet, 'cakefest');
});
Permite anexar um item com uma chave opcional à coleção. Se você especificar uma chave que já existe na coleção, o valor não será substituído:
$cakephpTweets = new Collection($tweets);
$myTimeline = $cakephpTweets->appendItem($newTweet, 99);
O método prepend()
retornará uma nova coleção contendo os valores das duas fontes:
$cakephpTweets = new Collection($tweets);
$myTimeline = $cakephpTweets->prepend($phpTweets);
Permite anexar um item com uma chave opcional à coleção. Se você especificar uma chave que já existe na coleção, o valor não será substituído:
$cakephpTweets = new Collection($tweets);
$myTimeline = $cakephpTweets->prependItem($newTweet, 99);
Warning
Ao anexar de fontes diferentes, você pode esperar que algumas chaves de
ambas as coleções sejam iguais. Por exemplo, ao anexar duas matrizes simples.
Isso pode apresentar um problema ao converter uma coleção em uma matriz usando
toArray()
. Se você não deseja que os valores de uma coleção substituam os
outros na coleção anterior com base em sua chave, certifique-se de chamar
toList()
para soltar as chaves e preservar todos os valores.
Às vezes, você pode ter dois conjuntos de dados separados que gostaria de inserir os elementos de um conjunto em cada um dos elementos do outro conjunto. Este é um caso muito comum quando você busca dados de uma fonte que não oferece suporte à mesclagem de dados ou se une nativamente.
As coleções oferecem um método insert()
que permitirá inserir cada um dos
elementos em uma coleção em uma propriedade dentro de cada um dos elementos de
outra coleção:
$users = [
['username' => 'mark'],
['username' => 'juan'],
['username' => 'jose']
];
$languages = [
['PHP', 'Python', 'Ruby'],
['Bash', 'PHP', 'Javascript'],
['Javascript', 'Prolog']
];
$merged = (new Collection($users))->insert('skills', $languages);
Quando convertida em uma matriz, a coleção $merged
ficará assim:
[
['username' => 'mark', 'skills' => ['PHP', 'Python', 'Ruby']],
['username' => 'juan', 'skills' => ['Bash', 'PHP', 'Javascript']],
['username' => 'jose', 'skills' => ['Javascript', 'Prolog']]
];
O primeiro parâmetro para o método insert()
é um caminho de propriedades
separado por pontos a seguir para que os elementos possam ser inseridos nessa
posição. O segundo argumento é qualquer coisa que possa ser convertida em um
objeto de coleção.
Observe que os elementos são inseridos pela posição em que foram encontrados, portanto, o primeiro elemento da segunda coleção é mesclado no primeiro elemento da primeira coleção.
Se não houver elementos suficientes na segunda coleção para inserir na primeira,
a propriedade target será preenchida com valores null
:
$languages = [
['PHP', 'Python', 'Ruby'],
['Bash', 'PHP', 'Javascript']
];
$merged = (new Collection($users))->insert('skills', $languages);
// Irá fornecer
[
['username' => 'mark', 'skills' => ['PHP', 'Python', 'Ruby']],
['username' => 'juan', 'skills' => ['Bash', 'PHP', 'Javascript']],
['username' => 'jose', 'skills' => null]
];
O método insert()
pode operar elementos ou objetos da matriz
implementando a interface ArrayAccess
.
Usar closures para métodos de coleta é ótimo quando o trabalho a ser feito é pequeno e focado, mas pode ficar confuso muito rapidamente. Isso se torna mais óbvio quando muitos métodos diferentes precisam ser chamados ou quando o comprimento dos métodos de closures é superior a apenas algumas linhas.
Também existem casos em que a lógica usada para os métodos de coleta pode ser reutilizada em várias partes do seu aplicativo. É recomendável considerar a extração de lógica de coleção complexa para separar classes. Por exemplo, imagine uma closure longa como esta:
$collection
->map(function ($row, $key) {
if (!empty($row['items'])) {
$row['total'] = collection($row['items'])->sumOf('price');
}
if (!empty($row['total'])) {
$row['tax_amount'] = $row['total'] * 0.25;
}
// Mais código aqui...
return $modifiedRow;
});
Isso pode ser refatorado criando outra classe:
class TotalOrderCalculator
{
public function __invoke($row, $key)
{
if (!empty($row['items'])) {
$row['total'] = collection($row['items'])->sumOf('price');
}
if (!empty($row['total'])) {
$row['tax_amount'] = $row['total'] * 0.25;
}
// Mais código aqui...
return $modifiedRow;
}
}
// Usa a lógica em sua chamada de map()
$collection->map(new TotalOrderCalculator)
Às vezes, uma cadeia de chamadas de método de coleção pode se tornar reutilizável
em outras partes do seu aplicativo, mas apenas se elas forem chamadas nessa ordem
específica. Nesses casos, você pode usar through()
em combinação com uma
classe implementando __invoke
para distribuir suas chamadas úteis de
processamento de dados:
$collection
->map(new ShippingCostCalculator)
->map(new TotalOrderCalculator)
->map(new GiftCardPriceReducer)
->buffered()
...
As chamadas de método acima podem ser extraídas para uma nova classe, para que não precisem ser repetidas sempre:
class FinalCheckOutRowProcessor
{
public function __invoke($collection)
{
return $collection
->map(new ShippingCostCalculator)
->map(new TotalOrderCalculator)
->map(new GiftCardPriceReducer)
->buffered()
...
}
}
// Agora você pode usar o método through() para chamar todos os métodos de uma só vez
$collection->through(new FinalCheckOutRowProcessor);
As coleções geralmente executam a maioria das operações que você cria usando suas funções de maneira lenta. Isso significa que, embora você possa chamar uma função, isso não significa que ela seja executada imediatamente. Isso é verdade para muitas funções nesta classe. A avaliação lenta permite economizar recursos em situações em que você não usa todos os valores em uma coleção. Você não pode usar todos os valores quando a iteração parar mais cedo ou quando um caso de exceção/falha for atingido mais cedo.
Além disso, a avaliação lenta ajuda a acelerar algumas operações. Considere o seguinte exemplo:
$collection = new Collection($oneMillionItems);
$collection = $collection->map(function ($item) {
return $item * 2;
});
$itemsToShow = $collection->take(30);
Se as coleções não tivessem sido preguiçosas, teríamos executado um milhão
de operações, embora desejássemos mostrar apenas 30 elementos. Em vez disso,
nossa operação de mapa foi aplicada apenas aos 30 elementos que usamos. Também
podemos obter benefícios dessa avaliação preguiçosa para coleções menores quando
fazemos mais de uma operação nelas. Por exemplo: chamando map()
duas vezes
e depois filter()
.
A avaliação preguiçosa também traz sua desvantagem. Você pode estar executando as mesmas operações mais de uma vez se otimizar uma coleção prematuramente. Considere este exemplo:
$ages = $collection->extract('age');
$youngerThan30 = $ages->filter(function ($item) {
return $item < 30;
});
$olderThan30 = $ages->filter(function ($item) {
return $item > 30;
});
Se iterarmos youngerThan30
e olderThan30
, infelizmente a coleção
executaria a operação extract()
duas vezes. Isso ocorre porque as coleções
são imutáveis e a operação de extração lenta é feita para os dois filtros.
Felizmente, podemos superar esse problema com uma única função. Se você planeja
reutilizar os valores de determinadas operações mais de uma vez, é possível
compilar os resultados em outra coleção usando a função buffered()
:
$ages = $collection->extract('age')->buffered();
$youngerThan30 = ...
$olderThan30 = ...
Agora, quando as duas coleções forem iteradas, elas chamarão a operação de extração apenas uma vez.
Às vezes, você precisa obter um clone dos elementos de outra coleção. Isso é
útil quando você precisa repetir o mesmo conjunto de locais diferentes ao mesmo
tempo. Para clonar uma coleção de outra, use o método compile()
:
$ages = $collection->extract('age')->compile();
foreach ($ages as $age) {
foreach ($collection as $element) {
echo h($element->name) . ' - ' . $age;
}
}