Como adicionar um \global antes de \pgfmathsetmacro?

Como adicionar um \global antes de \pgfmathsetmacro?

Parece que não se pode adicionar \globalantes de a, \pgfmathsetmacropois isso resulta:

Você não pode usar um prefixo com `\begingroup'.

Mas essa é a funcionalidade que preciso aqui.

Estou tentando calcular a média dos números inteiros em uma lista. No MWE a lista que tenho é

\newcommand*{\MyData}{3,7,X,Y,8,9,Z}%

e as \typeoutmensagens mostram:

**** NumberOfValidEnteries=1, \AverageValue=3.0
**** NumberOfValidEnteries=2, \AverageValue=3.5
**** NumberOfValidEnteries=3, \AverageValue=2.66666
**** NumberOfValidEnteries=4, \AverageValue=2.25

Portanto, parece que o \pgfmathsetmacroestá funcionando, exceto que não está salvando seu estado entre iterações sucessivas do \foreach, que é exatamente para o que o the \globalestava lá em primeiro lugar.

Código:

\documentclass{article}
\usepackage{tikz}
\usepackage{xstring}

% http://tex.stackexchange.com/questions/50111/how-to-check-if-the-value-of-a-parameter-is-a-number
\newcommand*{\IfIsInteger}[3]{%
    \IfStrEq{#1}{ }{%
        #3% is a blank string
    }{%
        \IfInteger{#1}{#2}{#3}%
    }%
}%

\newcommand*{\MyData}{3,7,X,Y,8,9,Z}%

\newcommand*{\AverageValue}{0}
\newcounter{NumberOfValidEnteries}
\newcommand*{\ComputeAverage}[1]{%
    %% Initialize (in case this is used more than once)
    \renewcommand*{\AverageValue}{0}
    \setcounter{NumberOfValidEnteries}{0}%
    %
    \edef\ExpandedDataEnteries{#1}% So that #1 can be a list or a macro defined list.
    \foreach \NewData in \ExpandedDataEnteries {%
        \IfIsInteger{\NewData}{%
            %\global% <---- I need this ????
            \pgfmathsetmacro{\AverageValue}{%  
                (\AverageValue * \arabic{NumberOfValidEnteries} + \NewData) /
                (\arabic{NumberOfValidEnteries} + 1)
            }%
            \stepcounter{NumberOfValidEnteries}%
            \typeout{**** NumberOfValidEnteries=\arabic{NumberOfValidEnteries}, \string\AverageValue=\AverageValue}%
        }{%
            % Not integer data so skip it
        }%
    }%
}

\begin{document}
    \ComputeAverage{\MyData}%
    Average of integers in ``\MyData" is ``\AverageValue".
\end{document}

Responder1

Você pode colocar legalmente \globalapenas antes de um comando de atribuição:

\global\def
\global\let
\global\mathchardef
\global\catcode
\global\advance

e assim por diante. Deve-se notar que\global quer e procuraum comando adequado seguindo-o, fazendo expansão. Por exemplo, etoolboxdefine

\newrobustcmd{\cslet}[2]{%
  \expandafter\let\csname#1\endcsname#2}

então \global\cslet{foo}{\bar}é legal, porque se tornará

\global\expandafter\let\csname foo\endcsname\bar

após a primeira expansão, mas como \expandafteré expansível, \globalaciona sua expansão e você obtém \global\let\foo\baro que deseja.

Por outro lado, \pgfmathsetmacroé definido pgfmathcalc.code.texpor

\def\pgfmathsetmacro#1#2{%
  \begingroup%
    \pgfmathparse{#2}%
        \let#1=\pgfmathresult
    \pgfmath@smuggleone{#1}%
  \endgroup%
}

então definitivamente não se pode adicionar \globalna frente dele.

A solução mais fácil é dizer

\pgfmathparse
  (\AverageValue * \arabic{NumberOfValidEnteries} + \NewData) /
   (\arabic{NumberOfValidEnteries} + 1)
  }%
\global\let\AverageValue\pgfmathresult

já que \pgfmathparsearmazena o resultado em \pgfmathresult.

informação relacionada