What is the accessibility of a package's `Private` context variables? The 2019 Stack Overflow Developer Survey Results Are In Unicorn Meta Zoo #1: Why another podcast? Announcing the arrival of Valued Associate #679: Cesar ManaraHow symbol lookup actually worksWhat are recommended guidelines for developing packages?How to properly handle mutual imports of multiple packages?How can Private functions be made completely opaque?Does one need to be careful about loading multiple (many) contexts or packages in the same session?WebServices context problemHow to pass rules to packagesA question regarding shadowed symbolsIs there any harm or benefit to Removing unneeded private symbols in packages?Information (??) of function defined in Package return the function with long name of variablesHow to resolve a context shadow problem (revised)

How to determine omitted units in a publication

Store Dynamic-accessible hidden metadata in a cell

Homework question about an engine pulling a train

How to support a colleague who finds meetings extremely tiring?

Is an up-to-date browser secure on an out-of-date OS?

Do working physicists consider Newtonian mechanics to be "falsified"?

Did the new image of black hole confirm the general theory of relativity?

What was the last x86 CPU that did not have the x87 floating-point unit built in?

Why doesn't a hydraulic lever violate conservation of energy?

Why not take a picture of a closer black hole?

Why did Peik Lin say, "I'm not an animal"?

Didn't get enough time to take a Coding Test - what to do now?

What information about me do stores get via my credit card?

Windows 10: How to Lock (not sleep) laptop on lid close?

Working through the single responsibility principle (SRP) in Python when calls are expensive

how can a perfect fourth interval be considered either consonant or dissonant?

How many cones with angle theta can I pack into the unit sphere?

How did the crowd guess the pentatonic scale in Bobby McFerrin's presentation?

What happens to a Warlock's expended Spell Slots when they gain a Level?

Using dividends to reduce short term capital gains?

ELI5: Why do they say that Israel would have been the fourth country to land a spacecraft on the Moon and why do they call it low cost?

Why are PDP-7-style microprogrammed instructions out of vogue?

Is there a way to generate uniformly distributed points on a sphere from a fixed amount of random real numbers per point?

What does "spokes" mean in this context?



What is the accessibility of a package's `Private` context variables?



The 2019 Stack Overflow Developer Survey Results Are In
Unicorn Meta Zoo #1: Why another podcast?
Announcing the arrival of Valued Associate #679: Cesar ManaraHow symbol lookup actually worksWhat are recommended guidelines for developing packages?How to properly handle mutual imports of multiple packages?How can Private functions be made completely opaque?Does one need to be careful about loading multiple (many) contexts or packages in the same session?WebServices context problemHow to pass rules to packagesA question regarding shadowed symbolsIs there any harm or benefit to Removing unneeded private symbols in packages?Information (??) of function defined in Package return the function with long name of variablesHow to resolve a context shadow problem (revised)










9












$begingroup$


I've been reading up on how Mathematica handles contexts, $Context, $ContextPath, and a few of the tutorials they have on Packages.



What I'm wondering about is how the functions defined in, say, CustomPackage` are able to access the variables in CustomPackage`Private`.



For example,



BeginPackage["CustomPackage`"]

MyFunction::usage = "MyFunction[arg1] adds 5 to arg1."

Begin["`Private`"]

abc=5;
MyFunction[arg1_] := arg1 + abc;

End[]
EndPackage[]


When I load the package <<CustomPackage` the $ContextPath will have CustomPackage` on it, but not CustomPackage`Private`



So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath










share|improve this question











