草庐IT

javascript - 无法在 javascript 中拥有基于类的对象?

coder 2023-07-05 原文

基于 javascript 原型(prototype)的面向对象编程风格很有趣,但在很多情况下,您需要能够从类创建对象。

例如,在矢量绘图应用程序中,工作区在绘图开始时通常是空的:我无法从现有的“线”创建新的“线”。更一般地说,动态创建对象的每种情况都需要使用类。

我已经阅读了很多教程和“Javascript:好的部分”一书,但在我看来,没有办法定义尊重 1) 封装和 2) 高效成员方法声明的类(我的意思是:成员定义一次并在每个类实例之间共享的方法)。

为了定义私有(private)变量,使用了闭包:

function ClassA()
{
    var value = 1;
    this.getValue = function()
    {
        return value;
    }
}

这里的问题是“ClassA”的每个实例都有它自己的成员函数“getValue”的副本,这是效率不高的。

为了有效地定义成员函数,使用原型(prototype):
function ClassB()
{
    this.value = 1;
}

ClassB.prototype.getValue = function()
{
    return this.value;
}

这里的问题是成员变量“value”是公开的。

我认为这个问题不容易解决,因为需要在对象创建期间定义“私有(private)”变量(以便对象可以访问其创建上下文,而不暴露这些值),而基于原型(prototype)的成员函数定义必须在对象创建之后完成,这样原型(prototype)才有意义(“this.prototype”不存在,我已经检查过)。

或者我错过了什么?

编辑:

首先,感谢您的有趣回答。

我只是想在我的初始消息中添加一点精确度:

我真正想做的是拥有 1)私有(private)变量(封装很好,因为人们只能访问他们需要的东西)和 2)有效的成员方法声明(避免复制)。

似乎简单的私有(private)变量声明实际上只能通过 javascript 中的闭包来实现,这就是我关注基于类的方法的根本原因。如果有一种方法可以使用基于原型(prototype)的方法实现简单的私有(private)变量声明,那对我来说没问题,我不是一个激烈的基于类的方法支持者。

阅读答案后,似乎简单的解决方案是忘记私有(private),并使用特殊的编码约定来阻止其他程序员直接访问“私有(private)”变量......

我同意,我的标题/第一句话对我想在这里讨论的问题具有误导性。

最佳答案

嘘,过来!想听个 secret 吗?

经典继承是一种经过测试和尝试的方法。

经常在 JavaScript 中实现它很有用。类是一个很好的概念,并且拥有在对象之后建模我们的世界的模板很棒。

经典继承只是一种模式。如果您的用例需要这种模式,那么在 JavaScript 中实现经典继承是完全可以的。

原型(prototype)继承侧重于共享功能,这很棒(dinasaur drumstick 很棒),但在某些情况下,您希望共享数据方案而不是功能。这是原型(prototype)继承根本无法解决的问题。

所以,你告诉我类(class)并不像每个人一直告诉我的那样邪恶?

不,他们不是。 JS 社区不赞成的不是类的概念,而是将自己限制在仅用于代码重用的类。就像语言不强制强类型或静态类型一样,它也不强制对象结构的方案。

事实上,语言的幕后巧妙实现can turn你的普通对象类似于经典继承类。

那么,类在 JavaScript 中是如何工作的

好吧,你真的只需要一个构造函数:

function getVehicle(engine){
    return { engine : engine };
}

var v = getVehicle("V6");
v.engine;//v6

我们现在有一个车辆类。我们不需要使用特殊关键字显式定义 Vehicle 类。现在,有些人不喜欢以这种方式做事,而是习惯了更经典的方式。为此,JS 通过执行以下操作提供(愚蠢的 imho)语法糖:
function Vehicle(engine){
     this.engine = engine;
}
var v = new Vehicle("V6");
v.engine;//v6

大多数情况下,这与上面的示例相同。

那么,我们还缺少什么?

继承和私有(private)成员。

如果我告诉你基本的子类型在 JavaScript 中非常简单怎么办?

JavaScript 的打字概念与我们在其他语言中习惯的不同。成为 JS 中某种类型的子类型是什么意思?
var a = {x:5};
var b = {x:3,y:3};

b的类型a 类型的子类型?假设它是 according to (strong) behavioral subtyping (the LSP):

