迭代列表並一次修改一個元素以建構新列表

迭代列表並一次修改一個元素以建構新列表

長期讀者,第一次海報。

我有一個關於創建循環訪問列表的巨集的問題。我已經閱讀了其他用戶發布的一些問題,但所有這些問題都與以複雜的方式使用 LaTeX 有關。我的問題是關於使用 plain TeX 迭代列表並透過修改舊列表中的每個元素來建構新列表。我目前正在自學 TeX,我認為編寫一些基本但多功能且強大的巨集將幫助我更好地了解 TeX 的內部結構是如何運作的。無論如何,足夠的背景知識來回答我的問題。

到目前為止,這是我的程式碼:

我遞歸地定義了迭代器。它一次讀取一個參數,確定該參數是逗號還是停止標記(我自己的定義,稱為 \myStop),如果是逗號則繼續,如果是停止標記則停止,否則將元素替換為本身和字串(或標記列表)“+ 1”。

\edef\myStop{)}
\def\Iterator#1{
        \ifx\myStop#1 %do nothing {end of list}
            \else %
                \ifx,#1 \expandafter\IteratorIntermediateOne % currently 2 if-statements deep
                    \else #1 + 1, \expandafter\IteratorIntermediateOne %currently 2 if-statements deep
            \fi%
        \fi%
}

我必須定義一個中間迭代器來呼叫IteratorIntermediateOne來容納命令,\expandafter\Iterator因為我目前不知道如何\expandafter以相當於\expandafter{\expandafter\Iterator}\fi\fi.所以我想這是我的第一個問題:有沒有辦法定義巢狀\expandafter指令?

現在一切都在上下文中,這是我的所有程式碼:

\edef\MyList{1,2,3}
\edef\myStop{)}
\def\IteratorIntermediateOne{\expandafter\Iterator}
\def\Iterator#1{%
        \ifx\myStop#1 %do nothing {end of list}
            \else %
                \ifx,#1 \expandafter\IteratorIntermediateOne % currently 2 if-statements deep
                    \else #1 + 1, \expandafter\IteratorIntermediateOne %currently 2 if-statements deep
            \fi%
        \fi%
}

\edef\newList{\expandafter\Iterator\MyList\myStop}

免責聲明:我知道最後一個元素後面有額外的逗號。我也知道沒有足夠的案例來檢測構造不良的清單。我對 TeX 的了解還不夠,甚至無法開始想像如何保護巨集免受不正確構造的標記清單的影響,所以如果你們中更有經驗的人忍不住嘲笑這段程式碼,我深表歉意。

好的,我的另一個問題是:是否有更有效的方法來定義呼叫自身的巨集?單獨定義迭代器巨集並使用 TeX 的內建\loop指令呼叫它是更好的做法嗎?如果是這樣,有人可以指導我如何做到這一點,因為我無法在 TeX 的眼睛、嘴巴、食道和胃過程的上下文中理解循環調用。循環是否擴展宏,將\repeat其傳遞到胃並返回到宏?我在任何地方都找不到好的解釋。

感謝你的幫助!

答案1

目的似乎是為每個項目添加 1 我會更像這樣編碼(假設 etex)

\edef\MyList{1,2,3,25,456,2}

\def\Iterator#1{\expandafter\xiterator#1\stopiteration,}

\def\xiterator#1,{\the\numexpr#1+1\relax,\xiterator}
\def\stopiteration#1\relax#2\xiterator{#1\relax}


\message{\Iterator\MyList}

\bye

這使得訊息

2,3,4,26,457,3 

答案2

\input listofitems
\def\MyList{1,2,3,25,456,2}
\def\processlist#1{%
  \readlist\myterms\MyList
  \foreachitem\z\in\myterms{%
    \ifnum\zcnt=1\else,\fi
    \the\numexpr\z+1\relax
  }%
}

\processlist\Mylist
\bye

在此輸入影像描述

如果你確實需要節省更新後的列表,我們可以在\mytoks令牌列表中這樣做:

\input listofitems
\newtoks\mytoks
\def\MyList{1,2,3,25,456,2}
\def\processlist#1{%
  \mytoks{}%
  \readlist\myterms\MyList
  \foreachitem\z\in\myterms{%
    \ifnum\zcnt=1\else\mytoks\expandafter{\the\mytoks,}\fi
    \mytoks\expandafter\expandafter\expandafter{%
      \expandafter\the\expandafter\mytoks\the\numexpr\z+1\relax}
  }%
}

\processlist\Mylist

