Прежде чем я прочитал эту статью , я думал, что контроль доступа в Ruby работает следующим образом:
public
- может быть доступен любой объект (напримерObj.new.public_method
)protected
- можно получить доступ только внутри самого объекта, а также любых подклассовprivate
- то же, что и protected, но метод не существует в подклассах
Однако похоже, что protected
и private
действуют так же, за исключением того факта, что вы не можете вызывать private
методы с явным получателем (т.е. self.protected_method
работает, но self.private_method
не работает).
какой в этом смысл? Когда существует сценарий, когда вы не хотите, чтобы ваш метод вызывался с явным получателем?
ruby
language-design
access-specifier
Кайл Слэттери
источник
источник
Object
было разрешено вызывать частные методы любого другого экземпляраObject
, можно было бы говорить такие вещи, как5.puts("hello world")
.Ответы:
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
источник
private
vsprotected
должен был делать, может ли подкласс наследовать метод, но на самом деле речь идет о том, откуда метод может быть вызван. Благодарность!send
?Различия
self
. Даже ты не можешь позвонитьself.some_private_method
; вы должны коллироватьprivate_method
сself
подразумеваемыми.self
приемник может быть явным,self.some_private_method
не допускаются. (Любой другой явный получатель по-прежнему запрещен, даже если значение времени выполнения такое же, как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>
источник
age=
, вы можете (и должны) вызвать его,self
чтобы отделить его от локальных переменных.gimli.greet
,gimli
не звонящий, а получатель. Вызывающим является «среда выполнения верхнего уровня», которая на самом деле является специальным экземпляромObject
. Попробуйте это:ruby -e 'p self; p self.class'
Частные методы в Ruby:
Если метод является частным в Ruby, он не может быть вызван явным получателем (объектом). Его можно вызвать только неявно. Он может вызываться неявно классом, в котором он был описан, а также подклассами этого класса.
Следующие примеры лучше проиллюстрируют это:
1) Класс Animal с частным методом 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) Подкласс 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, он может вызываться неявно как определяющим классом, так и его подклассами. Кроме того, они также могут быть вызваны явным получателем, если получатель является самим собой или того же класса, что и сам:
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) Класс амфибий, унаследованный от класса Animal (то же, что и класс млекопитающих)
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 частный метод должен быть приватным только для текущего экземпляра. Это то, что дает удаление опции явного получателя.
С другой стороны, я определенно должен отметить, что в сообществе Ruby довольно распространено вообще не использовать эти элементы управления видимостью, учитывая, что Ruby все равно дает способы их обойти. В отличие от мира Java, существует тенденция делать все доступным и доверять другим разработчикам, чтобы они не облажались.
источник
Частные методы могут быть доступны для подклассов в Ruby, потому что наследование Ruby с классами является тонким сахарным покрытием над Module includes - в Ruby класс, по сути, является своего рода модулем, который обеспечивает наследование и т. Д.
http://ruby-doc.org/core-2.0.0/Class.html
Это означает, что в основном подкласс «включает» родительский класс, поэтому функции родительского класса, включая частные функции , также определены в подклассе.
В других языках программирования вызов метода включает поднятие имени метода вверх по иерархии родительских классов и поиск первого родительского класса, который отвечает на метод. Напротив, в 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
Вы никогда не сможете вызвать частный метод из-за пределов иерархии классов, в которой он был определен.
Защищенный метод может быть вызван с неявным получателем, как и частный. Кроме того, защищенный метод также может быть вызван явным получателем (только), если получатель является «самим собой» или «объектом того же класса».
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: открытые методы имеют максимальную видимость
Защищенный: Защищенный метод может быть вызван с неявным получателем, как и частный. Кроме того, защищенный метод также может быть вызван явным получателем (только), если получатель является «самим собой» или «объектом того же класса».
Закрытый: закрытый метод в Ruby никогда не может быть вызван с явным получателем. Мы можем (только) вызвать частный метод с неявным получателем. Это также означает, что мы можем вызывать частный метод из класса, в котором он объявлен, а также из всех подклассов этого класса.
источник
First Three types of access specifiers and those define thier scope. 1.Public -> Access anywhere out side the class. 2.Private -> Can not access outside the class. 3.Protected -> This Method not access anywhere this method define scope. But i have a solution for this problem for all method how to access explain in depth. class Test attr_reader :name def initialize(name) @name = name end def add_two(number) @number = number end def view_address address("Anyaddress") end private def address(add) @add = add end protected def user_name(name) # p 'call method' @name = name end end class Result < Test def new_user user_name("test355") end end
источник