Overloading functions like Mean for distributions

The name of the pictureThe name of the pictureThe name of the pictureClash Royale CLAN TAG#URR8PPP












8














I'd like to understand how functions like Mean are overloaded to provide the right behavior for the distributions in Mathematica (like NormalDistribution or PoissonDistribution).



I originally assumed it was through the use of UpValues, but now I'm not so sure...



So, if I wanted to implement a function or distribution and define a behavior for it when another function like Mean is applied to it, how would I go about it? I know it's not the following:



f[x_]:=2+x
f /: Mean[f[x_]] := 3 x


Desired behavior:



f[3]
Mean[f[3]]

(*
==> 5
==> 9
*)









share|improve this question

















  • 1




    Have you seen ProbabilityDistribution?
    – Edmund
    Dec 22 '18 at 21:54










  • Nope - also really useful. Thanks!
    – hmode
    Dec 23 '18 at 1:39















8














I'd like to understand how functions like Mean are overloaded to provide the right behavior for the distributions in Mathematica (like NormalDistribution or PoissonDistribution).



I originally assumed it was through the use of UpValues, but now I'm not so sure...



So, if I wanted to implement a function or distribution and define a behavior for it when another function like Mean is applied to it, how would I go about it? I know it's not the following:



f[x_]:=2+x
f /: Mean[f[x_]] := 3 x


Desired behavior:



f[3]
Mean[f[3]]

(*
==> 5
==> 9
*)









share|improve this question

















  • 1




    Have you seen ProbabilityDistribution?
    – Edmund
    Dec 22 '18 at 21:54










  • Nope - also really useful. Thanks!
    – hmode
    Dec 23 '18 at 1:39













8












8








8


1





I'd like to understand how functions like Mean are overloaded to provide the right behavior for the distributions in Mathematica (like NormalDistribution or PoissonDistribution).



I originally assumed it was through the use of UpValues, but now I'm not so sure...



So, if I wanted to implement a function or distribution and define a behavior for it when another function like Mean is applied to it, how would I go about it? I know it's not the following:



f[x_]:=2+x
f /: Mean[f[x_]] := 3 x


Desired behavior:



f[3]
Mean[f[3]]

(*
==> 5
==> 9
*)









share|improve this question













I'd like to understand how functions like Mean are overloaded to provide the right behavior for the distributions in Mathematica (like NormalDistribution or PoissonDistribution).



I originally assumed it was through the use of UpValues, but now I'm not so sure...



So, if I wanted to implement a function or distribution and define a behavior for it when another function like Mean is applied to it, how would I go about it? I know it's not the following:



f[x_]:=2+x
f /: Mean[f[x_]] := 3 x


Desired behavior:



f[3]
Mean[f[3]]

(*
==> 5
==> 9
*)






upvalues






share|improve this question













share|improve this question











share|improve this question




share|improve this question










asked Dec 22 '18 at 3:16









hmode

32728




32728







  • 1




    Have you seen ProbabilityDistribution?
    – Edmund
    Dec 22 '18 at 21:54










  • Nope - also really useful. Thanks!
    – hmode
    Dec 23 '18 at 1:39












  • 1




    Have you seen ProbabilityDistribution?
    – Edmund
    Dec 22 '18 at 21:54










  • Nope - also really useful. Thanks!
    – hmode
    Dec 23 '18 at 1:39







1




1




Have you seen ProbabilityDistribution?
– Edmund
Dec 22 '18 at 21:54




Have you seen ProbabilityDistribution?
– Edmund
Dec 22 '18 at 21:54












Nope - also really useful. Thanks!
– hmode
Dec 23 '18 at 1:39




Nope - also really useful. Thanks!
– hmode
Dec 23 '18 at 1:39










2 Answers
2






active

oldest

votes


















11














Symbolic(!) distributions are recognized by their head, not by their PDF:



Mean[PDF[NormalDistribution[0, 1], x]] 



Mean[E^(-(x^2/2))/Sqrt[2 π]]




So instead of messing around with Mean, I would rather suggest



distro /: PDF[distro[μ_, σ_], x_] := E^(-((x - μ)^2/(2 σ^2)))/(Sqrt[2 π] σ);
distro /: Mean[distro[μ_, σ_]] := μ;
distro /: Variance[distro[μ_, σ_]] := σ^2
Mean[distro[0, 1]]
Variance[distro[0, 1]]



0



1







