Mathematica Map question

2019-09-15 03:34发布

问题:

Original question:

I know Mathematica has a built in map(f, x), but what does this function look like? I know you need to look at every element in the list.

Any help or suggestions?

Edit (by Jefromi, pieced together from Mike's comments):

I am working on a program what needs to move through a list like the Map, but I am not allowed to use it. I'm not allowed to use Table either; I need to move through the list without help of another function. I'm working on a recursive version, I have an empty list one down, but moving through a list with items in it is not working out. Here is my first case: newMap[#, {}] = {} (the map of an empty list is just an empty list)

回答1:

I posted a recursive solution but then decided to delete it, since from the comments this sounds like a homework problem, and I'm normally a teach-to-fish person.

You're on the way to a recursive solution with your definition newMap[f_, {}] := {}.

Mathematica's pattern-matching is your friend. Consider how you might implement the definition for newMap[f_, {e_}], and from there, newMap[f_, {e_, rest___}].

One last hint: once you can define that last function, you don't actually need the case for {e_}.

UPDATE:

Based on your comments, maybe this example will help you see how to apply an arbitrary function:

func[a_, b_] := a[b]

In[4]:= func[Abs, x]
Out[4]= Abs[x]

SOLUTION

Since the OP caught a fish, so to speak, (congrats!) here are two recursive solutions, to satisfy the curiosity of any onlookers. This first one is probably what I would consider "idiomatic" Mathematica:

map1[f_, {}] := {}
map1[f_, {e_, rest___}] := {f[e], Sequence@@map1[f,{rest}]}

Here is the approach that does not leverage pattern matching quite as much, which is basically what the OP ended up with:

map2[f_, {}] := {}
map2[f_, lis_] :=  {f[First[lis]], Sequence@@map2[f, Rest[lis]]}

The {f[e], Sequence@@map[f,{rest}]} part can be expressed in a variety of equivalent ways, for example:

  • Prepend[map[f, {rest}], f[e]]
  • Join[{f[e]}, map[f, {rest}] (@Mike used this method)
  • Flatten[{{f[e]}, map[f, {rest}]}, 1]

I'll leave it to the reader to think of any more, and to ponder the performance implications of most of those =)

Finally, for fun, here's a procedural version, even though writing it made me a little nauseous: ;-)

map3[f_, lis_] :=
 (* copy lis since it is read-only *)
 Module[{ret = lis, i},
  For[i = 1, i <= Length[lis], i++,
   ret[[i]] = f[lis[[i]]]
   ];
  ret
  ]


回答2:

To answer the question you posed in the comments, the first argument in Map is a function that accepts a single argument. This can be a pure function, or the name of a function that already only accepts a single argument like

In[1]:=f[x_]:= x + 2
       Map[f, {1,2,3}]
Out[1]:={3,4,5}

As to how to replace Map with a recursive function of your own devising ... Following Jefromi's example, I'm not going to give to much away, as this is homework. But, you'll obviously need some way of operating on a piece of the list while keeping the rest of the list intact for the recursive part of you map function. As he said, Part is a good starting place, but I'd look at some of the other functions it references and see if they are more useful, like First and Rest. Also, I can see where Flatten would be useful. Finally, you'll need a way to end the recursion, so learning how to constrain patterns may be useful. Incidentally, this can be done in one or two lines depending on if you create a second definition for your map (the easier way), or not.

Hint: Now that you have your end condition, you need to answer three questions:

  1. how do I extract a single element from my list,
  2. how do I reference the remaining elements of the list, and
  3. how do I put it back together?

It helps to think of a single step in the process, and what do you need to accomplish in that step.