Alexander Dymo - IT-клуб Николаева - April 2011 - Ruby: Beaty and the Beast
Upcoming SlideShare
Loading in...5
×
 

Alexander Dymo - IT-клуб Николаева - April 2011 - Ruby: Beaty and the Beast

on

  • 1,427 views

 

Statistics

Views

Total Views
1,427
Slideshare-icon Views on SlideShare
1,324
Embed Views
103

Actions

Likes
5
Downloads
8
Comments
0

4 Embeds 103

http://itclub.mk.ua 90
https://twitter.com 8
http://a0.twimg.com 3
http://webcache.googleusercontent.com 2

Accessibility

Categories

Upload Details

Uploaded via as OpenOffice

Usage Rights

© All Rights Reserved

Report content

Flagged as inappropriate Flag as inappropriate
Flag as inappropriate

Select your reason for flagging this presentation as inappropriate.

Cancel
  • Full Name Full Name Comment goes here.
    Are you sure you want to
    Your message goes here
    Processing…
Post Comment
Edit your comment
  • scrum, Тим Евграшин
  • 5-7 строк в день на человека
  • однострочные декларации необязательный return читабельность знакомый синтаксис (нет культурного шока) символы

Alexander Dymo - IT-клуб Николаева - April 2011 - Ruby: Beaty and the Beast Alexander Dymo - IT-клуб Николаева - April 2011 - Ruby: Beaty and the Beast Presentation Transcript

  • Ruby: Beauty and the Beast Александр Дымо IT-клуб Николаева www.acunote.com
  • О чем речь? Опыт разработки на Ruby (+Rails) веб-приложения Acunote www.acunote.com
    • разрабатывается с 2006-го года
    • 29 000 строк кода
    • 45 500 строк тестов
    • ~12000 зарегистрированных организаций
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Язык Ruby: Что понравилось Меньше кода – больше кислорода! ~3-4 разработчика * ~5 лет = 29k строк???
  • Язык Ruby: Что понравилось Меньше кода – больше кислорода! 400 строк на Ruby == 6000 на C++
  • Язык Ruby: Что понравилось Уж больно хорошо все выглядит! module SearchLanguage class Parser class ParseError < StandardError; end def self. parse (text) parser = Parser. new parser. parse (text) end private : def parse_search_query parse_and_expression end def parse_and_expression match_spaces node = ExpressionNode. new (:and) return node if empty_buffer? node. expressions << parse_or_expression while match_AND and !empty_buffer? and !end_of_subexpression? node. expressions << parse_or_expression end node end end end
  • Язык Ruby: Что понравилось Методы с ! и ? [1,2,3] . include? 2 &quot;foo bar&quot; . gsub ( &quot;o&quot; , &quot;a&quot; ) &quot;foo bar&quot; . gsub! ( &quot;o&quot; , &quot;a&quot; )
  • Язык Ruby: Что понравилось Методы с ! и ? class Foo def transform! #... end def transformed? #... end end
  • Язык Ruby: Что понравилось Элегантный подход к getters/setters class Foo attr_accessor :foo end class Foo def foo @foo end def foo=(value) @foo = value end end f = Foo.new f.foo = 2 # f.foo=(2) puts f.foo
  • Язык Ruby: Что понравилось Блоки def do_smth(&block) puts yield(10) puts yield(20) end do_smth { |value| value + 5 } > 15 > 25
  • Язык Ruby: Что понравилось Возможность встроиться куда угодно class Time def humanize strftime(&quot;%a %b %d, %Y %H:%M&quot;) end end puts Time.now.humanize > Thu Apr 21, 2011 13:37 ...мы дополнили 28 различных классов
  • Язык Ruby: Что понравилось Возможность встроиться куда угодно class BigDecimal alias_method :eq_without_boolean_comparison, :== def eq_with_boolean_comparison(other) if [FalseClass, TrueClass].include? other.class return false end eq_without_boolean_comparison(other) end alias_method :==, :eq_with_boolean_comparison end
  • Язык Ruby: Что понравилось Правильный способ встраивания/расширения class Foo def do_smth puts &quot;Hello World&quot; end end class Foo def do_smth_with_decoration puts &quot;--------------&quot; do_smth_without_decoration puts &quot;--------------&quot; end end f = Foo.new f.do_smth > -------------- > Hello World > --------------
  • Язык Ruby: Что понравилось Возможность переписать все что угодно class Foo { private: void doSmth() { std::cout << &quot;Hello World&quot; } }; ... class Foo def doSmth puts &quot;Hi&quot; end end f = Foo.new f.doSmth > Hi
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Язык Ruby: Что не понравилось Perl'измы в синтаксисе x = false or true puts x #???
  • Язык Ruby: Что не понравилось Perl'измы в синтаксисе x = false or true (x = false) or true puts x > false
  • Язык Ruby: Что не понравилось Perl'измы в синтаксисе x = false || true x = (false || true) puts x > true
  • Язык Ruby: Что не понравилось Perl'измы в синтаксисе x = 2 if false or true puts x > ?
  • Язык Ruby: Что не понравилось Perl'измы в синтаксисе x = 2 if false or true x = 2 if (false or true) puts x > 2
  • Язык Ruby: Что не понравилось Python'измы в синтаксисе class Foo attr_accessor :x def do_smth x = 10 end end f = Foo.new f.x > nil #WTF?
  • Язык Ruby: Что не понравилось Python'измы в синтаксисе class Foo attr_accessor :x def do_smth self. x = 10 end end f = Foo.new f.x > 10
  • Язык Ruby: Что не понравилось Переменные экземпляров классов class Foo attr_accessor :x def initialize @x = 10 #или self.x = 10 y = @x #или y = x end end
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Язык Ruby: Интересные особенности Модули: пространства имен или решение проблемы множественного наследования module ActiveRecord module ConnectionAdapters class MysqlAdapter def execute #... end end end end module ActiveRecord module Validations def valid? errors.empty? end end class Base include Validations end end
  • Язык Ruby: Интересные особенности Блоки: передача функций в другие функции или предусловия/постусловия def calculate(&algorithm) yield (2, 2) end result = calculate do |x,y| x + y end def calculate(&expression) return unless can_calculate? result = yield result.good? ? result : nil end result = calculate do 2+2 end
  • Язык Ruby: Интересные особенности method_missing class GoldenFish def method_missing(name, *args, &block) if name == :make_me_rich puts &quot;here you go: $1 000 000&quot; else puts &quot;you want too much!&quot; end end end fish = GoldenFish.new fish.make_me_rich > here you go: $1 000 000 fish.make_my_old_wife_a_supermodel > you want too much!
  • Язык Ruby: Интересные особенности Ассоциации в Rails: реализация с method_missing class User has_many :roles end user = User.find(:first) #метод ассоциации user.roles.find(:all, :conditions => &quot;name = 'Admin'&quot;) #метод массива user.roles.map { |role| role.name }
  • Язык Ruby: Интересные особенности Ассоциации в Rails: реализация с method_missing class AssociationProxy def reload @target = [1,2,3] #just for an example end def method_missing(method, *args, &block) if @target.respond_to? method @target.send(method, *args, &block) else raise NoMethodError end end end end p = AssociationProxy.new p.reload #вызываем свой метод p << 3 #вызываем метод класса Array (у @target)
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Ruby Runtime: Что с ним так и не так
  • Ruby Runtime: Что с ним так и не так Ruby – это красиво, удобно, мощно... но где же подвох?
  • Ruby Runtime: Что с ним так и не так Ruby – это медленно !
  • Ruby Runtime: Что с ним так и не так Ruby – это очень медленно !
  • Ruby Runtime: Что с ним так и не так Апрель 2008 Май 2008 Июнь 2008 Июль 2008 Время обслуживания запроса, % Худший случай: O(n a ) В среднем: O(log n)
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Ruby Runtime: Garbage Collection Сборка мусора – самое плохое что может произойти с вашим Ruby кодом!
  • Ruby Runtime: Garbage Collection Сборка мусора происходит: каждые 8 Мб выделенной памяти выделенная однажды память повторно используется но никогда не возвращается!
  • Ruby Runtime: Garbage Collection в среднем приложении 1 сборка мусора == 100ms выделяете 1Gb? - ожидайте 1024/8 = 128 вызовов GC - потеряете 128 * 0,1 = 12,8 sec!!! - навсегда оставите себе этот гиг
  • Ruby Runtime: Garbage Collection class TestController def index gc_statistics { #дайте-ка мне гиг памяти! 1024.times { &quot;x&quot;*1024*1024 } } end end app.get '/test' > : allocated: 1049602K total in 2052 allocations, GC calls: 146, GC time: 17281 msec
  • Ruby Runtime: Garbage Collection Патч, включающий сбор статистики про GC http://blog.pluron.com/2008/02/memory-profilin.html
  • Ruby Runtime: Garbage Collection Эффект сборки мусора: GC GC Area Burndown 120 0,94 0,65 1,5x Sprint 20 x (1+5) (C) 0,59 0,30 2,0x Sprint 20 x (1+5) 0,70 0,40 1,8x Move Left 0,77 0,46 1,7x Ruby приложение будет в 2 раза медленнее! &quot;Спасибо&quot; сборщику мусора!
  • Ruby Runtime: Garbage Collection Способы бороться с GC в Ruby Enterprise Edition (www.rubyenterpriseedition.com) по-умолчанию RUBY_HEAP_MIN_SLOTS=10000 RUBY_HEAP_SLOTS_INCREMENT=10000 RUBY_GC_MALLOC_LIMIT=8000000 RUBY_HEAP_SLOTS_GROWTH_FACTOR=1.8 оптимальные настройки надо подбирать, но примерно RUBY_HEAP_MIN_SLOTS=1250000 RUBY_HEAP_SLOTS_INCREMENT=100000 RUBY_GC_MALLOC_LIMIT=30000000 RUBY_HEAP_SLOTS_GROWTH_FACTOR=1
  • Ruby Runtime: Garbage Collection Альтернативы? JRuby
  • Ruby Runtime: Garbage Collection Альтернативы? кто хочет написать generational garbage collector для Ruby?
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Инструменты: Редакторы и IDE
    • Emacs (rinari, emacs-rails, ruby-mode для v23)
      • http://oldwiki.rubyonrails.org/rails/pages/HowToUseEmacsWithRails
    • RubyMine
      • http://www.jetbrains.com/ruby/
    • VIM
      • http://akitaonrails.com/2009/01/04/rails-on-vim-in-english
    • Textmate
      • http://macromates.com/
    • KDevelop, Aptana, Netbeans, etc, etc...
  • Инструменты: Редакторы и IDE Ruby – проклятье разработчиков IDE def foo(arg) arg. end foo(2) foo(Class.new)
  • Инструменты: Редакторы и IDE Ruby – проклятье разработчиков IDE class Foo 10.times do |i| define_method &quot;great_method_#{i}&quot; .to_sym do #... end end end f = Foo.new f.
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Инструменты: Тестирование TDD/BDD: Test::Unit, RSpec, Cucumber, etc. Continuous Integration: CruiseControl (http://cruisecontrol.sourceforge.net/)
  • Инструменты: Тестирование Не спешите выбрасывать Test::Unit, он еще послужит Обычно тестируют так: def test_file_tree tree = FilesTree.new( &quot;/home/user&quot; ) assert tree assert_equal &quot;.&quot; , tree[0], assert_equal &quot;..&quot; , tree[1], assert_equal &quot;dir1&quot; , tree[2] assert_equal &quot;file1&quot; , tree[2][0] assert_equal &quot;file2&quot; , tree[3] end
  • Инструменты: Тестирование Не спешите выбрасывать Test::Unit, он еще послужит: мечтают о таком (cucumber) scenario : sample file tree When I go to /home/user Then I see current directory, top directory, dir1 There's a file2 in current directory There's a file1 in dir1 directory end
  • Инструменты: Тестирование Не спешите выбрасывать Test::Unit, он еще послужит: но лучше всего так (Test::Unit): def test_file_tree tree = FilesTree.new( &quot;/home/user&quot; ) assert_equal serialize_tree(tree), <<-END . .. dir1 file1 file2 END end
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Инструменты: Производительность Медленный Ruby Что делать? Профилировать с Ruby-Prof Переписывать на C Переписывать на SQL
  • Инструменты: Производительность Ruby-Prof (http://ruby-prof.rubyforge.org/) Всегда скажет что не так: %self total self child calls name 8.39 0.54 0.23 0.31 602 Array#each_index 7.30 0.41 0.20 0.21 1227 Integer#gcd 6.20 0.49 0.17 0.32 5760 Timecell#date 5.11 0.15 0.14 0.01 1 Magick::Image#to_blob gem install ruby-prof KCachegrind для визуализации результатов http://kcachegrind.sourceforge.net
  • Инструменты: Производительность
  • Инструменты: Производительность До какой степени оптимизировать? Пока не будет так: %self total self child calls name 1.42 0.39 0.25 0.14 1033 Range#each-1 0.85 0.43 0.15 0.28 2221 Array#each-1 0.74 0.13 0.13 0.00 113447 Array#[] 0.40 0.07 0.07 0.00 5008 Hash#initialize_copy 0.40 0.07 0.07 0.00 45467 Hash#[]= 0.40 0.13 0.07 0.06 37784 Hash#[] 0.17 0.15 0.03 0.12 176 Array#collect 0.17 0.08 0.03 0.05 22371 String#gsub 0.17 0.03 0.03 0.00 44881 Array#<< 0.17 0.03 0.03 0.00 15936 String#== 0.17 0.75 0.03 0.72 1218 Array#each 0.17 0.03 0.03 0.00 37274 PGresult#getvalue 0.11 0.18 0.02 0.16 137 Array#collect! 0.11 0.02 0.02 0.00 3227 User#id
  • Инструменты: Производительность Что нужно помнить при профилировке Ruby приложений? не забывать профилировать память в среднем приложении 1 сборка мусора == 100ms выделяете 1Gb? - ожидайте 1024/8 = 128 вызовов GC - потеряете 128 * 0,1 = 12,8 sec!!!
  • Инструменты: Производительность Что нужно помнить при профилировке Ruby приложений? выключать GC (GC.disable), иначе можно увидеть: class Foo def do_smth return &quot;x&quot; * 1024 # займем 1Kb памяти end end smth = &quot;x&quot; * 7999999 # займем почти 8Mb памяти Foo.new.do_smth # здесь произойдет GC -------------------------- %self calls name 0.10 1 Foo#do_smth # do_smth такой медленный??? 0.01 2 String#*
  • Инструменты: Производительность Переписывать на C не такая уж плохая идея: date-performance github.com/rtomayko/date-performance ускоряет операции с датами в 13 раз! ускоряет приложения в 1.5 раза monkeysupport http://github.com/burke/monkeysupport ActiveSupport переписанный на C
  • Инструменты: Производительность Переписывать Rails приложение на SQL тоже неплохая идея: – иногда ускорение в 5х – иногда делает возможным то, что невозможно в Ruby операции над большим кол-вом (1млн) объектов – детали в моей презентации &quot;Быстрее, Выше, SQL'нее&quot; http://www.slideshare.net/adymo/alexander-dymo-barcamp-2009-faster-higher-sql-2644467
  • Инструменты: Производительность Почему все так плохо? нет реальных приложений, на которых Ruby (и Rails) core teams могли бы тестировать производительность некому написать generational GC никто не профилирует память
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Инструменты: Деплоймент monit скрипты мониторинга Отличительные особенности: - скорость ограничена CPU - нужно много памяти VPS'ы могут не подходить! Хорошо работает - capistrano - rinda unicorn unicorn unicorn unicorn unicorn nginx rinda (s) rinda (w)
  • О чем речь?
    • Язык Ruby
      • Что понравилось
      • Что не понравилось
      • Интересные особенности
    • Ruby Runtime
      • Что с ним так и не так
      • Garbage Collection
    • Инструменты
      • Редакторы и IDE
      • Тестирование
      • Профилировка и производительность
      • Деплоймент
    • Какой Ruby выбрать?
  • Какой Ruby выбрать?
  • Какой Ruby выбрать? Производительность различных версий Ruby, %
  • Спасибо за внимание! Александр Дымо Director of Engineering [email_address]