List is \the\mytoks
\bye

在此輸入影像描述

答案3

你說:

……如果更有經驗的人忍不住嘲笑這段程式碼,我深感抱歉。

當我開始學習 TeX 時,我感覺這是一個非常陡峭的學習曲線。

我時不時感到沮喪。 ;-)

我不認為像您這樣正在走這條學習曲線的人會處於嘲笑他們在巨集程式設計/TeX 程式設計方面的嘗試的情況。

我認為任何試圖以一種既基於理性又基於本身的方式實現或學習好的東西的嘗試並沒有不當行為,基本上值得尊重。

如果您對下面範例中的程式碼如何運作有疑問,請隨時詢問他們。然後,描述您認為程式碼如何運作以及您在理解上遇到困難的地方會很有用。根據我的經驗,這使得受訪者更容易準確地找出哪些資訊(例如,關於 TeX 原語如何運作以及 TeXbook 後面章節中簡要暗示的哪些「副作用」用於程式設計技巧)仍然存在。


假設逗號清單中的條目沒有被空格包圍,並且 -primitive\relax沒有出現在逗號清單中,並且\numexprε-TeX-extensions 中的條目可用,您可能可以執行以下操作:

\long\def\gobble#1{}%
\long\def\firstofone#1{#1}%

\def\Iterator#1#2,{%
  % #1 - element-separator to prepend; empty in the 1st iteration;
  %      comma in consecutive iterations
  % #2 - either current element of old list or the \relax that was
  %      appended for denoting the end of the list
  \ifx\relax#2\expandafter\gobble\else\expandafter\firstofone\fi
  {%
    #1\number\numexpr#2+1\relax\Iterator{,}%
  }%
}%

\def\MyList{1,2,3}

\edef\newList{\expandafter\Iterator\expandafter{\expandafter}\MyList,\relax,}

\begingroup

\tt

\string\MyList: \meaning\MyList

\string\newList: \meaning\newList

\endgroup

\bye

這個例子的重點是:

在TeX-primitive\edef的 -definition-text中用於擴展.也附加了序列。\newList\expandafter\MyList,\relax,

\newList \edef這樣,在定義驅動擴展定義文字時,\newList在某個階段會產生一個序列\Iterator{}Comma,sparated,items,from,\MyList,\relax,

因此\relax,標誌著清單的結束。

\edef現在,仍然由-expansion驅動 - \Iterator(遞歸地)選擇一個非分隔參數#1(在第一次迭代中為空,在連續迭代中包含一個逗號,即,它包含分隔符以添加到新列表的項目之前)和一個逗號分隔參數#2,它要么保存來自\myList逗號列表的下一個項目,要么保存結束標記\relax,並且在任何情況下都放置(嵌套在大括號中)表示下一次迭代的標記序列,由

  • undelimited-argument #1,即必須位於新列表的下一項之前的分隔符,
  • \number\numexpr#2+1\relax將逗號分隔參數表示的值加 1 的表達式,這樣形成新列表的下一項,
  • 呼叫自己來處理 擴充中剩餘的下一個項目\myList,這次在無分隔參數內提供一個逗號,表示下次新清單的下一個項目前面應有一個逗號。

透過\ifx\relax#2檢查是否到達逗號清單的末端/\relax附加到-expansion開頭的清單。\edef如果是這樣,則嵌套在大括號中表示下一次迭代的標記序列將通過“吞噬/刪除”,\gobble因此不會執行,從而終止迭代/尾遞歸。如果不是這樣,則透過應用\firstofone此後處理該序列來從該序列中刪除周圍的花括號。

#1的無定界參數\Iterator,它保存要新增到新清單的項目前面的分隔符,僅在第一次迭代中為空。在每個連續迭代中,它保存一個逗號,在該連續迭代中,前一個迭代由巨集本身提供,\Iterator作為令牌序列的一部分,然後形成下一個迭代。這樣(僅)新清單的第一項前面沒有逗號。


如果您沒有\numexpr可用的 ε-TeX-extensions,我可以提供一個用於遞增非負整數的程式。 (在“現實生活”中,您可能對這些包感興趣積分計算大整數計算.)

