6

我知道什么是静态,但不确定何时使用它。

静态变量:我只将它用于常量字段。有时一个类中有几十个常量,所以使用静态常量可以节省大量内存。还有其他典型用例吗?

静态方法:我在做关于算法的课程时使用它。例如,提供不同排序算法的类。是否违反 OOP 设计?我认为最好保持这种方式,而不是在需要使用它们的每个类中实现排序算法。我错了吗?有哪些好的用例?

此外,使用静态和非静态字段/方法之间是否存在性能差异?

4

6 回答 6

21

您正在描述使用静态的情况,但这并不能从根本上解释为什么要使用静态与非静态 - 它们不仅仅是常量和实用方法的关键字。

当某些东西不是静态的(实例)时,这意味着类的每个实例都有它的实例。每一个都可以独立改变。

当某些东西是静态的时,这意味着类的所有实例只有一个副本,因此从任何位置更改它都会影响所有其他位置。

静态变量/方法通常使用较少的内存,因为它们只有一个副本,无论您拥有多少类实例。如果使用得当,静力学在面向对象的设计中非常适用。

如果您有一个只需要一个实例的方法/变量(例如常量或实用程序方法),那么只需将其设为静态即可。理解虽然使方法静态意味着它不能被覆盖。因此,如果您有一个要在子类中覆盖的方法,请不要将其设为静态。

一般的经验法则是 - 如果您只需要一份副本,请将其设为静态。如果您需要每个实例的副本,请使其成为非静态的。

于 2013-07-12T19:00:31.527 回答
3

还有其他典型用例吗?

全局变量

是否违反 OOP 设计?

不准确,关键是静态方法是无状态的,因为您不需要类的特定实例。我最喜欢的方法是实用方法(如 Apache Commons)。但是您可能知道,某些方法可能更好地放置为类成员而不是静态的。

一旦您不能覆盖这些方法或用模拟实现替换,静态方法也会使类的可测试性变得更加困难。

性能差异?

谷歌有一个性能 Android 推荐,上面写着“更喜欢静态而不是虚拟”:

http://developer.android.com/training/articles/perf-tips.html#PreferStatic

我不确定JVM是否如此,因为Android使用不同的VM,但鉴于链接指出的原因,这是有道理的:

如果您不需要访问对象的字段,请将您的方法设为静态。调用将快 15%-20% 左右。这也是一种很好的做法,因为您可以从方法签名中看出调用该方法不能改变对象的状态。”

于 2013-07-12T19:19:22.390 回答
0

静态变量属于一个类,因此由所有对象共享,因此如果您真的希望共享变量,内存使用量会更少。如果您将变量声明为 public 和 static,那么它对所有人都是全局可用的。

静态方法通常是实用方法,取决于访问修饰符,它们可以在类内或跨类使用。静态实用程序类将有助于再次减少内存使用,因为您无需创建对象来调用这些方法。

于 2013-07-12T18:56:58.357 回答
0

除了非常特殊的情况外,我只对常量使用静态(和最终)变量。当然,使用它们是完全有效的。

我倾向于避免使用静态实用程序方法,因为它们使为代码编写单元测试变得更加困难(模拟方法调用的结果)。当你开始开发测试驱动方式时,这个问题就变得很明显了。我更喜欢使用依赖注入和单例bean(尽管这取决于您的需求和情况)。

于 2013-07-12T20:06:43.570 回答
0

我个人的经验法则是静态的东西“只是挂在那里”。它们是全局性的(免责声明,不完全正确),但包含在这一特定类中是有意义的。

如果您发现自己反复加载一些重量级对象,则静态字段非常有用。例如,我现在正在处理的项目有两个图像之间的切换。这些是与应用程序一起加载并保存在内存中的静态字段,而不是每次都重新加载它们并让 GC 处理这些混乱。

于 2013-07-12T18:58:26.190 回答
0

static field所有对象中都有一个值,他们称它为类成员也因为它与类有关。

  • 您可以将静态文件用作实用程序。

    一个例子假设我们需要知道我们有多少个实例:

班级Counter

     public class Counter {


     public static int instanceCount ;

        public Counter()
        {
            instanceCount++;
        }

        public int getInstanceCount()
        {
            return instanceCount;
        }



    }

Counter在创建了两个Class实例之后。但是它们共享相同的 instanceCount字段,因为它是一个静态字段,因此 的值 将在and中instanceCount 变得相同。firstCountersecondCounter

班级main

       Counter firstCounter = new Counter();
       // will print 1
       System.out.println(co.getInstanceCount());
       // will print 2
        Counter secondCounter = new Counter();

        System.out.println(co1.getInstanceCount());
于 2020-05-06T20:53:12.470 回答