<>
  • Contravariance子类型中的方法参数 - 在这种继承中完全保留。
  • Covariance子类型中的返回类型 - 在这种继承中完全保留。
  • 子类型的方法不应抛出新的异常,除非这些异常本身是父类(super class)型的方法抛出的异常的子类型。 ——完全保存在这种继承中。

  • 还,
  • Preconditions不能在子类型中加强。
  • Postconditions不能在亚型中被削弱。
  • Invariants必须保留在子类型中。
  • 历史规则

  • 所有这些都是再一次,由我们来保持。我们可以随心所欲地保持它们紧密或松散,我们不必这样做,但我们肯定可以。

    所以事实上,只要我们在实现继承时遵守上面的这些规则,我们就完全实现了强行为子类型,这是一种非常强大的子类型(见注释*)。

    >>>>> 结束技术部分

    简单地说,人们还可以看到结构子类型是成立的。

    这将如何适用于我们的 Car例子?
    function getCar(typeOfCar){
        var v = getVehicle("CarEngine");
        v.typeOfCar = typeOfCar;
        return v;
    }
    v = getCar("Honda");
    v.typeOfCar;//Honda;
    v.engine;//CarEngine
    

    不会太难吧?私有(private)成员(member)呢?
    function getVehicle(engine){
        var secret = "Hello"
        return {
            engine : engine,
            getSecret : function() {
                return secret;
            }
        };
    }
    

    见,secret是一个闭包变量。它是完全“私有(private)的”,它的工作方式与 Java 等语言中的私有(private)不同,但不可能从外部访问。

    在函数中使用私有(private)怎么样?

    啊!这是一个很好的问题。

    如果我们想在原型(prototype)上共享的函数中使用私有(private)变量,我们首先需要了解 JS 闭包和函数是如何工作的。

    在 JavaScript 中,函数是一流的。这意味着您可以传递函数。
    function getPerson(name){
        var greeting = "Hello " + name;
        return {
            greet : function() {
                return greeting;
            }
        };
    }
    
    var a = getPerson("thomasc");
    a.greet(); //Hello thomasc
    

    到目前为止一切顺利,但我们可以将绑定(bind)到 a 的函数传递给其他对象!这使您可以进行非常松散的解耦,这非常棒。
    var b = a.greet;
    b(); //Hello thomasc
    

    等待! b 是怎么做的知道这个人的名字叫thomasc吗? That's just the magic of closures.很厉害吧?

    您可能会担心性能。让我告诉你我是如何学会不再担心并开始喜欢优化 JIT 的。

    实际上,拥有这样的函数副本并不是什么大问题。 javascript 中的函数都是关于功能的!闭包是一个很棒的概念,一旦你掌握并掌握了它们,你就会发现它是非常值得的,而对性能的影响真的没有那么大的意义。 JS 一天比一天快,不用担心这些性能问题。

    如果你觉得它很复杂,下面的也很合理。与其他开发人员的共同契约(Contract)只是简单地说“如果我的变量以 _ 开头,请不要碰它,我们都是同意的成年人”。这看起来像:
    function getPerson(name){
        var greeter = {
            greet : function() {
                return "Hello" +greeter._name;
            }
        };
        greeter._name = name;
        return greeter;
    }
    

    或者古典风格
    function Person(name){
        this._name = name;
        this.greet = function(){
           return "Hello "+this._name;
        }
    }
    

    或者,如果您想在原型(prototype)上缓存函数而不是实例化副本:
    function Person(name){
        this._name = name;
    }
    Person.prototype.greet =  function(){
           return "Hello "+this._name;
    }
    

    所以,总结一下:
  • 您可以使用经典的继承模式,它们对于共享数据类型很有用
  • 您还应该使用原型(prototype)继承,它同样有效,而且在您想要共享功能的情况下还可以使用更多。
  • TheifMaster几乎钉住了它。在 JavaScript 中,私有(private)私有(private)确实不是什么大问题,只要您的代码定义了一个清晰的接口(interface),这根本就不会有问题。我们都是成年人 :)

  • *聪明的读者可能会想:嗯?你不是在用历史规则欺骗我吗?我的意思是,属性访问没有被封装。

    我说不,我不是。即使您没有明确地将字段封装为私有(private),您也可以简单地以不访问它们的方式定义您的契约(Contract)。经常像 TheifMaster 建议的那样 _ .此外,我认为只要我们不改变属性访问处理父对象属性的方式,历史规则在很多这样的场景中并不是什么大问题。再次,这取决于我们。

    关于javascript - 无法在 javascript 中拥有基于类的对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17008086/

    有关javascript - 无法在 javascript 中拥有基于类的对象?的更多相关文章

    1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    2. ruby-on-rails - 由于 "wkhtmltopdf",PDFKIT 显然无法正常工作 - 2

      我在从html页面生成PDF时遇到问题。我正在使用PDFkit。在安装它的过程中,我注意到我需要wkhtmltopdf。所以我也安装了它。我做了PDFkit的文档所说的一切......现在我在尝试加载PDF时遇到了这个错误。这里是错误:commandfailed:"/usr/local/bin/wkhtmltopdf""--margin-right""0.75in""--page-size""Letter""--margin-top""0.75in""--margin-bottom""0.75in""--encoding""UTF-8""--margin-left""0.75in""-

    3. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

      在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

    4. ruby-on-rails - 无法使用 Rails 3.2 创建插件? - 2

      我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby​​1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在

    5. ruby - 无法运行 Rails 2.x 应用程序 - 2

      我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

    6. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

      我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

    7. ruby-on-rails - 无法在centos上安装therubyracer(V8和GCC出错) - 2

      我正在尝试在我的centos服务器上安装therubyracer,但遇到了麻烦。$geminstalltherubyracerBuildingnativeextensions.Thiscouldtakeawhile...ERROR:Errorinstallingtherubyracer:ERROR:Failedtobuildgemnativeextension./usr/local/rvm/rubies/ruby-1.9.3-p125/bin/rubyextconf.rbcheckingformain()in-lpthread...yescheckingforv8.h...no***e

    8. Ruby 写入和读取对象到文件 - 2

      好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

    9. ruby - 无法让 RSpec 工作—— 'require' : cannot load such file - 2

      我花了三天的时间用头撞墙,试图弄清楚为什么简单的“rake”不能通过我的规范文件。如果您遇到这种情况:任何文件夹路径中都不要有空格!。严重地。事实上,从现在开始,您命名的任何内容都没有空格。这是我的控制台输出:(在/Users/*****/Desktop/LearningRuby/learn_ruby)$rake/Users/*******/Desktop/LearningRuby/learn_ruby/00_hello/hello_spec.rb:116:in`require':cannotloadsuchfile--hello(LoadError) 最佳

    10. ruby-on-rails - 如果 Object::try 被发送到一个 nil 对象,为什么它会起作用? - 2

      如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象

    随机推荐