Список понимания плюс сопоставление с образцом

Проблема, которую я пытаюсь решить, заключается в следующем: "найти все элементы списка L, за которым следует элемент со значением X".

Я пытался сделать это, используя понимание списка. Однако сейчас я сомневаюсь, что это возможно. Я понимаю, как все должно работать в более простом сценарии, как этот

[some_transformation(X) || X <- [...], some_conditional(X)].

Есть фильтр X, который исходит из генератора X, и преобразование, применяемое к тем, которые проходят.

У меня очень слабая надежда, потому что сила сопоставления с образцом в Erlang позволяет такие вещи:

[H1|[H2|T]] = [1,2,3,4].

, который связывает H1 и H2. Есть ли какой-то шаблон, который я могу использовать в понимании списка для достижения желаемого результата?

Когда я наивно пробую это, это, очевидно, не работает:

78> [X||[H1|[H2|_]]=X <- [1,2,3,4]].
[]

Возможно, мне следует перестать быть ленивым и хотеть, чтобы магия понимания списков и сопоставления с образцом сделала всю работу за меня? = Р

1 ответ

Решение

Вы можете сжать список с последним элементом, отброшенным хвостом списка, а затем использовать его понимание:

1> List = [1, 2, 3, 2, 1, 2, 2, 1, 1, 3, 1].
[1,2,3,2,1,2,2,1,1,3,1]
2> X = 1.
1
3> Zipped = lists:zip(lists:droplast(List), tl(List)).
[{1,2},
 {2,3},
 {3,2},
 {2,1},
 {1,2},
 {2,2},
 {2,1},
 {1,1},
 {1,3},
 {3,1}]
4> [A || {A, B} <- Zipped, B == X].
[2,2,1,3]

Это неэффективно, хотя - создает копию списка (для droplast) и новый заархивированный список. Вы можете использовать рекурсию вот так, чтобы сделать это эффективным:

-module(a).
-export([before/2]).

before(List, X) ->
  before(List, X, []).

before([A, X | Tail], X, Acc) ->
  before([X | Tail], X, [A | Acc]);
before([_ | Tail], X, Acc) ->
  before(Tail, X, Acc);
before([], _, Acc) ->
  lists:reverse(Acc).
1> c(a).
{ok,a}
2> a:before([1, 2, 3, 2, 1, 2, 2, 1, 1, 3, 1], 1).
[2,2,1,3]
Другие вопросы по тегам