Javascript-форум (https://javascript.ru/forum/)
-   Общие вопросы Javascript (https://javascript.ru/forum/misc/)
-   -   Особенности прототипного наследования. (https://javascript.ru/forum/misc/78538-osobennosti-prototipnogo-nasledovaniya.html)

alexsm993 29.09.2019 21:10

Особенности прототипного наследования.
 
Всем привет! Пост будет длинный, пожалуйста, дочитайте до конца и подскажите, буду очень благодарен!
Для начала приведу простой пример, на который буду ссылаться в дальнейшем.
let animal = {
    eat: true,
};

let rabbit = {
    __proto__: animal,
};

console.log( rabbit.eat ) // true

Тут всё просто и понятно, прототипом для rabbit'а является объект animal поэтому rabbit, не найдя свойство "eat" у себя, возьмёт его от прототипа. Идём дальше.
Известно, что функция-конструктор (далее ф-к) при вызове new Function()/Function() неявно возвращает объект this, который изначально равен пустому объекту {}.
Также, известно, что ф-к имеет скрытое свойство prototype, значением которого является объект с полем конструктор. Конструктор равен самой ф-к. Таким образом, вызов new Function()/Function() вернет объект, для которого прототипом изначально является ф-к.
Вопрос заключается вот в чём. Имеем следующий код:
function User() {
   
    function sayHi() {
        console.log('hello')
    }
}

let user = new User() // user = {}
user.sayHi() //sayHi не отрабатывает!

Моя логика следующая.
В user на 8 строчке "падает" пустой объект.
На 9 строке мы пытаемся вызвать метод sayHi().
В самом объекте этого метода нет. НО
Если объект не находит у себя поле/метод он идет искать их по цепочки прототипов (что и приведено в начале поста).
У user'a нет такого метода, за ним он идет в прототип, коим является ф-к, в которой этот метод есть. Почему же тогда он его не отрабатывает?
Понятно, что написав this.sayHi() {..} все будет работать, метод упадет сразу в объект при выполнении ф-к.
Но почему не работает принцип прототипного наследование?
Заранее спасибо!

Aetae 29.09.2019 21:24

function sayHi() {} - это просто создание обычной функции в текущей области видимости. К методам и прототипу это никакого отношения не имеет вообще.

Rise 29.09.2019 22:12

Цитата:

Сообщение от alexsm993
вызов new Function()/Function() вернет объект,

Вызов Func() без new не возвращает объект. Не используй для именования слово Function, даже в объяснениях, это встроенный конструктор, как Object, Array и тд, что может ввести в заблуждение.
Цитата:

Сообщение от alexsm993
вернет объект, для которого прототипом изначально является ф-к.

Вернет объект у которого __proto__ равно Func.prototype. А сама ф-к там находится в свойстве __proto__.constructor, Func.prototype.constructor.
function User() {}
console.log(User == User.prototype.constructor);
console.log(User == new User().constructor); // __proto__.constructor

Цитата:

Сообщение от alexsm993
ф-к имеет скрытое свойство prototype, значением которого является объект с полем конструктор.

Оно никакое не скрытое, его можно перезаписать в любой момент и оно всегда есть у функции независимо от вызова. Раньше давно скрытым считалось __proto__, но сейчас и его можно перезаписать напрямую. Например, можешь посмотреть здесь, как в jquery осуществляется подмена прототипа.


Часовой пояс GMT +3, время: 12:52.