我知道什么是静态,但不确定何时使用它。
静态变量:我只将它用于常量字段。有时一个类中有几十个常量,所以使用静态常量可以节省大量内存。还有其他典型用例吗?
静态方法:我在做关于算法的课程时使用它。例如,提供不同排序算法的类。是否违反 OOP 设计?我认为最好保持这种方式,而不是在需要使用它们的每个类中实现排序算法。我错了吗?有哪些好的用例?
此外,使用静态和非静态字段/方法之间是否存在性能差异?
我知道什么是静态,但不确定何时使用它。
静态变量:我只将它用于常量字段。有时一个类中有几十个常量,所以使用静态常量可以节省大量内存。还有其他典型用例吗?
静态方法:我在做关于算法的课程时使用它。例如,提供不同排序算法的类。是否违反 OOP 设计?我认为最好保持这种方式,而不是在需要使用它们的每个类中实现排序算法。我错了吗?有哪些好的用例?
此外,使用静态和非静态字段/方法之间是否存在性能差异?
您正在描述使用静态的情况,但这并不能从根本上解释为什么要使用静态与非静态 - 它们不仅仅是常量和实用方法的关键字。
当某些东西不是静态的(实例)时,这意味着类的每个实例都有它的实例。每一个都可以独立改变。
当某些东西是静态的时,这意味着类的所有实例只有一个副本,因此从任何位置更改它都会影响所有其他位置。
静态变量/方法通常使用较少的内存,因为它们只有一个副本,无论您拥有多少类实例。如果使用得当,静力学在面向对象的设计中非常适用。
如果您有一个只需要一个实例的方法/变量(例如常量或实用程序方法),那么只需将其设为静态即可。理解虽然使方法静态意味着它不能被覆盖。因此,如果您有一个要在子类中覆盖的方法,请不要将其设为静态。
一般的经验法则是 - 如果您只需要一份副本,请将其设为静态。如果您需要每个实例的副本,请使其成为非静态的。
还有其他典型用例吗?
全局变量
是否违反 OOP 设计?
不准确,关键是静态方法是无状态的,因为您不需要类的特定实例。我最喜欢的方法是实用方法(如 Apache Commons)。但是您可能知道,某些方法可能更好地放置为类成员而不是静态的。
一旦您不能覆盖这些方法或用模拟实现替换,静态方法也会使类的可测试性变得更加困难。
性能差异?
谷歌有一个性能 Android 推荐,上面写着“更喜欢静态而不是虚拟”:
http://developer.android.com/training/articles/perf-tips.html#PreferStatic
我不确定JVM是否如此,因为Android使用不同的VM,但鉴于链接指出的原因,这是有道理的:
如果您不需要访问对象的字段,请将您的方法设为静态。调用将快 15%-20% 左右。这也是一种很好的做法,因为您可以从方法签名中看出调用该方法不能改变对象的状态。”
静态变量属于一个类,因此由所有对象共享,因此如果您真的希望共享变量,内存使用量会更少。如果您将变量声明为 public 和 static,那么它对所有人都是全局可用的。
静态方法通常是实用方法,取决于访问修饰符,它们可以在类内或跨类使用。静态实用程序类将有助于再次减少内存使用,因为您无需创建对象来调用这些方法。
除了非常特殊的情况外,我只对常量使用静态(和最终)变量。当然,使用它们是完全有效的。
我倾向于避免使用静态实用程序方法,因为它们使为代码编写单元测试变得更加困难(模拟方法调用的结果)。当你开始开发测试驱动方式时,这个问题就变得很明显了。我更喜欢使用依赖注入和单例bean(尽管这取决于您的需求和情况)。
我个人的经验法则是静态的东西“只是挂在那里”。它们是全局性的(免责声明,不完全正确),但包含在这一特定类中是有意义的。
如果您发现自己反复加载一些重量级对象,则静态字段非常有用。例如,我现在正在处理的项目有两个图像之间的切换。这些是与应用程序一起加载并保存在内存中的静态字段,而不是每次都重新加载它们并让 GC 处理这些混乱。
在static field
所有对象中都有一个值,他们称它为类成员也因为它与类有关。
您可以将静态文件用作实用程序。
一个例子假设我们需要知道我们有多少个实例:
班级Counter
public class Counter {
public static int instanceCount ;
public Counter()
{
instanceCount++;
}
public int getInstanceCount()
{
return instanceCount;
}
}
Counter
在创建了两个Class实例之后。但是它们共享相同的 instanceCount
字段,因为它是一个静态字段,因此 的值 将在and中instanceCount
变得相同。firstCounter
secondCounter
班级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());