У меня есть эта специальная функция для извлечения частей списка в форме: Give[list, elem]
возвращает часть списка , которая соответствует позиции элемент в глобальном $Reference
переменная (если определена). Я активно использую эту функцию в своем коде, поэтому я решил оптимизировать ее. Вот где мне удалось продвинуться так далеко, но, честно говоря, я понятия не имею, как продвигаться.
ClearAll[Give, $Reference, set];
Give::noref = "No, non-list or empty $Reference was defined to refer to by Give.";
Give::noelem = "Element (or some of the elements in) `1` is is not part of the reference set `2`.";
Give::nodepth = "Give cannot return all the elements corresponding to `1` as the list only has depth `2`.";
give[list_, elem_List, ref_] := Flatten[Pick[list, ref, #] & /@ elem, 1];
give[list_, elem_, ref_] := First@Pick[list, ref, elem];
Options[Give] = {Reference :> $Reference}; (* RuleDelayed is necessary, for it is possible that $Reference changes between two subsequent Give calls, and without delaying its assignment, ref would use previous value of $Reference instead of actual one. *)
Give[list_List, elem___, opts___?OptionQ] := Module[{ref, pos},
ref = Reference /. {opts} /. Options@Give;
Which[
Or[ref === {}, Head@ref =!= List], Message[Give::noref]; {},
Complement[Union@Flatten@{elem}, ref] =!= {}, Message[Give::noelem, elem, ref]; {},
Length@{elem} > Depth@list - 1, Message[Give::nodepth, {elem}, Depth@list]; {},
True, Fold[give[#1, #2, ref] &, list, {elem}]
]];
In[106]:= $Reference = {"A", "B", "C"};
set = {{1, 2, 3}, {4, 5, 6}, {7, 8, 9}};
Give[set, "B"](* return specified row *)
Out[108]= {4, 5, 6}
In[109]:= Give[set, "B", "A"] (* return entry at specified row & column *)
Out[109]= 4
In[110]:= Give[set, {"B", "A"}] (* return multiple rows *)
Out[110]= {{4, 5, 6}, {1, 2, 3}}
Я решил отбросить отдельные вызовы функций подписи, поскольку версия списка может вызывать не-версию списка, что означает, что обработка ошибок должна выполняться несколько раз (для каждого элемента в списке). К сожалению, обработка ошибок не может быть отброшена. Если улучшенная версия более устойчива (например, может обрабатывать больше измерений), это не проблема, однако приведенных выше примеров будет достаточно.
In[139]:= First@Timing[Give[set, RandomChoice[$Reference, 10000]]] (* 1D test *)
Out[139]= 0.031
In[138]:= First@Timing[Table[Give[set, Sequence @@ RandomChoice[$Reference, 2]], {10000}]] (* 2d test *)
Out[138]= 0.499
Я уверен, что это неэффективный код, поэтому не стесняйтесь его улучшать. Любая помощь приветствуется, даже если она обрезается всего на несколько наносекунд.