Почему в Ruby есть как закрытые, так и защищенные методы? - PullRequest
129 голосов
/ 21 августа 2010

Прежде чем я прочитал эту статью , я думал, что управление доступом в Ruby работает так:

  • public - доступ к любому объекту (например, Obj.new.public_method)
  • protected - доступен только из самого объекта, а также из любых подклассов
  • private - такой же, как защищенный, но метод не существует в подклассах

Однако, похоже, что protected и private действуют одинаково, за исключением того, что вы не можете вызывать private методы с явным получателем (т.е. self.protected_method работает, но self.private_method не делает т).

Какой смысл в этом? Когда существует сценарий, когда вы не хотите, чтобы ваш метод вызывался с явным получателем?

Ответы [ 6 ]

147 голосов
/ 21 августа 2010

protected методы могут быть вызваны любым экземпляром определяющего класса или его подклассов.

private методы могут быть вызваны только из вызывающего объекта. Вы не можете получить доступ к закрытым методам другого экземпляра напрямую.

Вот краткий практический пример:

def compare_to(x)
 self.some_method <=> x.some_method
end

some_method не может быть private здесь. Это должно быть protected, потому что оно необходимо для поддержки явных получателей. Ваши типичные внутренние вспомогательные методы обычно могут быть private, поскольку их никогда не нужно вызывать так.

Важно отметить, что это отличается от способа работы Java или C ++. private в Ruby похож на protected в Java / C ++ в том, что подклассы имеют доступ к методу. В Ruby нет способа ограничить доступ к методу из его подклассов, как вы можете с помощью private в Java.

Видимость в Ruby в большинстве случаев является «рекомендацией», так как вы всегда можете получить доступ к методу, используя send:

irb(main):001:0> class A
irb(main):002:1>   private
irb(main):003:1>   def not_so_private_method
irb(main):004:2>     puts "Hello World"
irb(main):005:2>   end
irb(main):006:1> end
=> nil

irb(main):007:0> foo = A.new
=> #<A:0x31688f>

irb(main):009:0> foo.send :not_so_private_method
Hello World
=> nil
76 голосов
/ 18 марта 2013

разница

  • Любой может вызывать ваши публичные методы.
  • Вы можете вызывать ваши защищенные методы, или другой член вашего класса (или класс-потомок) может вызывать ваши защищенные методы извне. Никто другой не может.
  • Только вы можете вызывать ваши приватные методы, потому что они могут вызываться только с неявным получателем self. Даже вы не можете позвонить self.some_private_method; Вы должны позвонить private_method с self подразумевается. ( iGEL указывает: «Однако есть одно исключение. Если у вас есть закрытый метод age =, вы можете (и должны) вызвать его с помощью self, чтобы отделить его от локальных переменных.»)

В Ruby эти различия являются просто советами от одного программиста к другому. Непубличные методы - это способ сказать: «Я оставляю за собой право изменить это; не зависите от этого». Но вы все равно получаете острые ножницы send и можете вызвать любой метод, который вам нравится .

Краткое руководство

# dwarf.rb
class Dwarf
  include Comparable

  def initialize(name, age, beard_strength)
    @name           = name
    @age            = age
    @beard_strength = beard_strength
  end

  attr_reader :name, :age, :beard_strength
  public    :name
  private   :age
  protected :beard_strength

  # Comparable module will use this comparison method for >, <, ==, etc.
  def <=>(other_dwarf)
    # One dwarf is allowed to call this method on another
    beard_strength <=> other_dwarf.beard_strength
  end

  def greet
    "Lo, I am #{name}, and have mined these #{age} years.\
       My beard is #{beard_strength} strong!"
  end

  def blurt
    # Not allowed to do this: private methods can't have an explicit receiver
    "My age is #{self.age}!"
  end
end

require 'irb'; IRB.start

Тогда вы можете запустить ruby dwarf.rb и сделать это:

gloin = Dwarf.new('Gloin', 253, 7)
gimli = Dwarf.new('Gimli', 62,  9)

