25 | Ways to Apply Functions |
When you write f[x] it means “apply the function f to x”. An alternative way to write the same thing in the Wolfram Language is f@x.
f@x is the same as f[x]:
In[1]:=
data:image/s3,"s3://crabby-images/b2e95/b2e95d49071711f3209f8326402b16f84304236d" alt=""
Out[1]=
data:image/s3,"s3://crabby-images/04966/0496647243fcb6ccc456c327754e320275509a3a" alt=""
It’s often convenient to write out chains of functions using @:
In[2]:=
data:image/s3,"s3://crabby-images/390cd/390cd1a237b35365245300da30dbb916c6eefd6b" alt=""
Out[2]=
data:image/s3,"s3://crabby-images/5cab3/5cab376afe0ee66ea5b748574e069f304d5ef6df" alt=""
Avoiding the brackets can make code easier to type, and read:
In[3]:=
data:image/s3,"s3://crabby-images/4d1e9/4d1e9704568467321741809eb610455f33c262ca" alt=""
Out[3]=
data:image/s3,"s3://crabby-images/7ada2/7ada233bb8db86d16e84fecd087c81f0e20cd166" alt=""
There’s a third way to write f[x] in the Wolfram Language: as an “afterthought”, in the form x//f.
Apply f “as an afterthought” to x:
In[4]:=
data:image/s3,"s3://crabby-images/81d17/81d176e9dcfbfc1343b28c7670907c71bf201757" alt=""
Out[4]=
data:image/s3,"s3://crabby-images/5abcb/5abcb50eff94388b20063f3a3278e85971d588d8" alt=""
You can have a sequence of “afterthoughts”:
In[5]:=
data:image/s3,"s3://crabby-images/e89aa/e89aad08912fa2600ea9425261c3c5140701759b" alt=""
Out[5]=
data:image/s3,"s3://crabby-images/ecee6/ecee6b4ff36d525eb351ce24d5d45a5b69c00da5" alt=""
In[6]:=
data:image/s3,"s3://crabby-images/afbd2/afbd24410655cb5c8e68fa5678acf5c663e7e775" alt=""
Out[6]=
data:image/s3,"s3://crabby-images/dd231/dd23174af061800a859302b9154f81295065f8d9" alt=""
Apply numerical evaluation “as an afterthought”:
In[7]:=
data:image/s3,"s3://crabby-images/419c4/419c43f1014f7445f96ddb67001e424d32f06a1d" alt=""
Out[7]=
data:image/s3,"s3://crabby-images/7e553/7e553491824983e7acb8ef9e7b995be7b3ad2470" alt=""
In working with the Wolfram Language, a powerful notation that one ends up using all the time is /@, which means “apply to each element”.
Apply f to each element in a list:
In[8]:=
data:image/s3,"s3://crabby-images/fd7d6/fd7d68df56ee20304e8996e268562400daf769e1" alt=""
Out[8]=
data:image/s3,"s3://crabby-images/41e06/41e06621900d9acef4e0cafcd4cb69909299d1e9" alt=""
f usually would just get applied to the whole list:
In[9]:=
data:image/s3,"s3://crabby-images/b13ad/b13adea3163cdcb2cbfac1dfba1b49c273039569" alt=""
Out[9]=
data:image/s3,"s3://crabby-images/70933/70933c265c7522dc43c2f66ace25c70675742fda" alt=""
Framed is a function that displays a frame around something.
Display x framed:
In[10]:=
data:image/s3,"s3://crabby-images/aa67a/aa67ab55a726d1d45e800219d5c5a69b06dab705" alt=""
Out[10]=
data:image/s3,"s3://crabby-images/46b54/46b54d522df6380f55497fc00d139dcb05c27e69" alt=""
Applying Framed to a list just puts a frame around the whole list.
Apply Framed to a whole list:
In[11]:=
data:image/s3,"s3://crabby-images/a4b4e/a4b4e770a57163189cccb74e86bd185fc2b760a3" alt=""
Out[11]=
data:image/s3,"s3://crabby-images/76ead/76ead4651b2ad405882c1eb8f3bb69323179071b" alt=""
@ does exactly the same thing:
In[12]:=
data:image/s3,"s3://crabby-images/807b7/807b7bcf0babe9fb9fd1ce4140a4abea8e514775" alt=""
Out[12]=
data:image/s3,"s3://crabby-images/35fa4/35fa4e6697d8c3d39f22ee2d4befba22ada42200" alt=""
In[13]:=
data:image/s3,"s3://crabby-images/0cb80/0cb80072a92749eeffe290607b1960e004b3dc02" alt=""
Out[13]=
data:image/s3,"s3://crabby-images/661da/661da84bfcdc8391fe8e115ae4156bfa4692c85f" alt=""
The same thing works with any other function. For example, apply the function Hue separately to each number in a list.
In[14]:=
data:image/s3,"s3://crabby-images/0617a/0617a16d863d11b8982cff32cc4332274629af86" alt=""
Out[14]=
data:image/s3,"s3://crabby-images/6e735/6e735ef3a8710b9378b7b9fb8faaff1397bf8b49" alt=""
Here’s what the /@ is doing:
In[15]:=
data:image/s3,"s3://crabby-images/0a1fa/0a1fad859cba1869a5a250a4926876f948757b19" alt=""
Out[15]=
data:image/s3,"s3://crabby-images/74fc9/74fc9bcd0088fe9c1fd964b63c0bc2c428fc6a43" alt=""
It’s the same story with Range, though now the output is a list of lists.
In[16]:=
data:image/s3,"s3://crabby-images/f4b60/f4b6000dbeb3cb36252f75996bc27abb4950346f" alt=""
Out[16]=
data:image/s3,"s3://crabby-images/539c8/539c8c742672b08d523556ef0052b627173c8432" alt=""
Here’s the equivalent, all written out:
In[17]:=
data:image/s3,"s3://crabby-images/7ce31/7ce31e1bed12428ad13c2b05494b3ac7ac23f18f" alt=""
Out[17]=
data:image/s3,"s3://crabby-images/b3bea/b3bea15880f6fa55fdce76fd2067d8227e564a38" alt=""
Given a list of lists, /@ is what one needs to do an operation separately to each sublist.
Apply PieChart separately to each list in a list of lists:
In[18]:=
data:image/s3,"s3://crabby-images/bb691/bb6917d90b424c01f4cb194e782f957cd2acf46d" alt=""
Out[18]=
data:image/s3,"s3://crabby-images/345c4/345c4d8a251a23c87192b821527bc0ac907105c8" alt=""
Apply Length to each element, getting the length of each sublist:
In[19]:=
data:image/s3,"s3://crabby-images/86bc3/86bc38780071d9cb351cb39980680bd9e7a8b224" alt=""
Out[19]=
data:image/s3,"s3://crabby-images/d393b/d393b1476607d9d245bae32b6bd099e6eeec80ac" alt=""
Applying Length to the whole list just gives the total number of sublists:
In[20]:=
data:image/s3,"s3://crabby-images/27f1c/27f1c21a572da78ba7ae0d7bc48e9dd0ac926032" alt=""
Out[20]=
data:image/s3,"s3://crabby-images/7434d/7434db75577102e31e2ac7f48552e06880227876" alt=""
Apply Reverse to each element, getting three different reversed lists:
In[21]:=
data:image/s3,"s3://crabby-images/b61cd/b61cd36af5ae5f5633a0fc0b1eadfdbf7eb65b1f" alt=""
Out[21]=
data:image/s3,"s3://crabby-images/e854c/e854c8b4a404633ddf69b86e5efdd333f953a63b" alt=""
Apply Reverse to the whole list, reversing its elements:
In[22]:=
data:image/s3,"s3://crabby-images/74662/746620813467d738610001230752797361df866a" alt=""
Out[22]=
data:image/s3,"s3://crabby-images/59317/5931744cbf09f65b2363389d7e048e09298896b6" alt=""
As always, the form with brackets is exactly equivalent:
In[23]:=
data:image/s3,"s3://crabby-images/eefdd/eefdd2317634d3de8f4f8ab0b9f214a097a91d0f" alt=""
Out[23]=
data:image/s3,"s3://crabby-images/26a74/26a740b8b3f652550d9908ff5cd6634d519ba52b" alt=""
Some calculational functions are listable, which means they automatically apply themselves to elements in a list.
In[24]:=
data:image/s3,"s3://crabby-images/28319/28319aff690fff800e1a6d705db6dd122be8a5f9" alt=""
Out[24]=
data:image/s3,"s3://crabby-images/64ab5/64ab58c6ed6ebcba6da733992730b1e5035f1035" alt=""
The same is true with Prime:
In[25]:=
data:image/s3,"s3://crabby-images/373d0/373d0d3cce730844909540f7c7513b1d67f8fce5" alt=""
Out[25]=
data:image/s3,"s3://crabby-images/56bd1/56bd1ae25ed5b4880542bd67dcec15ebfcb254af" alt=""
A function like Graphics definitely isn’t listable.
This makes a single graphic with three objects in it:
In[26]:=
data:image/s3,"s3://crabby-images/63029/630298b12b3b2b92c68dcc9d6a15ef5467d345cf" alt=""
Out[26]=
data:image/s3,"s3://crabby-images/ce7b6/ce7b6cf8d93eba1864a5fd7ea928bc892e72ee72" alt=""
This gives three separate graphics, with Graphics applied to each object:
In[27]:=
data:image/s3,"s3://crabby-images/f5576/f557648c060fc2fd593e23685ea300e143ca7871" alt=""
Out[27]=
data:image/s3,"s3://crabby-images/c1432/c1432fbf1994d588e9cdfdcc8fc56b180593c2ef" alt=""
When you enter f/@{1,2,3}, the Wolfram Language interprets it as Map[f,{1,2,3}]. f/@x is usually read as “map f over x”.
The internal interpretation of f/@{1, 2, 3}:
In[28]:=
data:image/s3,"s3://crabby-images/e665b/e665b316db065e143fcaf060be5fbfb09fcd11d5" alt=""
Out[28]=
data:image/s3,"s3://crabby-images/bafc0/bafc009592273f8f84319bfd10c1ccc5283fed66" alt=""
f@x | equivalent to f[x] | |
x//f | equivalent to f[x] | |
f/@{a,b,c} | apply f separately to each element of the list | |
Map[ f,{a,b,c}] | alternative form of /@ | |
Framed[expr] | put a frame around something |
25.4Make a list of letters of the alphabet, with a frame around each one. »
25.5Color negate an image of each planet, giving a list of the results. »
25.7Binarize each flag in Europe, and make an image collage of the result. »
25.8Find a list of the dominant colors in images of the planets, putting the results for each planet in a column. »
25.9Find the total of the letter numbers given by LetterNumber for the letters in the word “wolfram”. »
Why not always use f@x instead of f[x]?
f@x is a fine equivalent to f[x], but the equivalent of f[1+1] is f@(1+1), and in that case, f[1+1] is shorter and easier to understand.
It comes from math. Given a set {1, 2, 3}, f/@{1, 2, 3} can be thought of as mapping of this set to another one.
Typically “slash slash” and “slash at”.
It’s determined by the precedence or binding of different operators. @ binds tighter than +, so f@1+1 means f[1]+1 not f@(1+1)or f[1+1]. // binds looser than +, so 1/2+1/3//N means (1/2+1/3)//N. In a notebook you can find how things are grouped by repeatedly clicking on your input, and seeing how the selection expands.