Skip to main content

variable definitions - How to Set parts of indexed lists?


I would like to assign a list to an indexed variable and then change it using Part and Set like this:


matM[i] = ConstantArray[1, {10, 10}];
matM[i][[1,10]] = 10


Unfortunately I will get the error:



Set::setps: "matM[i] in the part assignment is not a symbol. "



Using Subscript will not work either:


Subscript[matM,i] = ConstantArray[1, {10, 10}];
Subscript[matM,i][[1,10]] = 10

will give the same error.


How can this be done?




Answer



I can offer a sort of a solution, which has its shortcomings, but will allow you (more or less) to use the syntax you want. The idea is that you mark symbols you need, as "references", and then execute your code in a special dynamic environment where Part is overloaded.


Implementation


Here is the code. This function will mark you symbol as a reference.


ClearAll[makeReference];
makeReference[sym_Symbol] :=
sym /: Set[sym[index_], rhs_] :=
With[{index = index},
With[{heldVal = Hold[sym[index]] /. DownValues[sym]},
If[heldVal === Hold[sym[index]],

Module[{ref},
AppendTo[DownValues[sym],
HoldPattern[sym[index]] :> ref]
]
];
Hold[sym[index]] /. DownValues[sym] /. Hold[s_] :> (s = rhs);
]];

What happens is that we "softly" overload Set on this particular symbol via UpValues, so that an intermediate symbol is inserted where the actual data will be stored, and our symbol (for a given index) refers to that intermediate symbol. Since the latter has no restrictions on part assignments, we can assign parts of it directly at O(1) time.


