Project1

标题: 关于define_method的问题 [打印本页]

作者: taroxd    时间: 2014-1-27 08:44
标题: 关于define_method的问题
本帖最后由 taroxd 于 2014-1-27 11:43 编辑

1. define_method 如何设置block参数?
2. define_method 如何设置参数的默认值?
3. block 里面如何使用 return ? 虽说似乎可以用next,但是……(见下文)

或者伸手伸的更直接点=。=
http://rpg.blue/thread-347871-1-1.html 求修改里面定义的 do_after 方法的 block 形式:
RUBY 代码复制
  1. class Module
  2.   def simple_alias(method, prefix = '_')
  3.     prefix *= 2 while method_defined?("#{prefix}#{method}")
  4.     alias_method "#{prefix}#{method}", method
  5.     "#{prefix}#{method}".to_sym
  6.   end
  7.  
  8.   def do_after(method, &block)
  9.     old_method = simple_alias(method)
  10.     define_method method do |*args|
  11.       send old_method, *args
  12.       instance_exec(*args, &block)
  13.     end
  14.   end
  15. end


1. 不能设置要重定义的方法的block参数(虽然说RGSS里面基本没什么方法要block参数的=。=)
2. 也不能设置默认参数
3. block里面直接return会出错,在alias方法的时候用next感觉怪蛋疼的。另外,在这个设定里面,能不能把block里面的return直接使要定义的方法return?
4. 要是block形式能改好我就把那个诡异的字符串形式扔掉了=。= 字符串形式完全拿来补充block做不到的功能,自己一次都没用过……

脚本渣表示求脚本大神帮助
作者: 余烬之中    时间: 2014-1-27 09:25
没办法在define_method定义的方法上加区块 因为define_method里不能有yield
可以通过过程实现

RUBY 代码复制
  1. class E
  2.   define_method(:me){|x , y=0, z = :block| # 设置参数
  3.     puts x,y # 第一步 输出x,y
  4.     if z.is_a?(Proc) # 如果给出了过程
  5.       r = z.call(1,2) # 呼叫过程 获取返回值
  6.       return if r = -1 # 如果返回的是特定的错误代码 return
  7.     end
  8.     puts y,x # 如果没有问题 输出y,x
  9.   }
  10. end
  11.  
  12. ex = E.new
  13. p = Proc.new{|a,b|
  14.   puts a + b # 过程第一步 返回a+b的值
  15.   next -1 # 返回
  16.   puts a - b # 如果没返回 输出a-b的值
  17. }
  18.  
  19. ex.me(3,4,p)
  20. #=> 3 4 3

作者: taroxd    时间: 2014-1-27 09:55
余烬之中 发表于 2014-1-27 09:25
没办法在define_method定义的方法上加区块 因为define_method里不能有yield
可以通过过程实现

非常谢谢!关于默认参数的问题,似乎是不用修改代码就可以解决了
关于block的问题,虽然演示的没错,但是不是我想要的。(我问题没问好)

我编辑了问题,写上了希望实现的代码。
这种实现传递block的方式似乎很难在这种方法里实现。
如果可以的话,能不能给一个修改的方式?不行就算了,总之一个问题解决了~ Thank you~
作者: taroxd    时间: 2014-1-27 10:51
余烬之中 发表于 2014-1-27 09:25
没办法在define_method定义的方法上加区块 因为define_method里不能有yield
可以通过过程实现


就是一个方便alias的货

alias b a
def a(arg)
  b(arg)
  do_something_using_arg
end

简写成
do_after(:a) {|arg| do_something_using_arg }

但是上面的例子中,如果原来方法a是带有block参数的,那么这个do_after方法肯定会出问题
所以希望请教如何修改


作者: 余烬之中    时间: 2014-1-27 11:35
本帖最后由 余烬之中 于 2014-1-27 11:36 编辑
taroxd 发表于 2014-1-27 10:51
就是一个方便alias的货

alias b a


如果已有定义
RUBY 代码复制
  1. def okok(h,i)
  2.   # do something
  3.   yield(h,i) if block_given?
  4.   # do something
  5. end


然后你再alias
RUBY 代码复制
  1. alias :old_okok :okok
  2. def okok(h,i)
  3.   old_okok
  4.   haha_I_wanna_do_more # without args
  5. end


实现的话 可以是
RUBY 代码复制
  1. class I_AM_A_CLASS
  2.   # 已有的方法 带有区块
  3.   def okok(h, i = 1)
  4.     yield h, i
  5.   end
  6.  
  7.   # do_after的实现
  8.   def do_after(m, &add_on_block) # 方法名 新增的区块
  9.     eval("alias #{"_______#{m}"} #{m}") # 别名
  10.  
  11.     self.class.send :define_method, m do |*aaa, &block|
  12.                                        # *aaa是原方法的参数 &block是原方法的区块
  13.       method("_______#{m}").call(*aaa, &block) # 呼叫原方法
  14.       add_on_block.call # 呼叫新增的区块
  15.     end
  16.   end
  17. end
  18.  
  19. cl = I_AM_A_CLASS.new # 创建实例
  20. # 这里是在类中实现的 建议写在模块中 然后在类中include模块
  21.  
  22. cl.do_after(:okok){p true}
  23. # 重定义方法 a # 区块是新增的语句
  24.  
  25. cl.okok(3){|u,v| p u+v} # 呼叫a 区块是原a的区块
  26. #=> 4,true



作者: taroxd    时间: 2014-1-27 11:51
本帖最后由 taroxd 于 2014-1-27 11:52 编辑
余烬之中 发表于 2014-1-27 11:35
如果已有定义
def okok(h,i)
  # do something


非常感谢提供思路,至少让我知道了define_method 是可以直接带&block的
但是有问题,问题如下:

class I_AM_A_CLASS
  # 已有的方法 带有区块
  def okok(h, i = 1)
    yield h, i
  end

  # do_after的实现
  def do_after(m, &add_on_block) # 方法名 新增的区块
    eval("alias #{"_______#{m}"} #{m}") # 别名

    self.class.send :define_method, m do |*aaa, &block|
                                       # *aaa是原方法的参数 &block是原方法的区块
      method("_______#{m}").call(*aaa, &block) # 呼叫原方法
      add_on_block.call # 呼叫新增的区块
    end
  end
  
  def one
    1
  end
end

cl = I_AM_A_CLASS.new # 创建实例
# 这里是在类中实现的 建议写在模块中 然后在类中include模块

cl.do_after(:okok){p one}
# 重定义方法 a # 区块是新增的语句

cl.okok(3){|u,v| p u+v} # 呼叫a 区块是原a的区块
#=> undefined local variable or method `one' for main:Object (NameError)

不过我应该可以自己摸索出来了,谢啦~
作者: taroxd    时间: 2014-1-27 12:32
标题: 【新人乱改】自用的各种小脚本
余烬之中 发表于 2014-1-27 11:35
如果已有定义
def okok(h,i)
  # do something

RUBY 代码复制
  1. def do_after(method, &add)
  2.     old_method = simple_alias method
  3.     define_method method do |*args, &block|
  4.       send old_method, *args, &block
  5.       instance_exec *args, &add
  6.     end
  7.   end


虽然只是小小的改动,但至少在原来的方法中有block时不会报错了,但是没有把block传入instance_exec *args, &add(否则参数个数不定的时候会出问题),所以在 add 中不能调用方法的 block

总之非常感谢!!!




欢迎光临 Project1 (https://rpg.blue/) Powered by Discuz! X3.1