%------------------------------------------------------------------------------
% Expandable incrementing of non-negative integer number formed by a sequence
% of explicit catcode-12-character-tokens from the set {0,1,2,3,4,5,6,7,8,9}
%..............................................................................
% \Increment{<non-negative integer number k as sequence of explicit
%              catcode-12-character-tokens from the set 0123456789>}
% ->
% <natural number (k+1) as sequence of explicit catcode-12-character-tokens
%  from the set 0123456789>
% In expansion-contexts the result is delivered after two expansion-steps/is 
% obtained by "hitting" \Increment with \expandafter twice.
%------------------------------------------------------------------------------
\def\Increment#1{%
  \romannumeral0%
  \IncrementReverse{\IncrementFork{}}{\relax}{}#1\relax
}%
\def\IncrementReverse#1#2#3#4{%
  % #1 - tokens to prepend to reversed list
  % #2 - tokens to append to reversed list
  % #3 - reversed list constructed so far
  % #4 - current element of not-reversed list
  \ifx\relax#4%
    \expandafter\firstoftwo
  \else
    \expandafter\secondoftwo
  \fi
  {#1#3#2}{\IncrementReverse{#1}{#2}{#4#3}}%
}%
\def\IncrementSelect#10123456789\relax#2#3!!{#2}%
\def\IncrementFork#1#2{%
  % #1 - digits incremented so far
  % #2 - current digit to increment or end-marker \relax
  \IncrementSelect
  #2123456789\relax{\IncrementReverse{ }{}{}#11}%
  0#223456789\relax{\IncrementReverse{ }{}{}#12}%
  01#23456789\relax{\IncrementReverse{ }{}{}#13}%
  012#2456789\relax{\IncrementReverse{ }{}{}#14}%
  0123#256789\relax{\IncrementReverse{ }{}{}#15}%
  01234#26789\relax{\IncrementReverse{ }{}{}#16}%
  012345#2789\relax{\IncrementReverse{ }{}{}#17}%
  0123456#289\relax{\IncrementReverse{ }{}{}#18}%
  01234567#29\relax{\IncrementReverse{ }{}{}#19}%
  012345678#2\relax{\IncrementFork{#10}}%
  0123456789#2{\IncrementReverse{ }{}{}#11\relax}%
  0123456789\relax{\IncrementReverse{ }{}{}#11#2}%
  !!%
}%
%%-----------------------------------------------------------------------------
\long\def\firstoftwo#1#2{#1}%
\long\def\secondoftwo#1#2{#2}%

\def\Iterator#1#2,{%
  % #1 - element-separator to prepend
  % #2 - current element of old list
  \ifx\relax#2\expandafter\firstoftwo\else\expandafter\secondoftwo\fi
  {}{%
    #1\Increment{#2}\Iterator{,}%
  }%
}%

\def\MyList{1,2,3}

\edef\newList{\expandafter\Iterator\expandafter{\expandafter}\MyList,\relax,}

\begingroup

\tt

\string\MyList: \meaning\MyList

\string\newList: \meaning\newList

\endgroup

\bye

如果您希望一個沒有 的例程\edef,您可以使用\romannumeral0-expansion- 和參數交換技術 - -expansion 的要點\romannumeral0是:

  • TeX 擴充可擴充令牌,同時收集屬於⟨數字⟩- 以羅馬數字表示的數量。
  • 如果 TeX 在收集時找到的第一個標記⟨數字⟩-quantity 是一個數字,例如 ,0則收集屬於該數量的代幣的過程⟨數字⟩- 數量變成收集更多數字或非數字的東西的過程,因此終止收集過程。可擴展代幣在收集數字時會擴展。終止數字序列的空間令牌會終止收集更多數字的過程並被默默丟棄。
  • 如果收集到的數字不是正數,TeX 將默默吞下形成⟨數字⟩- 數量,無需交付任何代幣作為回報。

這意味著\romannumeral只要確保最終找到一個非正數,就可以用來欺騙 TeX 進行大量擴展和參數交換工作。