However, the subtlety is that when we call Set[Part[s[ind],1,2],something], Set holds its first argument, and therefore, s can not communicate to Set that this is special (UpValues won't work here since the s is too deep inside an expression - on level 2 - while UpValues are only looked at at level 1). To solve this problem, we will overload Part, but do it locally within a local dynamic environment, to make this operation safer. This is a dynamic environment:



ClearAll[withModifiedPart];
SetAttributes[withModifiedPart, HoldAll];
withModifiedPart[code_] :=
Internal`InheritedBlock[{Part},
Unprotect[Part];
Part /: Set[Part[sym_Symbol[index_], inds__], rhs_] :=
With[{index = index},
Hold[sym[index]] /. DownValues[sym] /.
Hold[s_] :> (s[[inds]] = rhs);
];

Protect[Part];
code];

Tests


Now, we can test this:


ClearAll[a];
makeReference[a];

and then


withModifiedPart[

a[1] = Range[10];
a[1][[2]] = 100;
a[1]
]


 {1, 100, 3, 4, 5, 6, 7, 8, 9, 10}

Let's now measure some timings:


withModifiedPart[

a[1] = Range[100000];
Do[a[1][[i]] = a[1][[i]] + 1, {i, a[1]}];
a[1]
] // Short // AbsoluteTiming


  {1.5126953,{2,3,4,5,6,7,8,9,<<99984>>,99994,99995,99996,99997,
99998,99999,100000,100001}}

We can compare this to the time it takes for direct assignments:



aa = Range[100000];
Do[aa[[i]] = aa[[i]] + 1, {i, aa}]; // Short // AbsoluteTiming


 {0.2470703,Null}

So, for massive assignments, we are about 6 times slower (which I think is OK). We can also see how costly is the overloaded Part for normal assignments:


withModifiedPart[
aa=Range[100000];
Do[aa[[i]]=aa[[i]]+1,{i,aa}]

];//Short//AbsoluteTiming


  {0.2822266,Null}

from where it looks that those are slower by about 15 percents.


Conclusions


The suggested solution requires 2 modifications to your code:



  • call makeReference on symbols which you wish to use as indexed symbols with part assignment, prior to assigning to them.


  • Execute all your code containing such assignments, inside the withModifiedPart environment.


So, your original code will be changed to


ClearAll[matM];
makeReference[matM];

withModifiedPart[
matM[i] = ConstantArray[1, {10, 10}];
matM[i][[1, 10]] = 10
]


What about safety? I would argue that, for this particular case, modifying Part is safe enough. This is because, first, Part is overloaded softly, via UpValues. This means that, when Part is not inside some head which holds arguments, it will execute normally before it would even "think" of a new definition. Now, when it is inside some head which holds arguments, the new definition will only work if that head is Set. Note that no ne rules were added to the Set itself. And since normally, assignments to indexed variables are not allowed anyway, we don't risk breaking some internal behavior.


The performance here is worse than for direct assignments to symbol's parts, but overall seems acceptable.


Comments

Popular posts from this blog

plotting - Filling between two spheres in SphericalPlot3D

Manipulate[ SphericalPlot3D[{1, 2 - n}, {θ, 0, Pi}, {ϕ, 0, 1.5 Pi}, Mesh -> None, PlotPoints -> 15, PlotRange -> {-2.2, 2.2}], {n, 0, 1}] I cant' seem to be able to make a filling between two spheres. I've already tried the obvious Filling -> {1 -> {2}} but Mathematica doesn't seem to like that option. Is there any easy way around this or ... Answer There is no built-in filling in SphericalPlot3D . One option is to use ParametricPlot3D to draw the surfaces between the two shells: Manipulate[ Show[SphericalPlot3D[{1, 2 - n}, {θ, 0, Pi}, {ϕ, 0, 1.5 Pi}, PlotPoints -> 15, PlotRange -> {-2.2, 2.2}], ParametricPlot3D[{ r {Sin[t] Cos[1.5 Pi], Sin[t] Sin[1.5 Pi], Cos[t]}, r {Sin[t] Cos[0 Pi], Sin[t] Sin[0 Pi], Cos[t]}}, {r, 1, 2 - n}, {t, 0, Pi}, PlotStyle -> Yellow, Mesh -> {2, 15}]], {n, 0, 1}]

plotting - Plot 4D data with color as 4th dimension

I have a list of 4D data (x position, y position, amplitude, wavelength). I want to plot x, y, and amplitude on a 3D plot and have the color of the points correspond to the wavelength. I have seen many examples using functions to define color but my wavelength cannot be expressed by an analytic function. Is there a simple way to do this? Answer Here a another possible way to visualize 4D data: data = Flatten[Table[{x, y, x^2 + y^2, Sin[x - y]}, {x, -Pi, Pi,Pi/10}, {y,-Pi,Pi, Pi/10}], 1]; You can use the function Point along with VertexColors . Now the points are places using the first three elements and the color is determined by the fourth. In this case I used Hue, but you can use whatever you prefer. Graphics3D[ Point[data[[All, 1 ;; 3]], VertexColors -> Hue /@ data[[All, 4]]], Axes -> True, BoxRatios -> {1, 1, 1/GoldenRatio}]

plotting - Mathematica: 3D plot based on combined 2D graphs

I have several sigmoidal fits to 3 different datasets, with mean fit predictions plus the 95% confidence limits (not symmetrical around the mean) and the actual data. I would now like to show these different 2D plots projected in 3D as in but then using proper perspective. In the link here they give some solutions to combine the plots using isometric perspective, but I would like to use proper 3 point perspective. Any thoughts? Also any way to show the mean points per time point for each series plus or minus the standard error on the mean would be cool too, either using points+vertical bars, or using spheres plus tubes. Below are some test data and the fit function I am using. Note that I am working on a logit(proportion) scale and that the final vertical scale is Log10(percentage). (* some test data *) data = Table[Null, {i, 4}]; data[[1]] = {{1, -5.8}, {2, -5.4}, {3, -0.8}, {4, -0.2}, {5, 4.6}, {1, -6.4}, {2, -5.6}, {3, -0.7}, {4, 0.04}, {5, 1.0}, {1, -6.8}, {2, -4.7}, {3, -1.