gloin > gimli         # false
gimli > gloin         # true

gimli.name            # 'Gimli'
gimli.age             # NoMethodError: private method `age'
                         called for #<Dwarf:0x007ff552140128>

gimli.beard_strength # NoMethodError: protected method `beard_strength'
                        called for #<Dwarf:0x007ff552140128>

gimli.greet          # "Lo, I am Gimli, and have mined these 62 years.\
                           My beard is 9 strong!"

gimli.blurt          # private method `age' called for #<Dwarf:0x007ff552140128>
44 голосов
/ 18 января 2015

Закрытые методы в Ruby:

Если метод является закрытым в Ruby, то он не может быть вызван явным получателем (объектом).Это можно только назвать неявно.Он может неявно вызываться классом, в котором он был описан, а также подклассами этого класса.

Следующие примеры иллюстрируют это лучше:

1) Класс Animalс закрытым методом имя_класса

class Animal
  def intro_animal
    class_name
  end
  private
  def class_name
    "I am a #{self.class}"
  end
end

В этом случае:

n = Animal.new
n.intro_animal #=>I am a Animal
n.class_name #=>error: private method `class_name' called

2) Подкласс Animal под названием Amphibian:

class Amphibian < Animal
  def intro_amphibian
    class_name
  end 
end 

В этом случае:

  n= Amphibian.new
  n.intro_amphibian #=>I am a Amphibian
  n.class_name #=>error: private method `class_name' called

Как видите, закрытые методы можно вызывать только неявно.Они не могут быть вызваны явными получателями.По той же причине частные методы не могут быть вызваны вне иерархии определяющего класса.

Защищенные методы в Ruby:

Если метод защищен в Ruby, то он может быть вызван неявно с помощьюи определяющий класс и его подклассы.Кроме того, они также могут вызываться явным получателем, если получатель является self или того же класса, что и self:

1) Класс Animal с защищенным методом protect_me

class Animal
  def animal_call
    protect_me
  end
  protected
  def protect_me
    p "protect_me called from #{self.class}"
  end  
end

В этом случае:

n= Animal.new
n.animal_call #=> protect_me called from Animal
n.protect_me #=>error: protected method `protect_me' called

2) Класс млекопитающих, унаследованный от класса животных

class Mammal < Animal
  def mammal_call
    protect_me
  end
end 

В этом случае

n= Mammal.new
n.mammal_call #=> protect_me called from Mammal

3) Класс амфибииунаследовано от класса животных (аналогично классу млекопитающих)

class Amphibian < Animal
  def amphi_call
    Mammal.new.protect_me #Receiver same as self
    self.protect_me  #Receiver is self
  end   
end

В этом случае

n= Amphibian.new
n.amphi_call #=> protect_me called from Mammal
             #=> protect_me called from Amphibian  

4) Класс под названием Tree

class Tree
  def tree_call
    Mammal.new.protect_me #Receiver is not same as self
  end
end

В этом случае:

n= Tree.new
n.tree_call #=>error: protected method `protect_me' called for #<Mammal:0x13410c0>
7 голосов
/ 21 августа 2010

Рассмотрим частный метод в Java.Конечно, он может быть вызван из одного и того же класса, но также может быть вызван другим экземпляром того же класса:

public class Foo {

   private void myPrivateMethod() {
     //stuff
   }

   private void anotherMethod() {
       myPrivateMethod(); //calls on self, no explicit receiver
       Foo foo = new Foo();
       foo.myPrivateMethod(); //this works
   }
}

Так что - если вызывающий объект - это другой экземпляр моего того же класса- Мой личный метод на самом деле доступен извне, так сказать.На самом деле это делает его не таким уж закрытым.

В Ruby, с другой стороны, закрытый метод действительно предназначен быть закрытым только для текущего экземпляра.Это то, что обеспечивает удаление опции явного получателя.

