草庐IT

java - 为什么拥有静态成员会使语言不那么面向对象?

coder 2024-03-27 原文

我目前正在学习 Scala,我在 Odersky 的 Programming Scala 2nd edition 中看到了这个语句:

one way in which Scala is more object-orientated than Java is that classes in Scala cannot have static members.

我在 Java 或 Scala 方面的经验不足,无法理解这种比较。为什么拥有静态成员会使语言不那么面向对象?

最佳答案

奥德斯基的说法是有道理的,意义重大,但有些人不明白他的意思。

假设在 Java 中你有一个类 Foo 和方法 f:

class Foo {
  int f() { /* does something great */ }
}

您可以编写一个接受 Foo 并在其上调用 f 的方法:

void g(Foo foo) { foo.f(); }

也许有一个类 SubFoo 扩展了 Foo; g 也致力于此。可以有一整套通过继承或接口(interface)相关的类,它们共享可以与 g 一起使用的事实。

现在让我们将 f 方法设为静态:

class Foo {
  static int f() { /* does something great */ }
}

我们可以将这个新的 Foo 与 g 一起使用吗,也许像这样?

g(Foo);  // No, this is nonsense.

该死的。好的,让我们更改 g 的签名,以便我们可以将 Foo 传递给它并让它调用 f。

糟糕——我们不能。我们不能传递对 Foo 的引用,因为 Foo 不是某个类的实例。在这里评论的一些人对 Foo 对应的 Class 对象感到困惑,但正如 Sotirios 试图解释的那样,Class 对象没有 f 方法并且 Foo 不是该类的实例。 Foo 不是任何东西的实例;它根本不是一个对象。 Foo 的 Class 对象是类 Class 的一个实例,其中包含有关 Foo 的信息(将其视为 Foo 的内部维基百科页面),并且与讨论完全不相关。 “老虎”的维基百科页面不是老虎。

在 Java 中,像 3 和 'x' 这样的“原语”不是对象。它们 Scala 中的对象。为了提高性能,您的程序将在执行期间尽可能使用 JVM 原语 3 和 'x',但在您编写代码的级别上,它们实际上是对象。它们不是 Java 中的对象这一事实对于任何试图编写处理所有数据类型的代码的人来说都是相当不幸的后果——您必须有特殊的逻辑和额外的方法来覆盖原语。如果您曾经看过或编写过那种代码,您就会知道它很糟糕。奥德斯基的说法不是“纯粹主义”;远非如此。

在 Scala 中,没有任何运行时数据不是对象,也没有任何东西可以调用不是对象的方法。在 Java 中,这些陈述都不正确; Java 是一种部分面向对象的语言。在 Java 中,有些东西不是对象,有些方法不是对象。

Scala 的新手通常认为 object Foo 是 Java 静态的一些奇怪的替代品,但这是您需要快速克服的问题。相反,将 Java 的静态方法视为非 OO 的疣,将 Scala 的 object Foo { ... } 视为以下几行:

class SomeHiddenClass { ... }
val Foo = new SomeHiddenClass  // the only instance of it

这里 Foo 是一个,而不是一个类型,它确实是一个对象。它可以传递给一个方法。它可以扩展其他一些类。例如:

abstract class AbFoo { def f:Int }
object Foo extends AbFoo { def f = 2 }

现在,你终于可以说

g(Foo)

的确,类的“伴随对象”是放置类的非实例方法和数据的好地方。但是那个伴随对象是一个对象,所以通常的规则和功能适用。

事实上,在 Java 中,您将此类方法放在非对象上——限制了它们的使用方式——这是一种责任,而不是一种特性。它肯定不是 OO。

关于java - 为什么拥有静态成员会使语言不那么面向对象?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22890024/

有关java - 为什么拥有静态成员会使语言不那么面向对象?的更多相关文章

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

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

  2. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  3. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

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

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

  5. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

    我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

  6. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  7. ruby - 为什么 4.1%2 使用 Ruby 返回 0.0999999999999996?但是 4.2%2==0.2 - 2

    为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返

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

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

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

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

  10. ruby - ruby 中的 TOPLEVEL_BINDING 是什么? - 2

    它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput

随机推荐