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/13968/139688adc93f5d3910b7163ac2c66cdd9095dd93" alt=""
Out[1]=
data:image/s3,"s3://crabby-images/7e3c2/7e3c2698bce60faf8427b40e09a39ad905cb1239" alt=""
It’s often convenient to write out chains of functions using @:
In[2]:=
data:image/s3,"s3://crabby-images/ebb9b/ebb9b27b2663b581a797afbbcc3390937a5113d6" alt=""
Out[2]=
data:image/s3,"s3://crabby-images/26ea5/26ea59001a10c98d7c518d97b97faf26d0847b47" alt=""
Avoiding the brackets can make code easier to type, and read:
In[3]:=
data:image/s3,"s3://crabby-images/9d3d3/9d3d30954559bd43961edf7952191c46d46ebd68" alt=""
Out[3]=
data:image/s3,"s3://crabby-images/dfbfb/dfbfb8bdcba8beb6201fc8e14bb9299c5267fee1" 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/13663/1366376105a2ca55a99491fe0f202eae2352bb2c" alt=""
Out[4]=
data:image/s3,"s3://crabby-images/63366/633669545ddd8c7aec82b8bae172a54e3f5f4da8" alt=""
You can have a sequence of “afterthoughts”:
In[5]:=
data:image/s3,"s3://crabby-images/7f055/7f055b9ee69f4343987a77c0070b54f58a3b8b51" alt=""
Out[5]=
data:image/s3,"s3://crabby-images/3715d/3715de836edde24673de014503fdee7269738a16" alt=""
In[6]:=
data:image/s3,"s3://crabby-images/40d29/40d293614fdd8828421a47618857597b16fa27cf" alt=""
Out[6]=
data:image/s3,"s3://crabby-images/c333f/c333f8c87c71c3c8c0dda9c942823b91a4b71e8f" alt=""
Apply numerical evaluation “as an afterthought”:
In[7]:=
data:image/s3,"s3://crabby-images/9ce0c/9ce0c0132e7f46b5b5aa0d2682cee43c5fe1115a" alt=""
Out[7]=
data:image/s3,"s3://crabby-images/63461/63461d1b9cd186dc630c1e32ad9de5d1fb2c2b31" 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/34482/344828a5f0d890107b1f73c75085abc2ed9e0c4e" alt=""
Out[8]=
data:image/s3,"s3://crabby-images/86260/862604ce30c0b9d79e7a7e572b04186547ead42f" alt=""
f usually would just get applied to the whole list:
In[9]:=
data:image/s3,"s3://crabby-images/10028/10028380994d56ca6c5e51b4bb4b6db6b1d480c5" alt=""
Out[9]=
data:image/s3,"s3://crabby-images/e8bba/e8bbae347f6fd370a162d9800d255f4763e10f10" alt=""
Framed is a function that displays a frame around something.
Display x framed:
In[10]:=
data:image/s3,"s3://crabby-images/3c6a1/3c6a1f9a43cc3e13888b217312b629d2ee446098" alt=""
Out[10]=
data:image/s3,"s3://crabby-images/5a9b4/5a9b45bdb3353abfbb46e391d8037f005e04f555" 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/aa384/aa384413902a7534a4fa90238cdcd1ceb7ea21d7" alt=""
Out[11]=
data:image/s3,"s3://crabby-images/841f0/841f02bb4e0226d76a51a202ccd1911a400ccc88" alt=""
@ does exactly the same thing:
In[12]:=
data:image/s3,"s3://crabby-images/4861b/4861b7c86a5657b1ba1f98afd08444c44d1377f5" alt=""
Out[12]=
data:image/s3,"s3://crabby-images/290bf/290bf31a18fefa88cb9c784a2a8ebb4e1d4200fd" alt=""
In[13]:=
data:image/s3,"s3://crabby-images/b76dc/b76dcec4ef4b84d1e3893fe288ef87b42582c636" alt=""
Out[13]=
data:image/s3,"s3://crabby-images/ac1dc/ac1dcfc03c14429e4e2d53b42066e3a5c7eec662" 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/094a7/094a737dad0ae0bf5c13976bd773b9f42916bf21" alt=""
Out[14]=
data:image/s3,"s3://crabby-images/10de7/10de7bf7348ccad4382e8b5a46b5b431646a43e5" alt=""
Here’s what the /@ is doing:
In[15]:=
data:image/s3,"s3://crabby-images/d3ad7/d3ad746ae213acee395c72d34e8e44c838319673" alt=""
Out[15]=
data:image/s3,"s3://crabby-images/2c392/2c392eab9453d47735bc9f4ff8d1bf5268425a29" 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/978fd/978fdededae8675c661a2abf9448ec0199c62abb" alt=""
Out[16]=
data:image/s3,"s3://crabby-images/b681c/b681cf7ad6a2cb5e571b6caaa6dd6b30cc345e2e" alt=""
Here’s the equivalent, all written out:
In[17]:=
data:image/s3,"s3://crabby-images/90fd4/90fd4d822dfba45bc417e02c339f6bacbde49eaa" alt=""
Out[17]=
data:image/s3,"s3://crabby-images/c3e68/c3e68e4bffa424f83014b338329424515d8b10f6" 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/641e5/641e5b6a67c5ca0f52ff75eb9b78db8536fdeacd" alt=""
Out[18]=
data:image/s3,"s3://crabby-images/36343/363433c923361bff99d0da0d16d2f938f0db15df" alt=""
Apply Length to each element, getting the length of each sublist:
In[19]:=
data:image/s3,"s3://crabby-images/c7e88/c7e881147dc130811fc4630baa222038603e7671" alt=""
Out[19]=
data:image/s3,"s3://crabby-images/f1b88/f1b882565f215f50555cf75519cea9061ac6e5bf" alt=""
Applying Length to the whole list just gives the total number of sublists:
In[20]:=
data:image/s3,"s3://crabby-images/0d845/0d8452486cb37e1ff67c3f92ebd967c999077241" alt=""
Out[20]=
data:image/s3,"s3://crabby-images/1468f/1468fa070b165bda3bd1ecd95e5ea0803ccacd95" alt=""
Apply Reverse to each element, getting three different reversed lists:
In[21]:=
data:image/s3,"s3://crabby-images/41d08/41d082b198a2e97995f801680aa71ea415c9dcb8" alt=""
Out[21]=
data:image/s3,"s3://crabby-images/c0506/c05066f3172d94909dbf0ff7ccf02581295b01c5" alt=""
Apply Reverse to the whole list, reversing its elements:
In[22]:=
data:image/s3,"s3://crabby-images/18855/18855d97443ff9c4540a209d2d8362a7d21a888c" alt=""
Out[22]=
data:image/s3,"s3://crabby-images/a361b/a361bcc723eec4e6e1edec6d2042edebcc93fd16" alt=""
As always, the form with brackets is exactly equivalent:
In[23]:=
data:image/s3,"s3://crabby-images/b529a/b529a29e41742f8c4706be930ccfd9127ca3923f" alt=""
Out[23]=
data:image/s3,"s3://crabby-images/e6811/e68119e52b2abcb9ed0c2d734cb54661aa82853e" 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/20af9/20af95734211adc2588044ebd6452d7c7a2d12ce" alt=""
Out[24]=
data:image/s3,"s3://crabby-images/608ab/608abdacba90c74fe8c6e9e96268aafc6d0cfa81" alt=""
The same is true with Prime:
In[25]:=
data:image/s3,"s3://crabby-images/2a5f9/2a5f9ed634b829374627e3894b7d360a39c3010f" alt=""
Out[25]=
data:image/s3,"s3://crabby-images/6b5c9/6b5c91432e7373f07881a76f289db369722dd840" 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/79617/79617547fc6ed767a4312dc2771c89688850f875" alt=""
Out[26]=
data:image/s3,"s3://crabby-images/85ebc/85ebc6bdc8474853ce0ec59e6e04d3eb261ce0cd" alt=""
This gives three separate graphics, with Graphics applied to each object:
In[27]:=
data:image/s3,"s3://crabby-images/e1665/e16651ee18e3f788889d263a1c72036f2ef06fbd" alt=""
Out[27]=
data:image/s3,"s3://crabby-images/46b70/46b705e273c2676639d0beb0b509ece7d90e1e5e" 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/aa26c/aa26c4daa376d78edf79a0a1b31627f8b552dfa2" alt=""
Out[28]=
data:image/s3,"s3://crabby-images/44db6/44db6f3364490a668c05651a909dfa07f146d161" 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.