С другой стороны, я, безусловно, должен отметить, что в сообществе Ruby довольно часто вообще не используют эти элементы управления видимостью, учитывая, что Ruby дает вамспособы обойти их в любом случае.В отличие от мира Java, тенденция состоит в том, чтобы сделать все доступным и довериться другим разработчикам, чтобы они не облажались.

2 голосов
/ 18 июля 2017

Сравнение контроля доступа Java с Ruby: Если метод объявлен как закрытый в Java, доступ к нему возможен только с помощью других методов в том же классе.Если метод объявлен защищенным, к нему могут обращаться другие классы, которые существуют в том же пакете, а также подклассы класса в другом пакете.Когда метод публичный, он виден всем.В Java концепция видимости управления доступом зависит от того, где находятся эти классы в иерархии наследования / пакета.

Тогда как в Ruby иерархия наследования или пакет / модуль не подходят.Все дело в том, какой объект является получателем метода.

Для частного метода в Ruby его никогда нельзя вызвать с явным получателем.Мы можем (только) вызывать закрытый метод с неявным получателем.

Это также означает, что мы можем вызвать закрытый метод из класса, в котором он объявлен, а также из всех подклассов этогоclass.

class Test1
  def main_method
    method_private
  end

  private
  def method_private
    puts "Inside methodPrivate for #{self.class}"
  end
end

class Test2 < Test1
  def main_method
    method_private
  end
end

Test1.new.main_method
Test2.new.main_method

Inside methodPrivate for Test1
Inside methodPrivate for Test2

class Test3 < Test1
  def main_method
    self.method_private #We were trying to call a private method with an explicit receiver and if called in the same class with self would fail.
  end
end

Test1.new.main_method
This will throw NoMethodError

Никогда нельзя вызывать закрытый метод извне иерархии классов, в которой он был определен.

Защищенный метод можно вызывать с неявным получателем,как частный.Кроме того, защищенный метод может также вызываться явным получателем (только), если получатель является «я» или «объектом того же класса».

 class Test1
  def main_method
    method_protected
  end

  protected
  def method_protected
    puts "InSide method_protected for #{self.class}"
  end
end

class Test2 < Test1
  def main_method
    method_protected # called by implicit receiver
  end
end

class Test3 < Test1
  def main_method
    self.method_protected # called by explicit receiver "an object of the same class"
  end
end


InSide method_protected for Test1
InSide method_protected for Test2
InSide method_protected for Test3


class Test4 < Test1
  def main_method
    Test2.new.method_protected # "Test2.new is the same type of object as self"
  end
end

Test4.new.main_method

class Test5
  def main_method
    Test2.new.method_protected
  end
end

Test5.new.main_method
This would fail as object Test5 is not subclass of Test1
Consider Public methods with maximum visibility

Сводка

Public: открытые методы имеют максимальную видимость

Protected: защищенный метод может вызываться с неявным получателем, как и private.Кроме того, защищенный метод может также вызываться явным получателем (только), если получатель является «собой» или «объектом того же класса».

Private: для закрытого метода в Ruby он никогда не сможетвызываться с явным получателем.Мы можем (только) вызвать приватный метод с неявным получателем.Это также означает, что мы можем вызывать закрытый метод из класса, в котором он объявлен, а также из всех подклассов этого класса.

0 голосов
/ 21 февраля 2015

Частично причина, по которой частные методы могут быть доступны для подклассов в Ruby, заключается в том, что наследование Ruby с классами является тонким приукрашиванием по сравнению с модулем - в Ruby класс, по сути, является своего рода модулем, который обеспечивает наследование и т. Д.

http://ruby -doc.org / ядро-2.0.0 / Class.html

Это означает, что в основном подкласс «включает» родительский класс, так что эффективно функции родительского класса, , включая частные функции , также определены в подклассе.

В других языках программирования вызов метода включает в себя всплывание имени метода вверх по иерархии родительского класса и поиск первого родительского класса, который отвечает на метод. Напротив, в Ruby, пока иерархия родительского класса все еще существует, методы родительского класса непосредственно включены в список методов подкласса, который был определен.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...