假设我想包装一个任意函数来添加类似日志记录(或其他任何东西)。包装器应该是通用的,适用于可以使用任何类型的参数调用的任何f
。
createWrapper= (f) ->
wrapper= ->
# do the logging or whatever I want before I call the original
result= ???
# do whatever after the original function call
result
用法是:
g=createWrapper(f)
现在g
应该以任何方式调用f
,例如
result = g(1,2,3)
应该返回与
相同的内容result = f(1,2,3)
现在我可以在g
使用f
,但是它会调用我的包装函数。
我的问题是:如何调用f
来获得结果?
答案 0 :(得分:2)
答案是
f.apply @, arguments
因此包装器看起来像
createWrapper= (f) ->
->
# do the logging or whatever I want before I call the original
result= f.apply @, arguments
# do whatever after the original function call
result
或者如果您在通话后不想做任何事情
createWrapper= (f) ->
->
# do the logging or whatever I want before I call the original
f.apply @, arguments
或者如果您想更明确一点,可以使用splat。 这样做的好处是参数是一个列表,你可以申请 列出对参数的操作。
createWrapper= (f) ->
(args...) ->
# do the logging or whatever I want before I call the original
result= f.apply @, args
# do whatever after the original function call
result
以下是一些测试code:
createWrapper= (f) ->
(args...) ->
# do the logging or whatever I want before I call the original
result= f.apply @, args
# do whatever after the original function call
result
class cl
i: 10
f: (x) ->
x + @i
getf: (x) ->
(x)=>@f(x)
c =new cl
f1 = c.getf()
g1 = createWrapper(f1)
if f1(1)!=g1(1)
throw new Error("failed f1 #{f1 1} != #{g1 1}")
f2 = (x) -> x+20
g2 = createWrapper(f2)
if f2(1)!=g2(1)
throw new Error("failed f2 #{f2 1} != #{g2 1}")
f3 = (args...) -> "#{args}"
g3 = createWrapper(f3)
if f3(1,2,3)!=g3(1,2,3)
throw new Error("failed f3 #{f3 1,2,3} != #{g3 1,2,3}")
f4 = -> arguments[0]
g4 = createWrapper(f4)
if f4(1)!=g4(1)
throw new Error("failed f4 #{f4 1} != #{g4 1}")
f5 = c.f
g5 = createWrapper(f5)
if f5.apply(c,[1])!=g5.apply(c,[1])
throw new Error("failed f5 #{f5.apply c,[1]} 1= #{g5.apply c,[1]}")
alert("""
Everything is OK:
f1 #{f1 1} == #{g1 1}
f2 #{f2 1} == #{g2 1}
f3 #{f3 1,2,3} == #{g3 1,2,3}
f4 #{f4 1} == #{g4 1}
f5 #{f5.apply c,[1]} == #{g5.apply c,[1]}
""")
答案 1 :(得分:0)
只需运行f()
createWrapper = (f) ->
wrapper= ->
# do the logging or whatever I want before I call the original
result= f()
# do whatever after the original function call
result