%------------------------------------------------------------------------------
% Expandable incrementing of non-negative integer number formed by a sequence
% of explicit catcode-12-character-tokens from the set {0,1,2,3,4,5,6,7,8,9}
%..............................................................................
% \Increment{<non-negative integer number k as sequence of explicit
%              catcode-12-character-tokens from the set 0123456789>}
% ->
% <natural number (k+1) as sequence of explicit catcode-12-character-tokens
%  from the set 0123456789>
% In expansion-contexts the result is delivered after two expansion-steps/is 
% obtained by "hitting" \Increment with \expandafter twice.
%------------------------------------------------------------------------------
\def\Increment#1{%
  \romannumeral0%
  \IncrementReverse{\IncrementFork{}}{\relax}{}#1\relax
}%
\def\IncrementReverse#1#2#3#4{%
  % #1 - tokens to prepend to reversed list
  % #2 - tokens to append to reversed list
  % #3 - reversed list constructed so far
  % #4 - current element of not-reversed list
  \ifx\relax#4%
    \expandafter\firstoftwo
  \else
    \expandafter\secondoftwo
  \fi
  {#1#3#2}{\IncrementReverse{#1}{#2}{#4#3}}%
}%
\def\IncrementSelect#10123456789\relax#2#3!!{#2}%
\def\IncrementFork#1#2{%
  % #1 - digits incremented so far
  % #2 - current digit to increment or end-marker \relax
  \IncrementSelect
  #2123456789\relax{\IncrementReverse{ }{}{}#11}%
  0#223456789\relax{\IncrementReverse{ }{}{}#12}%
  01#23456789\relax{\IncrementReverse{ }{}{}#13}%
  012#2456789\relax{\IncrementReverse{ }{}{}#14}%
  0123#256789\relax{\IncrementReverse{ }{}{}#15}%
  01234#26789\relax{\IncrementReverse{ }{}{}#16}%
  012345#2789\relax{\IncrementReverse{ }{}{}#17}%
  0123456#289\relax{\IncrementReverse{ }{}{}#18}%
  01234567#29\relax{\IncrementReverse{ }{}{}#19}%
  012345678#2\relax{\IncrementFork{#10}}%
  0123456789#2{\IncrementReverse{ }{}{}#11\relax}%
  0123456789\relax{\IncrementReverse{ }{}{}#11#2}%
  !!%
}%
%%-----------------------------------------------------------------------------

\long\def\firstoftwo#1#2{#1}%
\long\def\secondoftwo#1#2{#2}%
\long\def\exchange#1#2{#2#1}%

\def\Iterator#1,#2\relax#3#4{%
  % #1 - current element of old list
  % #2 - remaining elements of old list
  % #3 - element-separator to prepend
  % #4 - new list constructed so far
  \ifx\relax#1\expandafter\firstoftwo\else\expandafter\secondoftwo\fi
  { #4}{%
    \expandafter\exchange
    \expandafter{%
    \expandafter{%
    \romannumeral0%
    \expandafter\expandafter\expandafter\exchange
    \expandafter\expandafter\expandafter{%
    \Increment{#1}}{ #4#3}}}{\Iterator#2\relax{,}}%
  }%
}%

\def\MyList{0,1,2,3}

\expandafter\def
\expandafter\newList
\expandafter{%
\romannumeral0\expandafter\Iterator\MyList,{\relax},\relax{}{}}%

\begingroup

\tt

\string\MyList: \meaning\MyList

\string\newList: \meaning\newList

\endgroup

\bye

答案4

既然你是新手,你可以從 開始expl3

\documentclass{article}
\usepackage{xparse,xfp}

\ExplSyntaxOn

\NewDocumentCommand{\generatelist}{mmm}
 {% #1=output, #2=input, #3=iterator
  \harry_list_generate:nnn { #1 } { #2 } { #3 }
 }

% variables
\clist_new:N \l__harry_list_input_clist
\clist_new:N \l__harry_list_output_clist

% the main function
\cs_new_protected:Nn \harry_list_generate:nnn
 {
  % if the input is a single token, assume it is a control sequence
  \tl_if_single:nTF { #2 }
   { \clist_set_eq:NN \l__harry_list_input_clist #2 }
   { \clist_set:Nn \l__harry_list_input_clist { #2 } }
  % now \l__harry_list_input_clist contains the input

  % clear the output list
  \clist_clear:N \l__harry_list_output_clist

  % map the input list applying the iterator to each item
  \clist_map_inline:Nn \l__harry_list_input_clist
   {
    \clist_put_right:Nx \l__harry_list_output_clist { #3 { ##1 } }
   }

  % make the output list
  \clist_set_eq:NN #1 \l__harry_list_output_clist
 }

\ExplSyntaxOff

% two example iterators
\newcommand{\addone}[1]{\inteval{#1+1}}
\newcommand{\addhyphens}[1]{-#1-}

% a control sequence expanding to a list
\newcommand{\List}{1,2,3,41}

\generatelist{\ListA}{\List}{\addone}

\generatelist{\ListB}{1,2,3}{\addhyphens}

\show\ListA
\show\ListB

這輸出

> \ListA=macro:
->2,3,4,42.
l.50 \show\ListA

?
> \ListB=macro:
->-1-,-2-,-3-.
l.51 \show\ListB

相關內容