Programmatically creating multivariate functions in Mathematica

前端 未结 3 1792
别那么骄傲
别那么骄傲 2021-01-24 10:29

This is a split from discussion on earlier question.

Suppose I need to define a function f which checks if given labeling of a graph is a proper coloring. In other words

相关标签:
3条回答
  • 2021-01-24 10:58

    I'm still puzzled by the difficulties that you seem to be having. Here's a function which checks that no 2 consecutive elements in a list are identical:

    f[l_List] := Length[Split[l]] == Length[l]
    

    No trouble with Part, no error messages for the simple examples I've tried so far including OP's 'test' cases. I also contend that this is more concise and more readable than either of the other approaches seen so far.

    0 讨论(0)
  • 2021-01-24 11:00

    There are a couple other solutions to this kind of problem which don't require you to use Hold or ReleaseHold, but instead rely on the fact that Function already has the HoldAll attribute. You first locally "erase" the definitions of Part using Block, so the expression you're interested in can be safely constructed, and then uses With to interpolate that into a Function which can then be safely returned outside of the Block, and also uses the fact that Slot doesn't really mean anything outside of Function.

    Using your example:

    coloringChecker[edges_List] := 
     Block[{Part},
      With[{body = And @@ Table[#[[First@edge]] != #[[Last@edge]], {edge, edges}]},
       body &]]
    

    I don't know if this is all that much less cumbersome than using Hold, but it is different.

    0 讨论(0)
  • 2021-01-24 11:06

    Here's something. When nothing else is working, Hold and rules can usually get the job done. I'm not sure it produces the correct results w.r.t. your graph-coloring question but hopefully gives you a starting place. I ended up using Slot instead of named variable because there were some scoping issues (also present in my previous suggestion, x$ vs. x) when I used a named variable that I didn't spend the time trying to work around.

    g[edges_] := 
     With[{ors = (Hold @@ edges) /. {a_, b_} :> #[[a]] == #[[b]]},
      Function[!ors] /. Hold -> Or
      ]
    
    In[90]:= f = g[GraphData[{"Path", 3}, "EdgeIndices"]]
    Out[90]= !(#1[[1]] == #1[[2]] || #1[[2]] == #1[[3]]) &
    
    In[91]:= f[{1, 2, 3}]
    Out[91]= True
    
    In[92]:= f[{1, 1, 2}]
    Out[92]= False
    

    I feel like it lacks typical Mathematica elegance, but it works. I'll update if I'm inspired with something more beautiful.

    0 讨论(0)
提交回复
热议问题