设为首页收藏本站|繁體中文

Project1

 找回密码
 注册会员
搜索
查看: 2113|回复: 2
打印 上一主题 下一主题

[有事请教] RUBY技术性问题(因为不能算是XP或者VX所以就在这里问了)

[复制链接]

Lv2.观梦者

梦石
0
星屑
250
在线时间
227 小时
注册时间
2009-7-26
帖子
216
跳转到指定楼层
1
发表于 2011-6-22 05:00:10 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

加入我们,或者,欢迎回来。

您需要 登录 才可以下载或查看,没有帐号?注册会员

x
本帖最后由 darkscout3000 于 2011-6-21 16:03 编辑

RUBY如何获得当前方法的调用方法以及其参数?
现有代码
  1. def foo(*params)
  2.   p params
  3.   p caller[1][/`([^']*)'/, 1]
  4. end

  5. def bar1(param1)
  6.   foo(param1, "+1")
  7. end

  8. def bar2(param2)
  9.   foo("-1", param2)
  10. end
复制代码
如果我调用 bar1 方法,它会自动调用 foo 方法
现已知以下代码
  1. p caller[1][/`([^']*)'/, 1]
复制代码
会告诉我是 bar1 方法而不是 bar2 方法调用了 foo
问题是我不知道如何直接获取bar1以及其参数
考虑到我也有可能会调用bar2,我就不能假设正好是第一个参数
同时也考虑到我可能有无数个方法调用foo所以一个一个比较非常费事

有没有什么高级的办法⋯⋯
不要吐槽我⋯⋯

Lv1.梦旅人

梦石
0
星屑
110
在线时间
953 小时
注册时间
2007-4-25
帖子
805
2
发表于 2011-6-22 09:45:55 | 只看该作者
最直观的方法是调用者传递自己的绑定给被调用者:
  1. def foo(a_binding, *params)
  2.   p eval("local_variables", a_binding)
  3. end

  4. def bar1(param1)
  5.   foo(binding, param1, "+1")
  6. end

  7. def bar2(param2)
  8.   foo(binding, "-1", param2)
  9. end

  10. bar2(5) # => 1.8: ["param2"]; 1.9: [:param2]
复制代码
有了调用者的绑定,就可以通过 Kernel#local_variables 这个反射函数来获取局部变量名。方法的参数在局部变量列表的开头,按词法分析器看见局部变量的顺序排列。比如:local_variables[0] 就是第一个参数。如果需要获取参数的值,就再套一层 eval 求值即可(见第三个方法的例子代码)。

另一种方法是在被调用者需要调用者绑定的时候让调用者调用时附带一个 Ruby 块(如果被调用者方法本身就需要块就更好),然后在被调用者里通过 Proc#binding 方法获取调用者的绑定:
  1. def foo(*params, &block)
  2.   p eval("local_variables", block.binding)
  3. end

  4. def bar1(param1)
  5.   foo(param1, "+1") {}
  6. end

  7. def bar2(param2)
  8.   foo("-1", param2) {}
  9. end

  10. bar2(5) # => 1.8: ["param2"]; 1.9: [:param2]
复制代码
如果觉得这种方法太啰嗦,或者是不想更改方法接口,那可以用一个绑定链来跟踪整个调用栈的绑定,只不过在间开销相对就比较大了:

  1. def foo(*params)
  2.   eval("p eval(local_variables[0].to_s)", Binding.caller_binding)
  3. end

  4. def bar1(param1)
  5.   foo(param1, "+1")
  6. end

  7. def bar2(param2)
  8.   foo("-1", param2)
  9. end

  10. class Binding
  11.   @@caller_bindings = [ nil, TOPLEVEL_BINDING ]
  12.   class << self
  13.     def caller_binding
  14.       @@caller_bindings[-2]
  15.     end
  16.     def push_caller_binding(a_binding)
  17.       @@caller_bindings.push(a_binding)
  18.     end
  19.     def pop_caller_binding
  20.       @@caller_bindings.pop
  21.     end
  22.   end
  23. end

  24. set_trace_func lambda { |event, file, line, id, binding, klass|
  25.   return if klass == Binding
  26.   case event
  27.   when 'call'
  28.     Binding.push_caller_binding(binding)
  29.   when 'return'
  30.     Binding.pop_caller_binding
  31.   end
  32. }

  33. bar1 999 # => 999
  34. bar2 888 # => 888
复制代码

点评

紫苏酱依然那么犀利.....o.o  发表于 2011-6-24 00:10

评分

参与人数 1星屑 +600 收起 理由
fux2 + 600 用公费交学费&gt;&lt;

查看全部评分

[email protected]:~> repeat 1 fortune
Matz is nice, so we are nice.
回复 支持 反对

使用道具 举报

Lv2.观梦者

梦石
0
星屑
250
在线时间
227 小时
注册时间
2009-7-26
帖子
216
3
 楼主| 发表于 2011-6-22 23:37:49 | 只看该作者
苏小脉 发表于 2011-6-21 20:45
最直观的方法是调用者传递自己的绑定给被调用者:有了调用者的绑定,就可以通过 Kernel#local_variables 这 ...

首先⋯⋯膜拜下⋯⋯
然后多谢,真是帮了大忙了⋯⋯

这个绑定的概念我还真是头一次听说⋯⋯
话说这种概念从哪儿学到,一般写脚本根本想不到啊⋯⋯



版主如果看到帮我+300分给苏小脉吧,我权限不够⋯⋯
不要吐槽我⋯⋯
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 注册会员

本版积分规则

拿上你的纸笔,建造一个属于你的梦想世界,加入吧。
 注册会员
找回密码

站长信箱:[email protected]|手机版|小黑屋|无图版|Project1游戏制作

GMT+8, 2024-11-15 00:45

Powered by Discuz! X3.1

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表