A definição de macro alterada volta ao original

A definição de macro alterada volta ao original

Eu tenho dois comandos \testonee \testtwo. \testtwoaparece apenas se houve um \testoneantes relevante e leva os mesmos argumentos. Portanto, quero \testonearmazenar seus argumentos \@repeat@meonde \testtwoos seleciona novamente, para não precisar digitá-los duas vezes.

No entanto, isso funciona apenas se ambos estiverem no mesmo ambiente. Mas em alguns casos decide que o valor alterado de \@repeat@medeve ser ignorado.

\documentclass{article}
\makeatletter

\def\@repeat@me{}

\newcommand\testtwo{\bf{\@repeat@me}% print what is currently defined.
% reset the definition so it does not interfere with next call
\def\@repeat@me{}}

% print it and then remember it.
\newcommand\testone[1]{\def\@repeat@me{#1}\emph{\@repeat@me}}%

\makeatother
\begin{document}

% prints: \emph{Hello World} \bf{Hello World}
\testone{Hello World} \testtwo{}

% prints: \bf{\emph{Hello World}}
% intended: \bf{\emph{Hello World}} \bf{Hello World}
\bf{\testone{Hello World}} \testtwo{}

\end{document}

Responder1

Se você deseja uma atribuição global (para escapar de grupos e ambientes, você precisa) \global\def(ou \gdef):

\documentclass{scrartcl}

\makeatletter
\newcommand*\@repeat@me{}
\newcommand*\testtwo{\textbf{\@repeat@me}\gdef\@repeat@me{}}
\newcommand\testone[1]{\gdef\@repeat@me{#1}\emph{\@repeat@me}}
\makeatother

\begin{document}

\testone{Hello World} \testtwo
\textbf{\testone{Hello World}} \testtwo

\end{document}

Responder2

Observe que o TeX impõe um escopo dentro do qual (re)definições podem existir e ser revertidas externamente. É isso que está acontecendo aqui. Vejamos as expansões:

\testone{Hello World} \testtwo{}

expande para (com comentários)

\def\@repeat@me{Hello World}\emph{\@repeat@me} % \testone{Hello World}
\textbf{\@repeat@me}\def\@repeat@me{}% \testtwo{}

que se expande para

\emph{Hello World} \textbf{Hello World}% Using the new definition of \@repeat@me

insira a descrição da imagem aqui

Agora observe as expansões do segundo conjunto de macros:

\textbf{\testone{Hello World}} \testtwo{}

expande para (com comentários)

\textbf{% <--- start of a group/scope
  def\@repeat@me{Hello World}\emph{\@repeat@me}% \testone{Hello World}
} % <--- end of a group/scope
\textbf{\@repeat@me}\def\@repeat@me{}% \testtwo{}

que se expande para

\textbf{\emph{Hello World}} % At scope-end, \@repeat@me reverts to its original definition
\textbf{}% Since \@repeat@me is empty {}

insira a descrição da imagem aqui

Usar uma definição global como \gdeffaz com que a redefinição de \@repeat@mesobreviva além do escopo fornecido por \textbf{..}.


insira a descrição da imagem aqui

\documentclass{article}
\makeatletter

\def\@repeat@me{}

% print it and then remember it.
\newcommand\testone[1]{\gdef\@repeat@me{#1}\emph{\@repeat@me}}%

\newcommand\testtwo{\textbf{\@repeat@me}% print what is currently defined.
  % reset the definition so it does not interfere with next call
  \def\@repeat@me{}}

\makeatother
\begin{document}

% prints: \emph{Hello World} \textbf{Hello World}
\testone{Hello World} \testtwo{}

% prints: \textbf{\emph{Hello World}} \textbf{Hello World}
\textbf{\testone{Hello World}} \testtwo{}

\end{document}

informação relacionada