share|improve this answer






























    6














    This works:



    Unprotect[Mean];
    SetAttributes[Mean, HoldFirst];
    Protect[Mean];
    f[x_] := 2 + x
    f /: Mean[f[x_]] := 3 x


    Since using Unprotect is not reccomended here's another way.



    mean[x_] := Mean[x]
    SetAttributes[mean, HoldFirst]
    f[x_] := 2 + x
    f /: mean[f[x_]] := 3 x





    share|improve this answer
















    • 2




      It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
      – Jason B.
      Dec 22 '18 at 15:05










    • Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
      – hmode
      Dec 23 '18 at 1:35










    Your Answer





    StackExchange.ifUsing("editor", function ()
    return StackExchange.using("mathjaxEditing", function ()
    StackExchange.MarkdownEditor.creationCallbacks.add(function (editor, postfix)
    StackExchange.mathjaxEditing.prepareWmdForMathJax(editor, postfix, [["$", "$"], ["\\(","\\)"]]);
    );
    );
    , "mathjax-editing");

    StackExchange.ready(function()
    var channelOptions =
    tags: "".split(" "),
    id: "387"
    ;
    initTagRenderer("".split(" "), "".split(" "), channelOptions);

    StackExchange.using("externalEditor", function()
    // Have to fire editor after snippets, if snippets enabled
    if (StackExchange.settings.snippets.snippetsEnabled)
    StackExchange.using("snippets", function()
    createEditor();
    );

    else
    createEditor();

    );

    function createEditor()
    StackExchange.prepareEditor(
    heartbeatType: 'answer',
    autoActivateHeartbeat: false,
    convertImagesToLinks: false,
    noModals: true,
    showLowRepImageUploadWarning: true,
    reputationToPostImages: null,
    bindNavPrevention: true,
    postfix: "",
    imageUploader:
    brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
    contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
    allowUrls: true
    ,
    onDemand: true,
    discardSelector: ".discard-answer"
    ,immediatelyShowMarkdownHelp:true
    );



    );













    draft saved

    draft discarded


















    StackExchange.ready(
    function ()
    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fmathematica.stackexchange.com%2fquestions%2f188306%2foverloading-functions-like-mean-for-distributions%23new-answer', 'question_page');

    );

    Post as a guest















    Required, but never shown

























    2 Answers
    2






    active

    oldest

    votes








    2 Answers
    2






    active

    oldest

    votes









    active

    oldest

    votes






    active

    oldest

    votes









    11














    Symbolic(!) distributions are recognized by their head, not by their PDF:



    Mean[PDF[NormalDistribution[0, 1], x]] 



    Mean[E^(-(x^2/2))/Sqrt[2 π]]




    So instead of messing around with Mean, I would rather suggest



    distro /: PDF[distro[μ_, σ_], x_] := E^(-((x - μ)^2/(2 σ^2)))/(Sqrt[2 π] σ);
    distro /: Mean[distro[μ_, σ_]] := μ;
    distro /: Variance[distro[μ_, σ_]] := σ^2
    Mean[distro[0, 1]]
    Variance[distro[0, 1]]



    0



    1







    share|improve this answer



























      11














      Symbolic(!) distributions are recognized by their head, not by their PDF:



      Mean[PDF[NormalDistribution[0, 1], x]] 



      Mean[E^(-(x^2/2))/Sqrt[2 π]]




      So instead of messing around with Mean, I would rather suggest



      distro /: PDF[distro[μ_, σ_], x_] := E^(-((x - μ)^2/(2 σ^2)))/(Sqrt[2 π] σ);
      distro /: Mean[distro[μ_, σ_]] := μ;
      distro /: Variance[distro[μ_, σ_]] := σ^2
      Mean[distro[0, 1]]
      Variance[distro[0, 1]]



      0



      1







      share|improve this answer

























        11












        11








        11






        Symbolic(!) distributions are recognized by their head, not by their PDF:



        Mean[PDF[NormalDistribution[0, 1], x]] 



        Mean[E^(-(x^2/2))/Sqrt[2 π]]




        So instead of messing around with Mean, I would rather suggest



        distro /: PDF[distro[μ_, σ_], x_] := E^(-((x - μ)^2/(2 σ^2)))/(Sqrt[2 π] σ);
        distro /: Mean[distro[μ_, σ_]] := μ;
        distro /: Variance[distro[μ_, σ_]] := σ^2
        Mean[distro[0, 1]]
        Variance[distro[0, 1]]



        0



        1







        share|improve this answer














        Symbolic(!) distributions are recognized by their head, not by their PDF:



        Mean[PDF[NormalDistribution[0, 1], x]] 



        Mean[E^(-(x^2/2))/Sqrt[2 π]]




        So instead of messing around with Mean, I would rather suggest



        distro /: PDF[distro[μ_, σ_], x_] := E^(-((x - μ)^2/(2 σ^2)))/(Sqrt[2 π] σ);
        distro /: Mean[distro[μ_, σ_]] := μ;
        distro /: Variance[distro[μ_, σ_]] := σ^2
        Mean[distro[0, 1]]
        Variance[distro[0, 1]]



        0



        1








        share|improve this answer














        share|improve this answer



        share|improve this answer








        edited Dec 23 '18 at 9:08

























        answered Dec 22 '18 at 11:49









        Henrik Schumacher

        49k467139




        49k467139





















            6














            This works:



            Unprotect[Mean];
            SetAttributes[Mean, HoldFirst];
            Protect[Mean];
            f[x_] := 2 + x
            f /: Mean[f[x_]] := 3 x


            Since using Unprotect is not reccomended here's another way.



            mean[x_] := Mean[x]
            SetAttributes[mean, HoldFirst]
            f[x_] := 2 + x
            f /: mean[f[x_]] := 3 x





            share|improve this answer
















            • 2




              It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
              – Jason B.
              Dec 22 '18 at 15:05










            • Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
              – hmode
              Dec 23 '18 at 1:35















            6














            This works:



            Unprotect[Mean];
            SetAttributes[Mean, HoldFirst];
            Protect[Mean];
            f[x_] := 2 + x
            f /: Mean[f[x_]] := 3 x


            Since using Unprotect is not reccomended here's another way.



            mean[x_] := Mean[x]
            SetAttributes[mean, HoldFirst]
            f[x_] := 2 + x
            f /: mean[f[x_]] := 3 x





            share|improve this answer
















            • 2




              It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
              – Jason B.
              Dec 22 '18 at 15:05










            • Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
              – hmode
              Dec 23 '18 at 1:35













            6












            6








            6






            This works:



            Unprotect[Mean];
            SetAttributes[Mean, HoldFirst];
            Protect[Mean];
            f[x_] := 2 + x
            f /: Mean[f[x_]] := 3 x


            Since using Unprotect is not reccomended here's another way.



            mean[x_] := Mean[x]
            SetAttributes[mean, HoldFirst]
            f[x_] := 2 + x
            f /: mean[f[x_]] := 3 x





            share|improve this answer












            This works:



            Unprotect[Mean];
            SetAttributes[Mean, HoldFirst];
            Protect[Mean];
            f[x_] := 2 + x
            f /: Mean[f[x_]] := 3 x


            Since using Unprotect is not reccomended here's another way.



            mean[x_] := Mean[x]
            SetAttributes[mean, HoldFirst]
            f[x_] := 2 + x
            f /: mean[f[x_]] := 3 x






            share|improve this answer












            share|improve this answer



            share|improve this answer










            answered Dec 22 '18 at 5:57









            Andrew

            1,9011115




            1,9011115







            • 2




              It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
              – Jason B.
              Dec 22 '18 at 15:05










            • Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
              – hmode
              Dec 23 '18 at 1:35












            • 2




              It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
              – Jason B.
              Dec 22 '18 at 15:05










            • Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
              – hmode
              Dec 23 '18 at 1:35







            2




            2




            It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
            – Jason B.
            Dec 22 '18 at 15:05




            It should be mentioned that SetAttributes[Mean, HoldFirst] is likely to break many internal functions that depend on Mean
            – Jason B.
            Dec 22 '18 at 15:05












            Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
            – hmode
            Dec 23 '18 at 1:35




            Thanks for this solution - I did learn something from your answer but I was trying to avoid setting DownValues.
            – hmode
            Dec 23 '18 at 1:35

















            draft saved

            draft discarded
















































            Thanks for contributing an answer to Mathematica Stack Exchange!


            • Please be sure to answer the question. Provide details and share your research!

            But avoid


            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.

            Use MathJax to format equations. MathJax reference.


            To learn more, see our tips on writing great answers.





            Some of your past answers have not been well-received, and you're in danger of being blocked from answering.


            Please pay close attention to the following guidance:


            • Please be sure to answer the question. Provide details and share your research!

            But avoid


            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.

            To learn more, see our tips on writing great answers.




            draft saved


            draft discarded














            StackExchange.ready(
            function ()
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fmathematica.stackexchange.com%2fquestions%2f188306%2foverloading-functions-like-mean-for-distributions%23new-answer', 'question_page');

            );

            Post as a guest















            Required, but never shown





















































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown

































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown






            Popular posts from this blog

            How to check contact read email or not when send email to Individual?

            Displaying single band from multi-band raster using QGIS

            How many registers does an x86_64 CPU actually have?