Почему Ruby имеет как частные, так и защищенные методы?
прежде чем я прочитала в этой статье, Я думал, что контроль доступа в Ruby работает следующим образом:
-
public
- доступ к любому объекту (например,Obj.new.public_method
) -
protected
- доступ возможен только изнутри самого объекта, а также любых подклассов -
private
- то же, что и protected, но метод не существует в подклассах
однако, похоже, что protected
и private
действовать так же, за исключением факт, что вы не можете позвонить private
методы с явным приемником (т. е. self.protected_method
работает, а self.private_method
нет).
какой в этом смысл? Когда есть сценарий, когда вы не хотите, чтобы ваш метод с явной приемник?
6 ответов
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
разница
- любой может вызвать ваши публичные методы.
- вы можете вызвать защищенные методы,или другой член вашего класса (или потомок класса) может вызывать защищенные методы извне. Никто другой не может.
- только вы можете вызвать свои частные методы, потому что они могут быть вызваны только с неявным получателем
self
. даже ты не могу назватьself.some_private_method
; вы должны позвонитьprivate_method
Сself
подразумевается. (Игель указывает: "однако есть одно исключение. Если у вас есть частный метод 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>
частные методы в Ruby:
если метод является частным в Ruby, то он не может быть вызван явным получателем (объектом). Его можно вызвать только неявно. Он может быть вызван неявно классом, в котором он был описан, а также подклассами этого класса.
следующие примеры проиллюстрируют это лучше:
1) класс животных с частным методом class_name
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) подкласс животного под названием амфибия:
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) класс животных с защищенным методом 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>
рассмотрим частный метод в 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, с другой стороны, частный метод действительно должен быть частным только для текущего пример. Это то, что предоставляет удаление опции явного приемника.
сравнение элементов управления доступом Java с Ruby: если метод объявлен частным в Java, он может быть доступен только другими методами в том же классе. Если метод объявлен защищенным, к нему могут обращаться другие классы, существующие в том же пакете, а также подклассы класса в другом пакете. Когда метод является общедоступным, он виден каждому. В Java концепция видимости управления доступом зависит от того, где эти классы лежат в наследование / иерархия пакетов.
тогда как в Ruby иерархия наследования или пакет/модуль не подходят. Все дело в том, какой объект является приемником метода.
для частного метода в Ruby он никогда не может быть вызван с явным приемником. Мы можем (только) вызвать частный метод с неявным приемником.
Это также означает, что мы можем вызвать частный метод из класса, в котором он объявлен, а также всех подклассов этот класс.
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
вы никогда не можете вызвать частный метод вне иерархии классов, где он был определен.
защищенный метод может быть вызван с неявным приемником, как и private. Кроме того, защищенный метод также может быть вызван явным получателем (только), если получатель является "само " или"объектом того же класса".
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 он никогда не может быть вызван с явным приемником. Мы можем (только) вызвать частный метод с неявным приемником. Это также означает, что мы можем вызвать Private метод из класса объявлено в, а также во всех подклассах этого класса.
одна из причин, почему частные методы могут быть доступны на подклассы в Рубин, что Рубин наследования с классами тонкая прикрас за модуль включает в Рубин, класс, по сути, является своего рода модуль, который обеспечивает наследование и т. д.
http://ruby-doc.org/core-2.0.0/Class.html
Это означает, что в основном подкласс "включает" родительский класс, так что эффективно функции родительского класса,в том числе частных функции, также определяются в подклассе.
в других языках программирования вызов метода включает в себя барботирование имени метода вверх по иерархии родительского класса и поиск первого родительского класса, который отвечает на метод. Напротив, в Ruby, пока иерархия родительского класса все еще существует, методы родительского класса непосредственно включены в список методов подкласса.