Effective Ruby

杰瑞发布于2025-03-04

48 Specific ways to write better ruby

26. 限制retry次数,retry间隔指数增加:
retries = 0; begin service.update(record); rescue VendorDeadlockError ; raise if retries >= 3 ; retries += 1; logger.warn("API failure: #{e}, retrying...");sleep(5 ** retries); retry; end
27. throw is better than raise to jump out 作用域:
loop do {... raise(..) if ...}; throw(:jump, [character, color]) 传参数到上层,catch处理。简单方法控制程序结果。return, catch和throw;
28. 模块和类的钩子方法:
Ruby的hook在类和模块的级别实现元编程。在类和模块中定义方法-单例方法。混合引入模块时,ruby都会调用钩子方法included or extended;这个钩子可以看作是一个,通知通知当前模块即将要被扩展到另一个类中。代理的方法只是重定向了。interited; method_added method_removed method_undefined 实例方法;singleton_method_added singleton_method_remove singleton_mothod_undefined 可以被用于模块方法或类方法。这些方法只接受一个symbol ,方法名; Triggers singleton_method_removed(:hello) class << self; remove_mothod(hello);end
29.钩子方法中执行super方法:
def self.inherited(subclass) { super; handlers << subclass};
30.define_method or method_missing:
如果没有找到任何方法,method_missing就会被执行;但是因为又super的原因,这里会有迷惑。def method_missing(name, *args, &block) if @hash.respond_to?(name) @hash.send(name, *args, &block) else super end end; Hash.public_instance_methods(false).each do |name| define_method(name) do |*args, &block| @hash.send(name, *args, &block) end; h.public_methods(false).sort.take(5); 实现了Hash的方法。
AuditDecorator @logger = Logger.new($stdout); private def method_messing(name, *arg, &block) @logger.info("calling '#{name}' on #{@object.inspect}"); @object.send(name, *args, &block); define_method更加适合做这个了。 mod=Module.new do object.public_methods.each do |name| define_method(name) do |*args, &block| @logger.info("") @object.send(name, *args, &block) end end end extend(mod) ; 创建了一个匿名的模块。 —— define_method恢复了内省方法。。 respond_to? and respond_to_missing?
31. eval; instance_eval定义的是单例方法;:
def glass_case_of_emotion x="I'm in a " + __method__.to_s.tr('_',' ') binding; binding 可以获得当前的临时域并把这个临时域封装到Binding对象中作为返回结果。这个指定的上下文是eval方法的第二个参数; eval("x", glass_case_of_emotion); class_eval很像是重新打开类,实际上是被定义在Module里面,only被模块和类使用。= moudle_eval; instance_eval访问实例变量;class_eval定义实例方法。 class Widget attr_accessor(:name, :quantity) def initialize(&block) instance_eval(&block) if block end ;;; w= Widget.new do |widget| widget.name= "Elbow Grease" ; @quantity = 0; end ; instance_exec, class_exec, moudle_exec; only accept 代码块, no string; object.instance_eval("@#{name} = DEFAULT")
module Reset def self.reset_var (object, name) object.instance_exec("@#{name}.to_sym") do |var| const = self.class.const_get(:DEFAULT); instance_variable_set(var, const) end ...
32.猴子补丁Monkey Patching and refinement:
Active Support的问题; module OnlySpace ONLY_SPACE_UNICODE_RE= %r/\A[[:space:]]*\z/
def self.only_space?(str) if str.ascii_only? !str.bytes.any {|b| !32 && !b.between?(9,13)} else ONLY_SPACE_UNICODE_RE === str end end
33. alias method and alias chaining:
为现有方法d一个别名,instance_methods.include?(orig) # make sure name is unique.
34.Proc:
code block is a Proc; 弱Proc,参数可以不一致。强Proc参数数量严格一致。lambda 为强Proc.
func= ->(x) {"Hello #{x}} ; func.call("a","b") #ArgumentError: wrong number of arguments(2 for 1);
def test(&block) block.lambda? end;
35. module的顺序:
include() 类定义方法覆盖; and prepend() 类定义的方法无法覆盖。前置模块。
36.MiniTest:
Ruby标准库require('minitest/autorun');assert() and assert_equal(); require('rake/testtask'); Assertion and refutaion in MiniTest::Assertions;
37. MiniTest的测试需求:
unit testing or spec testing= behavioral specification; 在单元测试基础上封装而来。
describe("when comparing") do ... end before do ...end it("orders correctly") do ..end; before == setup; after = teardown; 如果喜欢需求测试风格,参考RSpec cucumber等。 MiniTest::Expectations
38. Mock模拟对象 define一个method,并模仿mock需要调用的特定对象;:
def alive? echo = Time.now.to_f.to_s response = get(echo) response.success? && response.body ==echo end; private get(echo) url=RUI::HTTP.build(host:@server, path: "/echo/#{echo}") HTTP.get(url.to_s) end
monitor = Monitor.new("example.com"); response = MiniTest::Mock.new ; monitor.define_singleton_method(:get) do |echo| response.expect(:success? , true); response.expect(body, echo); response end assert(monitor.alive?, "should be alive") response.verify end; 用Mock格里外部系统不稳定因素;Mock#verify
39.测试的重要性; fuzzbert and mrproper属性测试;SimpleCov 测试覆盖率 ZenTest监测代码:
尽可能的自动化测试;运行代码才知道代码在干啥;测试又happy path and exception path; fuzz testing and property testing; require('fuzzbert') require('uri') fuzz('URI::HTTP:build') do data("random server names") do FuzzBert::Generators.random end deploy do |data| URI::HTTP.build(host: data, path: '/') end 会持续运行,手动停止。 MrProper; properties("Version") do data([Integer, Integer, Integer]); property("new(str).to_s == str") do |data| str= data.join('.'); assert_equal(str, Version.new(str).to_s); 测试驱动开发里面的测试并不好写,
40.ri rdoc:
ri 读取文档;rdoc 生成文档;rdoc -f darkfish RDoc::Markup
41. IRB and Pry:
.irb_history; .irbrc; irb 里面可以启动新的irb; _ 获取上一个表达式的结果;
42. bundler gem:
bundle init; source("https://rubygems.org");Bundler.require(:production); gem('money', '>=1.0')未来不可预测; gem('money' '>=5.1.0', '<5.2.0')
43. Gem:
Gem::Specification.new do |gem\ gem.add_dependency('money', '>=4.2.0', '<5.2.0') ;悲观版本操作符;
44.垃圾收集器:
GC.stat; 槽<页;标记对象;环境变量调优;
45.Finalizer构建资源安全网:
Finalizer是最后一道防线;
46.Ruby性能分析工具 ruby-prof and stackprof gem:
ruby -rprofile script.rb 诊断性能第一是收集数据; stackprof gem ; objspace 查看对象的info; memory_profiler gem;
47.循环中的对象字面量:
FATal_CODES = %w(F1 F2 F3).map(&:freeze).freeze; def fatal?(errors) errors.any? {|e| FATAL_CODES.include?(e.code)} end 将数组变成常量;减少对象的生成数量。
48.memoization优化模式:
def current_user
@current_user ||= User.find(logged_in_user_id);
def shipped?(order_id) @status ||= begin file = fetch_confirmation_file ; file? parse_confirmation_file(file) : {} end @status[order_id] == :shipped end
返回的是true or false; 而current_user 返回的是变异对象;如果不希望调用者修改缓存的变量,那应该考虑让被记忆化的方法返回冻结对象。