$endgroup$
















    9












    $begingroup$


    I've been reading up on how Mathematica handles contexts, $Context, $ContextPath, and a few of the tutorials they have on Packages.



    What I'm wondering about is how the functions defined in, say, CustomPackage` are able to access the variables in CustomPackage`Private`.



    For example,



    BeginPackage["CustomPackage`"]

    MyFunction::usage = "MyFunction[arg1] adds 5 to arg1."

    Begin["`Private`"]

    abc=5;
    MyFunction[arg1_] := arg1 + abc;

    End[]
    EndPackage[]


    When I load the package <<CustomPackage` the $ContextPath will have CustomPackage` on it, but not CustomPackage`Private`



    So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath










    share|improve this question











    $endgroup$














      9












      9








      9





      $begingroup$


      I've been reading up on how Mathematica handles contexts, $Context, $ContextPath, and a few of the tutorials they have on Packages.



      What I'm wondering about is how the functions defined in, say, CustomPackage` are able to access the variables in CustomPackage`Private`.



      For example,



      BeginPackage["CustomPackage`"]

      MyFunction::usage = "MyFunction[arg1] adds 5 to arg1."

      Begin["`Private`"]

      abc=5;
      MyFunction[arg1_] := arg1 + abc;

      End[]
      EndPackage[]


      When I load the package <<CustomPackage` the $ContextPath will have CustomPackage` on it, but not CustomPackage`Private`



      So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath










      share|improve this question











      $endgroup$




      I've been reading up on how Mathematica handles contexts, $Context, $ContextPath, and a few of the tutorials they have on Packages.



      What I'm wondering about is how the functions defined in, say, CustomPackage` are able to access the variables in CustomPackage`Private`.



      For example,



      BeginPackage["CustomPackage`"]

      MyFunction::usage = "MyFunction[arg1] adds 5 to arg1."

      Begin["`Private`"]

      abc=5;
      MyFunction[arg1_] := arg1 + abc;

      End[]
      EndPackage[]


      When I load the package <<CustomPackage` the $ContextPath will have CustomPackage` on it, but not CustomPackage`Private`



      So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath







      packages core-language scoping contexts






      share|improve this question















      share|improve this question













      share|improve this question




      share|improve this question








      edited 2 days ago









      m_goldberg

      88.6k873200




      88.6k873200










      asked 2 days ago









      w1resw1res

      22514




      22514




















          4 Answers
          4






          active

          oldest

          votes


















          9












          $begingroup$


          So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath




          There is a misunderstanding here. You are assuming that abc is searched for in some context only when MyFunction[something] is evaluated. This is not the case.



          $Context and $ContextPath only affect how source code is parsed (not how expressions are evaluated). In other words, they only affect how the text you write in the package file is interpreted and converted into in-memory expressions. Once the package has been loaded with Get, this interpretation has already happened. MyFunction has been interpreted as the symbol CustomPackage`MyFunction and abc has been interpreted as CustomPackage`Private`abc, according to the value of $Context and $ContextPath at the time each was read. These are the full names of these symbols and this is how they exist in memory.



          Load the package and try this:



          Block[$ContextPath,
          Print@Definition[MyFunction]
          ]


          You'll see the following printed:



          CustomPackage`MyFunction[CustomPackage`Private`arg1_] := 
          CustomPackage`Private`arg1+CustomPackage`Private`abc


          As you can see, a context is always associated with every symbol.






          share|improve this answer











          $endgroup$




















            6












            $begingroup$

            Begin["`Private`"]; sets the current $Context to "CustomPackage `Private`". This causes two things:



            • The symbol abc will be searched in the current context first, thus in"CustomPackage`Private`". Only if it is not found there, the search goes on along $ContextPath.


            • If no matching symbol is found this way, a new symbol abc is created, namely in the current $Context which is "CustomPackage`Private`". So the full symbol name is "CustomPackage`Private`abc".


            For example, running your code in a fresh kernel and executing



            ??MyFunction


            reveals that the full definition of MyFunction is




            MyFunction[CustomPackage`Private`arg1_]:=CustomPackage`Private`arg1+CustomPackage`Private`abc




            Moreover, with



             ?*`abc


            you see that the only symbol in all contexts that matches abc is CustomPackage`Private`abc and has the value 5 assigned to it.






            share|improve this answer











            $endgroup$




















              5












              $begingroup$

              All symbols are created at load time, so when you do:



              BeginPackage["X`"];

              x::usage="Declaring x as an exported symbol in the X` context";

              Begin["`SomePrivateContext`"];

              x[a_]:=b

              End[];

              EndPackage[];


              x was created as X`x but the DownValues of x reference X`SomePrivateContext`a and X`SomePrivateContext`b which were created at the time the function was defined. These symbols are unique, so that reference only ever points that a single object.






              share|improve this answer









              $endgroup$




















                2












                $begingroup$


                So how does MyFunction know the value of abc
                at the delayed function call (when it is called)
                if the Private` context isn't on the $ContextPath?




                because "CustomPackage`Private`" is the value of $Context when MyFunction is defined (i.e. it is not just $ContextPath that determines what a function sees but also what is on $Context).




                TL:DR



                This is a timely question because it indirectly touches upon the competing imperatives of developers and end-users. To the question itself:



                The whole point of packages is that they are a form of encapsulation that allows developers to, without interferance, implement functionality for end-users without bothering them with the underlying details. In particular, the encapsulation involves controlling namespaces so that the underlying details can involve symbols that help implement the functionality but ultimately don't end-up polluting a user's namespace. All symbols defined in a "*`Private`" namespace are created for exactly this purpose.



                Hence in the OP's example, the variable abc is an underlying detail for the implementation of the public MyFunction. The developer needs the "detail" of abc but this particular symbol but is of no direct interest to an end-user who typically just ends up calling MyFunction[].



                The package layout achieves this encapsulation by manipulating $ContextPath and $Context as the control-flow passes through the package when it is first loaded. This is described in the other answers and documentation but it can be useful to see it directly:



                loc[n_] := Sow[<|
                "Location" -> n,
                "$Context" -> $Context,
                "$ContextPath" -> $ContextPath|>];

                Reap[

                loc@1;

                BeginPackage["CustomPackage`"];

                loc@2;

                MyFunction::usage = "MyFunction[arg1] adds 5 to arg1.";

                Begin["`Private`"];

                loc@3;

                abc = 5;
                MyFunction[arg1_] := arg1 + abc;

                End[];

                loc@4;

                EndPackage[];

                loc@5
                ]// Last // Dataset


                enter image description here




                When I load the package <the $ContextPath will have CustomPackage on it, but not CustomPackagePrivate




                Yes, this implements both the public exporting of all CustomPackage functions but without polluting end-users namespaces with implementation details. In code around Location 3, all packages are cleared out thereby eliminating possible conflicts with existing abc definitions in currently loaded packages. This is encapsulation benefitting developers but the encapsulation benefitting end-users, as observed, is that on exiting (at Location 5) $ContextPath contains "CustomPackage`" (to provide access to MyFunction) but not "CustomPackage`Private`" thereby shielding users from symbols used in MyFunction's implementation.



                A programmatic confirmation at Location 5 gives:



                MemberQ["CustomPackage`"]@$ContextPath, 
                MemberQ["CustomPackage`Private`"]@$ContextPath,
                Context["abc"]

                True, False, "Global`"



                At Location 3 in the control-flow, the symbol abc is not contained in any of the contexts defined in $ContextPath, ("CustomPackage`", or "System`") nor is it (yet) in the context defined in $Context ("CustomPackage`Private`"). Consequently, the name abc gets created in the context currently set to $Context. At this location $Context has value "CustomPackage`Private`" and hence the symbol CustomPackage`Private`abc is created. When the control flow then moves on to MyFunction[], "CustomPackage`Private`" is still the value of $Context so this function "sees" abc (hence it not just $ContextPath that determines what a function sees but what is on both$ContextPath and $Context).



                Note how the convention of placing usage definitions at Location 2 is ostensibly for documentation purposes but its more important role is to ensure that the function goes into the package's context (see $Context at Location 2) before subsequently being made available in the implementation and for end-users (see $ContextPath at Locations 3 and 5).



                IMO it is kind of cool how these placement protocols just work intuitively without necessarily keeping front-of-mind all the control-flow manipulations, variable-creation mechanisms etc taking place behind the scenes. Hence this means being very careful changing the framework but also IMHO the time is ripe for such extensions given that the line between users/developers may well be in the process of blurring.






                share|improve this answer











                $endgroup$













                  Your Answer








                  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%2f194963%2fwhat-is-the-accessibility-of-a-packages-private-context-variables%23new-answer', 'question_page');

                  );

                  Post as a guest















                  Required, but never shown

























                  4 Answers
                  4






                  active

                  oldest

                  votes








                  4 Answers
                  4






                  active

                  oldest

                  votes









                  active

                  oldest

                  votes






                  active

                  oldest

                  votes









                  9












                  $begingroup$


                  So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath




                  There is a misunderstanding here. You are assuming that abc is searched for in some context only when MyFunction[something] is evaluated. This is not the case.



                  $Context and $ContextPath only affect how source code is parsed (not how expressions are evaluated). In other words, they only affect how the text you write in the package file is interpreted and converted into in-memory expressions. Once the package has been loaded with Get, this interpretation has already happened. MyFunction has been interpreted as the symbol CustomPackage`MyFunction and abc has been interpreted as CustomPackage`Private`abc, according to the value of $Context and $ContextPath at the time each was read. These are the full names of these symbols and this is how they exist in memory.



                  Load the package and try this:



                  Block[$ContextPath,
                  Print@Definition[MyFunction]
                  ]


                  You'll see the following printed:



                  CustomPackage`MyFunction[CustomPackage`Private`arg1_] := 
                  CustomPackage`Private`arg1+CustomPackage`Private`abc


                  As you can see, a context is always associated with every symbol.






                  share|improve this answer











                  $endgroup$

















                    9












                    $begingroup$


                    So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath




                    There is a misunderstanding here. You are assuming that abc is searched for in some context only when MyFunction[something] is evaluated. This is not the case.



                    $Context and $ContextPath only affect how source code is parsed (not how expressions are evaluated). In other words, they only affect how the text you write in the package file is interpreted and converted into in-memory expressions. Once the package has been loaded with Get, this interpretation has already happened. MyFunction has been interpreted as the symbol CustomPackage`MyFunction and abc has been interpreted as CustomPackage`Private`abc, according to the value of $Context and $ContextPath at the time each was read. These are the full names of these symbols and this is how they exist in memory.



                    Load the package and try this:



                    Block[$ContextPath,
                    Print@Definition[MyFunction]
                    ]


                    You'll see the following printed:



                    CustomPackage`MyFunction[CustomPackage`Private`arg1_] := 
                    CustomPackage`Private`arg1+CustomPackage`Private`abc


                    As you can see, a context is always associated with every symbol.






                    share|improve this answer











                    $endgroup$















                      9












                      9








                      9





                      $begingroup$


                      So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath




                      There is a misunderstanding here. You are assuming that abc is searched for in some context only when MyFunction[something] is evaluated. This is not the case.



                      $Context and $ContextPath only affect how source code is parsed (not how expressions are evaluated). In other words, they only affect how the text you write in the package file is interpreted and converted into in-memory expressions. Once the package has been loaded with Get, this interpretation has already happened. MyFunction has been interpreted as the symbol CustomPackage`MyFunction and abc has been interpreted as CustomPackage`Private`abc, according to the value of $Context and $ContextPath at the time each was read. These are the full names of these symbols and this is how they exist in memory.



                      Load the package and try this:



                      Block[$ContextPath,
                      Print@Definition[MyFunction]
                      ]


                      You'll see the following printed:



                      CustomPackage`MyFunction[CustomPackage`Private`arg1_] := 
                      CustomPackage`Private`arg1+CustomPackage`Private`abc


                      As you can see, a context is always associated with every symbol.






                      share|improve this answer











                      $endgroup$




                      So how does MyFunction know the value of abc at the delayed function call (when it is called) if the Private` context isn't on the $ContextPath




                      There is a misunderstanding here. You are assuming that abc is searched for in some context only when MyFunction[something] is evaluated. This is not the case.



                      $Context and $ContextPath only affect how source code is parsed (not how expressions are evaluated). In other words, they only affect how the text you write in the package file is interpreted and converted into in-memory expressions. Once the package has been loaded with Get, this interpretation has already happened. MyFunction has been interpreted as the symbol CustomPackage`MyFunction and abc has been interpreted as CustomPackage`Private`abc, according to the value of $Context and $ContextPath at the time each was read. These are the full names of these symbols and this is how they exist in memory.



                      Load the package and try this:



                      Block[$ContextPath,
                      Print@Definition[MyFunction]
                      ]


                      You'll see the following printed:



                      CustomPackage`MyFunction[CustomPackage`Private`arg1_] := 
                      CustomPackage`Private`arg1+CustomPackage`Private`abc


                      As you can see, a context is always associated with every symbol.







                      share|improve this answer














                      share|improve this answer



                      share|improve this answer








                      edited 2 days ago

























                      answered 2 days ago









                      SzabolcsSzabolcs

                      164k14448947




                      164k14448947





















                          6












                          $begingroup$

                          Begin["`Private`"]; sets the current $Context to "CustomPackage `Private`". This causes two things:



                          • The symbol abc will be searched in the current context first, thus in"CustomPackage`Private`". Only if it is not found there, the search goes on along $ContextPath.


                          • If no matching symbol is found this way, a new symbol abc is created, namely in the current $Context which is "CustomPackage`Private`". So the full symbol name is "CustomPackage`Private`abc".


                          For example, running your code in a fresh kernel and executing



                          ??MyFunction


                          reveals that the full definition of MyFunction is




                          MyFunction[CustomPackage`Private`arg1_]:=CustomPackage`Private`arg1+CustomPackage`Private`abc




                          Moreover, with



                           ?*`abc


                          you see that the only symbol in all contexts that matches abc is CustomPackage`Private`abc and has the value 5 assigned to it.






                          share|improve this answer











                          $endgroup$

















                            6












                            $begingroup$

                            Begin["`Private`"]; sets the current $Context to "CustomPackage `Private`". This causes two things:



                            • The symbol abc will be searched in the current context first, thus in"CustomPackage`Private`". Only if it is not found there, the search goes on along $ContextPath.


                            • If no matching symbol is found this way, a new symbol abc is created, namely in the current $Context which is "CustomPackage`Private`". So the full symbol name is "CustomPackage`Private`abc".


                            For example, running your code in a fresh kernel and executing



                            ??MyFunction


                            reveals that the full definition of MyFunction is




                            MyFunction[CustomPackage`Private`arg1_]:=CustomPackage`Private`arg1+CustomPackage`Private`abc




                            Moreover, with



                             ?*`abc


                            you see that the only symbol in all contexts that matches abc is CustomPackage`Private`abc and has the value 5 assigned to it.






                            share|improve this answer











                            $endgroup$















                              6












                              6








                              6





                              $begingroup$

                              Begin["`Private`"]; sets the current $Context to "CustomPackage `Private`". This causes two things:



                              • The symbol abc will be searched in the current context first, thus in"CustomPackage`Private`". Only if it is not found there, the search goes on along $ContextPath.


                              • If no matching symbol is found this way, a new symbol abc is created, namely in the current $Context which is "CustomPackage`Private`". So the full symbol name is "CustomPackage`Private`abc".


                              For example, running your code in a fresh kernel and executing



                              ??MyFunction


                              reveals that the full definition of MyFunction is




                              MyFunction[CustomPackage`Private`arg1_]:=CustomPackage`Private`arg1+CustomPackage`Private`abc




                              Moreover, with



                               ?*`abc


                              you see that the only symbol in all contexts that matches abc is CustomPackage`Private`abc and has the value 5 assigned to it.






                              share|improve this answer











                              $endgroup$



                              Begin["`Private`"]; sets the current $Context to "CustomPackage `Private`". This causes two things:



                              • The symbol abc will be searched in the current context first, thus in"CustomPackage`Private`". Only if it is not found there, the search goes on along $ContextPath.


                              • If no matching symbol is found this way, a new symbol abc is created, namely in the current $Context which is "CustomPackage`Private`". So the full symbol name is "CustomPackage`Private`abc".


                              For example, running your code in a fresh kernel and executing



                              ??MyFunction


                              reveals that the full definition of MyFunction is




                              MyFunction[CustomPackage`Private`arg1_]:=CustomPackage`Private`arg1+CustomPackage`Private`abc




                              Moreover, with



                               ?*`abc


                              you see that the only symbol in all contexts that matches abc is CustomPackage`Private`abc and has the value 5 assigned to it.







                              share|improve this answer














                              share|improve this answer



                              share|improve this answer








                              edited 2 days ago

























                              answered 2 days ago









                              Henrik SchumacherHenrik Schumacher

                              60k582168




                              60k582168





















                                  5












                                  $begingroup$

                                  All symbols are created at load time, so when you do:



                                  BeginPackage["X`"];

                                  x::usage="Declaring x as an exported symbol in the X` context";

                                  Begin["`SomePrivateContext`"];

                                  x[a_]:=b

                                  End[];

                                  EndPackage[];


                                  x was created as X`x but the DownValues of x reference X`SomePrivateContext`a and X`SomePrivateContext`b which were created at the time the function was defined. These symbols are unique, so that reference only ever points that a single object.






                                  share|improve this answer









                                  $endgroup$

















                                    5












                                    $begingroup$

                                    All symbols are created at load time, so when you do:



                                    BeginPackage["X`"];

                                    x::usage="Declaring x as an exported symbol in the X` context";

                                    Begin["`SomePrivateContext`"];

                                    x[a_]:=b

                                    End[];

                                    EndPackage[];


                                    x was created as X`x but the DownValues of x reference X`SomePrivateContext`a and X`SomePrivateContext`b which were created at the time the function was defined. These symbols are unique, so that reference only ever points that a single object.






                                    share|improve this answer









                                    $endgroup$















                                      5












                                      5








                                      5





                                      $begingroup$

                                      All symbols are created at load time, so when you do:



                                      BeginPackage["X`"];

                                      x::usage="Declaring x as an exported symbol in the X` context";

                                      Begin["`SomePrivateContext`"];

                                      x[a_]:=b

                                      End[];

                                      EndPackage[];


                                      x was created as X`x but the DownValues of x reference X`SomePrivateContext`a and X`SomePrivateContext`b which were created at the time the function was defined. These symbols are unique, so that reference only ever points that a single object.






                                      share|improve this answer









                                      $endgroup$



                                      All symbols are created at load time, so when you do:



                                      BeginPackage["X`"];

                                      x::usage="Declaring x as an exported symbol in the X` context";

                                      Begin["`SomePrivateContext`"];

                                      x[a_]:=b

                                      End[];

                                      EndPackage[];


                                      x was created as X`x but the DownValues of x reference X`SomePrivateContext`a and X`SomePrivateContext`b which were created at the time the function was defined. These symbols are unique, so that reference only ever points that a single object.







                                      share|improve this answer












                                      share|improve this answer



                                      share|improve this answer










                                      answered 2 days ago









                                      b3m2a1b3m2a1

                                      28.7k359165




                                      28.7k359165





















                                          2












                                          $begingroup$


                                          So how does MyFunction know the value of abc
                                          at the delayed function call (when it is called)
                                          if the Private` context isn't on the $ContextPath?




                                          because "CustomPackage`Private`" is the value of $Context when MyFunction is defined (i.e. it is not just $ContextPath that determines what a function sees but also what is on $Context).




                                          TL:DR



                                          This is a timely question because it indirectly touches upon the competing imperatives of developers and end-users. To the question itself:



                                          The whole point of packages is that they are a form of encapsulation that allows developers to, without interferance, implement functionality for end-users without bothering them with the underlying details. In particular, the encapsulation involves controlling namespaces so that the underlying details can involve symbols that help implement the functionality but ultimately don't end-up polluting a user's namespace. All symbols defined in a "*`Private`" namespace are created for exactly this purpose.



                                          Hence in the OP's example, the variable abc is an underlying detail for the implementation of the public MyFunction. The developer needs the "detail" of abc but this particular symbol but is of no direct interest to an end-user who typically just ends up calling MyFunction[].



                                          The package layout achieves this encapsulation by manipulating $ContextPath and $Context as the control-flow passes through the package when it is first loaded. This is described in the other answers and documentation but it can be useful to see it directly:



                                          loc[n_] := Sow[<|
                                          "Location" -> n,
                                          "$Context" -> $Context,
                                          "$ContextPath" -> $ContextPath|>];

                                          Reap[

                                          loc@1;

                                          BeginPackage["CustomPackage`"];

                                          loc@2;

                                          MyFunction::usage = "MyFunction[arg1] adds 5 to arg1.";

                                          Begin["`Private`"];

                                          loc@3;

                                          abc = 5;
                                          MyFunction[arg1_] := arg1 + abc;

                                          End[];

                                          loc@4;

                                          EndPackage[];

                                          loc@5
                                          ]// Last // Dataset


                                          enter image description here




                                          When I load the package <the $ContextPath will have CustomPackage on it, but not CustomPackagePrivate




                                          Yes, this implements both the public exporting of all CustomPackage functions but without polluting end-users namespaces with implementation details. In code around Location 3, all packages are cleared out thereby eliminating possible conflicts with existing abc definitions in currently loaded packages. This is encapsulation benefitting developers but the encapsulation benefitting end-users, as observed, is that on exiting (at Location 5) $ContextPath contains "CustomPackage`" (to provide access to MyFunction) but not "CustomPackage`Private`" thereby shielding users from symbols used in MyFunction's implementation.



                                          A programmatic confirmation at Location 5 gives:



                                          MemberQ["CustomPackage`"]@$ContextPath, 
                                          MemberQ["CustomPackage`Private`"]@$ContextPath,
                                          Context["abc"]

                                          True, False, "Global`"



                                          At Location 3 in the control-flow, the symbol abc is not contained in any of the contexts defined in $ContextPath, ("CustomPackage`", or "System`") nor is it (yet) in the context defined in $Context ("CustomPackage`Private`"). Consequently, the name abc gets created in the context currently set to $Context. At this location $Context has value "CustomPackage`Private`" and hence the symbol CustomPackage`Private`abc is created. When the control flow then moves on to MyFunction[], "CustomPackage`Private`" is still the value of $Context so this function "sees" abc (hence it not just $ContextPath that determines what a function sees but what is on both$ContextPath and $Context).



                                          Note how the convention of placing usage definitions at Location 2 is ostensibly for documentation purposes but its more important role is to ensure that the function goes into the package's context (see $Context at Location 2) before subsequently being made available in the implementation and for end-users (see $ContextPath at Locations 3 and 5).



                                          IMO it is kind of cool how these placement protocols just work intuitively without necessarily keeping front-of-mind all the control-flow manipulations, variable-creation mechanisms etc taking place behind the scenes. Hence this means being very careful changing the framework but also IMHO the time is ripe for such extensions given that the line between users/developers may well be in the process of blurring.






                                          share|improve this answer











                                          $endgroup$

















                                            2












                                            $begingroup$


                                            So how does MyFunction know the value of abc
                                            at the delayed function call (when it is called)
                                            if the Private` context isn't on the $ContextPath?




                                            because "CustomPackage`Private`" is the value of $Context when MyFunction is defined (i.e. it is not just $ContextPath that determines what a function sees but also what is on $Context).




                                            TL:DR



                                            This is a timely question because it indirectly touches upon the competing imperatives of developers and end-users. To the question itself:



                                            The whole point of packages is that they are a form of encapsulation that allows developers to, without interferance, implement functionality for end-users without bothering them with the underlying details. In particular, the encapsulation involves controlling namespaces so that the underlying details can involve symbols that help implement the functionality but ultimately don't end-up polluting a user's namespace. All symbols defined in a "*`Private`" namespace are created for exactly this purpose.



                                            Hence in the OP's example, the variable abc is an underlying detail for the implementation of the public MyFunction. The developer needs the "detail" of abc but this particular symbol but is of no direct interest to an end-user who typically just ends up calling MyFunction[].



                                            The package layout achieves this encapsulation by manipulating $ContextPath and $Context as the control-flow passes through the package when it is first loaded. This is described in the other answers and documentation but it can be useful to see it directly:



                                            loc[n_] := Sow[<|
                                            "Location" -> n,
                                            "$Context" -> $Context,
                                            "$ContextPath" -> $ContextPath|>];

                                            Reap[

                                            loc@1;

                                            BeginPackage["CustomPackage`"];

                                            loc@2;

                                            MyFunction::usage = "MyFunction[arg1] adds 5 to arg1.";

                                            Begin["`Private`"];

                                            loc@3;

                                            abc = 5;
                                            MyFunction[arg1_] := arg1 + abc;

                                            End[];

                                            loc@4;

                                            EndPackage[];

                                            loc@5
                                            ]// Last // Dataset


                                            enter image description here




                                            When I load the package <the $ContextPath will have CustomPackage on it, but not CustomPackagePrivate




                                            Yes, this implements both the public exporting of all CustomPackage functions but without polluting end-users namespaces with implementation details. In code around Location 3, all packages are cleared out thereby eliminating possible conflicts with existing abc definitions in currently loaded packages. This is encapsulation benefitting developers but the encapsulation benefitting end-users, as observed, is that on exiting (at Location 5) $ContextPath contains "CustomPackage`" (to provide access to MyFunction) but not "CustomPackage`Private`" thereby shielding users from symbols used in MyFunction's implementation.



                                            A programmatic confirmation at Location 5 gives:



                                            MemberQ["CustomPackage`"]@$ContextPath, 
                                            MemberQ["CustomPackage`Private`"]@$ContextPath,
                                            Context["abc"]

                                            True, False, "Global`"



                                            At Location 3 in the control-flow, the symbol abc is not contained in any of the contexts defined in $ContextPath, ("CustomPackage`", or "System`") nor is it (yet) in the context defined in $Context ("CustomPackage`Private`"). Consequently, the name abc gets created in the context currently set to $Context. At this location $Context has value "CustomPackage`Private`" and hence the symbol CustomPackage`Private`abc is created. When the control flow then moves on to MyFunction[], "CustomPackage`Private`" is still the value of $Context so this function "sees" abc (hence it not just $ContextPath that determines what a function sees but what is on both$ContextPath and $Context).



                                            Note how the convention of placing usage definitions at Location 2 is ostensibly for documentation purposes but its more important role is to ensure that the function goes into the package's context (see $Context at Location 2) before subsequently being made available in the implementation and for end-users (see $ContextPath at Locations 3 and 5).



                                            IMO it is kind of cool how these placement protocols just work intuitively without necessarily keeping front-of-mind all the control-flow manipulations, variable-creation mechanisms etc taking place behind the scenes. Hence this means being very careful changing the framework but also IMHO the time is ripe for such extensions given that the line between users/developers may well be in the process of blurring.






                                            share|improve this answer











                                            $endgroup$















                                              2












                                              2








                                              2





                                              $begingroup$


                                              So how does MyFunction know the value of abc
                                              at the delayed function call (when it is called)
                                              if the Private` context isn't on the $ContextPath?




                                              because "CustomPackage`Private`" is the value of $Context when MyFunction is defined (i.e. it is not just $ContextPath that determines what a function sees but also what is on $Context).




                                              TL:DR



                                              This is a timely question because it indirectly touches upon the competing imperatives of developers and end-users. To the question itself:



                                              The whole point of packages is that they are a form of encapsulation that allows developers to, without interferance, implement functionality for end-users without bothering them with the underlying details. In particular, the encapsulation involves controlling namespaces so that the underlying details can involve symbols that help implement the functionality but ultimately don't end-up polluting a user's namespace. All symbols defined in a "*`Private`" namespace are created for exactly this purpose.



                                              Hence in the OP's example, the variable abc is an underlying detail for the implementation of the public MyFunction. The developer needs the "detail" of abc but this particular symbol but is of no direct interest to an end-user who typically just ends up calling MyFunction[].



                                              The package layout achieves this encapsulation by manipulating $ContextPath and $Context as the control-flow passes through the package when it is first loaded. This is described in the other answers and documentation but it can be useful to see it directly:



                                              loc[n_] := Sow[<|
                                              "Location" -> n,
                                              "$Context" -> $Context,
                                              "$ContextPath" -> $ContextPath|>];

                                              Reap[

                                              loc@1;

                                              BeginPackage["CustomPackage`"];

                                              loc@2;

                                              MyFunction::usage = "MyFunction[arg1] adds 5 to arg1.";

                                              Begin["`Private`"];

                                              loc@3;

                                              abc = 5;
                                              MyFunction[arg1_] := arg1 + abc;

                                              End[];

                                              loc@4;

                                              EndPackage[];

                                              loc@5
                                              ]// Last // Dataset


                                              enter image description here




                                              When I load the package <the $ContextPath will have CustomPackage on it, but not CustomPackagePrivate




                                              Yes, this implements both the public exporting of all CustomPackage functions but without polluting end-users namespaces with implementation details. In code around Location 3, all packages are cleared out thereby eliminating possible conflicts with existing abc definitions in currently loaded packages. This is encapsulation benefitting developers but the encapsulation benefitting end-users, as observed, is that on exiting (at Location 5) $ContextPath contains "CustomPackage`" (to provide access to MyFunction) but not "CustomPackage`Private`" thereby shielding users from symbols used in MyFunction's implementation.



                                              A programmatic confirmation at Location 5 gives:



                                              MemberQ["CustomPackage`"]@$ContextPath, 
                                              MemberQ["CustomPackage`Private`"]@$ContextPath,
                                              Context["abc"]

                                              True, False, "Global`"



                                              At Location 3 in the control-flow, the symbol abc is not contained in any of the contexts defined in $ContextPath, ("CustomPackage`", or "System`") nor is it (yet) in the context defined in $Context ("CustomPackage`Private`"). Consequently, the name abc gets created in the context currently set to $Context. At this location $Context has value "CustomPackage`Private`" and hence the symbol CustomPackage`Private`abc is created. When the control flow then moves on to MyFunction[], "CustomPackage`Private`" is still the value of $Context so this function "sees" abc (hence it not just $ContextPath that determines what a function sees but what is on both$ContextPath and $Context).



                                              Note how the convention of placing usage definitions at Location 2 is ostensibly for documentation purposes but its more important role is to ensure that the function goes into the package's context (see $Context at Location 2) before subsequently being made available in the implementation and for end-users (see $ContextPath at Locations 3 and 5).



                                              IMO it is kind of cool how these placement protocols just work intuitively without necessarily keeping front-of-mind all the control-flow manipulations, variable-creation mechanisms etc taking place behind the scenes. Hence this means being very careful changing the framework but also IMHO the time is ripe for such extensions given that the line between users/developers may well be in the process of blurring.






                                              share|improve this answer











                                              $endgroup$




                                              So how does MyFunction know the value of abc
                                              at the delayed function call (when it is called)
                                              if the Private` context isn't on the $ContextPath?




                                              because "CustomPackage`Private`" is the value of $Context when MyFunction is defined (i.e. it is not just $ContextPath that determines what a function sees but also what is on $Context).




                                              TL:DR



                                              This is a timely question because it indirectly touches upon the competing imperatives of developers and end-users. To the question itself:



                                              The whole point of packages is that they are a form of encapsulation that allows developers to, without interferance, implement functionality for end-users without bothering them with the underlying details. In particular, the encapsulation involves controlling namespaces so that the underlying details can involve symbols that help implement the functionality but ultimately don't end-up polluting a user's namespace. All symbols defined in a "*`Private`" namespace are created for exactly this purpose.



                                              Hence in the OP's example, the variable abc is an underlying detail for the implementation of the public MyFunction. The developer needs the "detail" of abc but this particular symbol but is of no direct interest to an end-user who typically just ends up calling MyFunction[].



                                              The package layout achieves this encapsulation by manipulating $ContextPath and $Context as the control-flow passes through the package when it is first loaded. This is described in the other answers and documentation but it can be useful to see it directly:



                                              loc[n_] := Sow[<|
                                              "Location" -> n,
                                              "$Context" -> $Context,
                                              "$ContextPath" -> $ContextPath|>];

                                              Reap[

                                              loc@1;

                                              BeginPackage["CustomPackage`"];

                                              loc@2;

                                              MyFunction::usage = "MyFunction[arg1] adds 5 to arg1.";

                                              Begin["`Private`"];

                                              loc@3;

                                              abc = 5;
                                              MyFunction[arg1_] := arg1 + abc;

                                              End[];

                                              loc@4;

                                              EndPackage[];

                                              loc@5
                                              ]// Last // Dataset


                                              enter image description here




                                              When I load the package <the $ContextPath will have CustomPackage on it, but not CustomPackagePrivate




                                              Yes, this implements both the public exporting of all CustomPackage functions but without polluting end-users namespaces with implementation details. In code around Location 3, all packages are cleared out thereby eliminating possible conflicts with existing abc definitions in currently loaded packages. This is encapsulation benefitting developers but the encapsulation benefitting end-users, as observed, is that on exiting (at Location 5) $ContextPath contains "CustomPackage`" (to provide access to MyFunction) but not "CustomPackage`Private`" thereby shielding users from symbols used in MyFunction's implementation.



                                              A programmatic confirmation at Location 5 gives:



                                              MemberQ["CustomPackage`"]@$ContextPath, 
                                              MemberQ["CustomPackage`Private`"]@$ContextPath,
                                              Context["abc"]

                                              True, False, "Global`"



                                              At Location 3 in the control-flow, the symbol abc is not contained in any of the contexts defined in $ContextPath, ("CustomPackage`", or "System`") nor is it (yet) in the context defined in $Context ("CustomPackage`Private`"). Consequently, the name abc gets created in the context currently set to $Context. At this location $Context has value "CustomPackage`Private`" and hence the symbol CustomPackage`Private`abc is created. When the control flow then moves on to MyFunction[], "CustomPackage`Private`" is still the value of $Context so this function "sees" abc (hence it not just $ContextPath that determines what a function sees but what is on both$ContextPath and $Context).



                                              Note how the convention of placing usage definitions at Location 2 is ostensibly for documentation purposes but its more important role is to ensure that the function goes into the package's context (see $Context at Location 2) before subsequently being made available in the implementation and for end-users (see $ContextPath at Locations 3 and 5).



                                              IMO it is kind of cool how these placement protocols just work intuitively without necessarily keeping front-of-mind all the control-flow manipulations, variable-creation mechanisms etc taking place behind the scenes. Hence this means being very careful changing the framework but also IMHO the time is ripe for such extensions given that the line between users/developers may well be in the process of blurring.







                                              share|improve this answer














                                              share|improve this answer



                                              share|improve this answer








                                              edited yesterday

























                                              answered yesterday









                                              Ronald MonsonRonald Monson

                                              3,1631633




                                              3,1631633



























                                                  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.




                                                  draft saved


                                                  draft discarded














                                                  StackExchange.ready(
                                                  function ()
                                                  StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fmathematica.stackexchange.com%2fquestions%2f194963%2fwhat-is-the-accessibility-of-a-packages-private-context-variables%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

                                                  Grendel Contents Story Scholarship Depictions Notes References Navigation menu10.1093/notesj/gjn112Berserkeree

                                                  Area configuration aggregation error after install Porto themeMagento 2.1 CE Installed but front/backend not loading/workingCSS not loading on page within Magento 2 pageCannot install module in Magento 2no commands defined in the “setup” namespace. in Magento2Magento 2: Static files are present but shows 404Why do i have to always run the commands to clean cache in Magento 2.1.8?Failure reason: 'Unable to unserialize value.'Error 500 after magento migrationIn production mode the site does not loadMagento 2 : Error 500 after installing

                                                  Middle Expansion Olielle Resaix Definition: Uttering songs of triumph shouting with joy triumphant exulting Sejunction Journal 붙다 달 고급 품목 외출 The stretch trades the screeching tin. Definition: The act of speaking with a drawl a drawl Cough Sand Definition: An uproar a quarrel a noisy outbreak Shake Iron Publicize Horse House Baby 사과 Resaix Flaggy Jelly Temporary Unequaled Puppet A drop in the bucket Shrew 성격 회원 성질 미팅 The burn frames the tacky quality. Materialistic The smoke reduces the way. Yammoe Nondescript Cheek 얼굴 배 약하다 날리다 타다 The illegal country shows the iron. Help Rule Drearien Smoke Teaching Meaty Wasp Abraham Lincoln Jaws 진심 수리하다 Size Cork Idea Convert Think Lark John Lennon 거울 청소 군 추천하다 아이스크림