ChatGPT解决这个技术问题 Extra ChatGPT

何时使用静态方法

我想知道什么时候使用静态方法?假设我有一个带有几个 getter 和 setter 的类,一个或两个方法,并且我希望这些方法只能在类的实例对象上调用。这是否意味着我应该使用静态方法?

例子:

Obj x = new Obj();
x.someMethod();

...或者:

Obj.someMethod(); // Is this the static way?

我比较糊涂!

这是一篇关于该主题的文章:tutorialspoint.com/When-to-use-static-methods-in-Java

L
Lii

一条经验法则:问问自己“调用此方法是否有意义,即使尚未构造任何对象?”如果是这样,它肯定应该是静态的。

因此,在类 Car 中,您可能有一个方法:

double convertMpgToKpl(double mpg)

...这将是静态的,因为人们可能想知道 35mpg 转换为什么,即使没有人构建过 Car。但是这种方法(它设置了一个特定 Car 的效率):

void setMileage(double mpg)

...不能是静态的,因为在构造任何 Car 之前调用该方法是不可想象的。

(顺便说一句,反过来并不总是正确的:您有时可能有一个涉及两个 Car 对象的方法,并且仍然希望它是静态的。例如:

Car theMoreEfficientOf(Car c1, Car c2)

尽管这可以转换为非静态版本,但有些人会争辩说,由于没有哪个 Car 更重要的“特权”选择,因此您不应强制调用者选择一个 Car 作为您将调用该方法的对象。不过,这种情况只占所有静态方法的一小部分。


这里有几个很好的例子。但是,我要补充一点,当您知道某些内容不会在实例之间发生变化时,“静态”通常很有价值。如果是这种情况,我真的会考虑“单一责任原则”,这意味着一个类应该有一个责任,因此只有一个改变的理由。我觉得应该考虑将“ConvertMpgToKpl(double mpg)”函数和类似方法移到他们自己的类中。汽车对象的目的是允许汽车的实例化,而不是提供它们之间的比较。那些应该在课堂之外。
我想我更喜欢方法Car#isMoreEfficientThan(Car)。它的优点是,你在平局中返回哪辆车不是任意的。从方法的标题可以明显看出平局返回的内容。
我也会小心创建一个使用某些外部资源(文件系统、数据库等)的静态方法,这种类型的静态方法会使测试使用方法变得可怕。我个人试图将静态保持在“效用”领域。
事实上,它应该实现为 Comparator
@B1KMusic 当然。我所说的“平局归还哪辆车”的意思是“真实映射到被调用的汽车,而虚假映射到经过的汽车”。这是没有歧义的。
S
Simon Forsberg

仅在以下场景中定义静态方法:

如果您正在编写实用程序类并且它们不应该被更改。如果该方法没有使用任何实例变量。如果任何操作不依赖于实例创建。如果有一些代码可以很容易地被所有实例方法共享,则将该代码提取到静态方法中。如果您确定方法的定义永远不会被更改或覆盖。因为静态方法不能被覆盖。


好点,但如果你想让一个方法静态,它们是要求,而不是制作一个。
@Mohd 关于要求 5:您什么时候可以 100% 确定方法永远不会被更改或覆盖?在您编写静态方法的那一刻,您是否总是无法考虑未知因素?
“实用程序类”很难推理,坏事是迟早一切都开始“看起来像”实用程序(是的,我指的是那个“实用程序”包,它臃肿、不可触碰且测试不佳),并且您的测试用例将需要更多的工作(模拟静态工具是困难的)。首先喜欢对象。
@Mohd 这个答案正是我想要的。我在多线程中使用静态方法遇到了很多问题。您能否详细说明第 2 点、第 3 点(例如,为您竖起 100 个大拇指)
如果您要使用静态变量和方法,我认为应该发明一个“静态类”。
t
tetsuo

使用静态方法有一些正当理由:

性能:如果您希望运行某些代码,并且不想实例化额外的对象来执行此操作,请将其推入静态方法。 JVM 还可以优化静态方法很多(我想我曾经读过 James Gosling 声明你不需要 JVM 中的自定义指令,因为静态方法会一样快,但找不到源 - 因此这可能是完全错误的)。是的,它是微优化,可能不需要。而且我们程序员永远不会仅仅因为它们很酷而做不必要的事情,对吧?

实用性:不要调用 new Util().method(arg),而是调用 Util.method(arg),或者使用静态导入的 method(arg)。更简单,更短。

添加方法:您确实希望 String 类具有 removeSpecialChars() 实例方法,但它不存在(也不应该存在,因为您的项目的特殊字符可能与其他项目的不同),并且您无法添加它(因为 Java 有点理智),所以您创建了一个实用程序类,并调用 removeSpecialChars(s) 而不是 s.removeSpecialChars()。甜的。

纯度:采取一些预防措施,你的静态方法将是一个纯函数,也就是说,它唯一依赖的就是它的参数。数据输入,数据输出。这更易于阅读和调试,因为您无需担心继承问题。您也可以使用实例方法来做到这一点,但编译器会在静态方法方面为您提供更多帮助(通过不允许引用实例属性、覆盖方法等)。

如果你想创建一个单例,你还必须创建一个静态方法,但是......不要。我的意思是,三思而后行。

现在,更重要的是,为什么您不想创建静态方法?基本上,多态性消失了。您将无法覆盖该方法,也无法在接口中声明它(Java 8 之前)。您的设计需要很大的灵活性。此外,如果您需要状态,如果您不小心,最终会遇到很多并发错误和/或瓶颈。


这里列出了很多关于静态何时有用的充分理由。我能想到的另一件事是,为这些方法编写单元测试非常简单
@tetsuo 谢谢!您的解释非常清楚,提供的理由非常合乎逻辑并且很有意义。
而且我们程序员永远不会仅仅因为它们很酷而做不必要的事情,对吧? +1
也就是说,静态方法变成了一个完整的命名函数 stackoverflow.com/questions/155609/…
我同意性能和实用性,但不同意纯度。静态方法可以修改类的静态成员(可能是私有的)。这可能很有用。例如,您可以有一个类似“static synchronized int allocateID() {return idNext++;}”的方法。事实上,就副作用而言,静态方法可以与非静态方法一样纯或不纯。
A
Alfred

阅读 Misko 的文章后,我认为从测试的角度来看,static methods 是不好的。您应该改用 factories(也许使用像 Guice 这样的依赖注入工具)。

我如何确保我只有其中之一

只有一件东西“我如何确保我只拥有一件东西”的问题被很好地回避了。您只在 main 中实例化了一个 ApplicationFactory,因此,您只实例化了所有单例的一个实例。

静态方法的基本问题是它们是过程代码

静态方法的基本问题是它们是过程代码。我不知道如何对程序代码进行单元测试。单元测试假设我可以单独实例化我的应用程序的一部分。在实例化期间,我将依赖项与替换真正依赖项的模拟/友好连接。对于过程式编程,没有什么可以“连接”的,因为没有对象,代码和数据是分开的。


我不明白无法对程序代码进行单元测试的部分。您不只是使用静态方法以及作为“单元”的类来设置将正确输入映射到正确输出的测试用例吗?
您可以这样做来测试这些功能。但是当在你想测试的其他类中使用这些静态方法时,我相信你不能伪造它们(模拟/友好)或任何东西,因为你不能实例化一个类。
@Alfred:请看一下能够模拟静态方法的 PowerMock。在使用 PowerMock 的情况下,如果有的话,您会发现无法模拟的方法依赖关系。
你可以使用 PowerMock 对静力学进行单元测试,但是你很快就会发现你的 Permgen 空间已经用完了(完成了,得到了 T 恤),而且它仍然很糟糕。除非您知道(基于您自己在真正的 OO 语言方面至少十年的经验,而不是从 C 语言迁移),否则不要这样做。说真的,我见过的最糟糕的代码来自嵌入式开发人员对静态的使用,在大多数情况下,我们永远都被它困住了,添加更多代码只会让我们更加紧密地锁定在不可修改的单体中。松散耦合:不,可测试:几乎没有,可修改:从不。避免!
我可以理解测试依赖于静态状态的静态方法的难度。但是,当您测试 无状态 静态方法(如 Math.abs()Arrays.sort())时,即使是您可以将所有依赖项传递到的方法,我看不出这会如何阻碍单元测试。我想说一个简单的经验法则是:如果您有任何理由模拟程序逻辑,那么不要将它放在静态方法中。我从来没有理由嘲笑 Arrays.sort()Math.abs()
J
Jonah

static 方法是一种不需要初始化任何对象即可调用的方法。您是否注意到在 Java 中的 main 函数中使用了 static?程序执行从那里开始,而没有创建对象。

考虑以下示例:

 class Languages 
 {
     public static void main(String[] args) 
     {
         display();
     }

     static void display() 
     {
         System.out.println("Java is my favorite programming language.");
     }
  }

实际上最好的答案
K
Kevin Sylvestre

java中的静态方法属于类(不是它的实例)。它们不使用实例变量,通常会从参数中获取输入,对其执行操作,然后返回一些结果。实例方法与对象相关联,顾名思义,可以使用实例变量。


d
duffymo

不,静态方法与实例无关;他们属于这个阶级。静态方法是你的第二个例子;实例方法是第一个。


如果不需要对象的状态操作,您应该使用静态方法。
Y
Yosidroid

如果您将静态关键字应用于任何方法,则称为静态方法。

静态方法属于类而不是类的对象。无需创建类实例即可调用的静态方法。静态方法可以访问静态数据成员并可以更改它的值。仅使用类点静态名称的名称就可以访问静态方法。 . .示例:Student9.change();如果要使用类的非静态字段,则必须使用非静态方法。

//改变所有对象的公共属性(静态字段)的程序。

class Student9{  
 int rollno;  
 String name;  
 static String college = "ITS";  

 static void change(){  
 college = "BBDIT";  
 }  

 Student9(int r, String n){  
 rollno = r;  
 name = n;  
 }  

 void display (){System.out.println(rollno+" "+name+" "+college);}  

public static void main(String args[]){  
Student9.change();  

Student9 s1 = new Student9 (111,"Indian");  
Student9 s2 = new Student9 (222,"American");  
Student9 s3 = new Student9 (333,"China");  

s1.display();  
s2.display();  
s3.display();  
}  }

O/P: 111 印度 BBDIT 222 美国 BBDIT 333 中国 BBDIT


C
Carsten

静态方法不与实例关联,因此它们不能访问类中的任何非静态字段。

如果方法不使用类的任何字段(或仅静态字段),您将使用静态方法。

如果使用类的任何非静态字段,则必须使用非静态方法。


清晰、简短和简单的答案。
C
Charlie Seligman

静态方法应该在类上调用,实例方法应该在类的实例上调用。但这在现实中意味着什么?这是一个有用的例子:

汽车类可能有一个名为 Accelerate() 的实例方法。如果汽车确实存在(已构建),您只能加速汽车,因此这将是一个实例方法。

汽车类也可能有一个称为 GetCarCount() 的计数方法。这将返回创建(或建造)的汽车总数。如果没有构建汽车,此方法将返回 0,但它仍然应该能够被调用,因此它必须是一个静态方法。


j
jamz

当您希望能够在没有类实例的情况下访问方法时,请使用静态方法。


这并没有为程序的设计提供任何理由。
S
Sagar

实际上,我们在一个类中使用静态属性和方法,当我们想要使用程序的某些部分时,应该存在于我们的程序运行之前。我们知道,要操作静态属性,我们需要静态方法,因为它们不是实例变量的一部分。如果没有静态方法,操作静态属性非常耗时。


出于多种原因,将状态保存在静态变量中是一件坏事——比如多线程安全、调试、数据封装……等静态方法如果它们是纯函数是可以的(仅使用参数,无需更改它们)。一个很好的例子是实用程序类,比如数学计算。
F
Finbarr

静态:Obj.someMethod

当您想要提供对方法的类级别访问时使用 static,即该方法应该可以在没有类实例的情况下调用。


V
Vaishak Suresh

不需要在对象上调用静态方法,即在您使用它时。示例:您的 Main() 是静态的,您没有创建对象来调用它。


耶!看看我在谷歌搜索 Java noobie 问题时来到了哪里!这是一个小世界:-)
@Deepak 确实是小世界 :)
E
Echilon

静态方法和变量是 Java 中“全局”函数和变量的受控版本。其中方法可以作为classname.methodName()classInstanceName.methodName() 访问,即静态方法和变量可以使用类名以及类的实例来访问。

类不能声明为静态(因为它没有意义。如果一个类声明为public,则可以从任何地方访问),内部类可以声明为静态。


A
Amit Kaneria

可以使用静态方法,如果

不想对实例执行操作(实用程序方法)正如本文上述几个答案中提到的,将英里转换为公里,或计算从华氏到摄氏的温度,反之亦然。通过这些使用静态方法的示例,它不需要在堆内存中实例化整个新对象。考虑下面 1. new ABCClass(double farenheit).convertFarenheitToCelcium() 2. ABCClass.convertFarenheitToCelcium(double farenheit) 前者为每个方法调用创建一个新的类足迹,性能,实用。下面的例子是 Math 和 Apache-Commons 库 StringUtils 类: Math.random() Math.sqrt(double) Math.min(int, int) StringUtils.isEmpty(String) StringUtils.isBlank(String)

一个人想用作一个简单的功能。输入被显式传递,并将结果数据作为返回值。继承,对象实例化没有出现。简洁,可读。

注意:很少有人反对静态方法的可测试性,但静态方法也可以测试!使用 jMockit,可以模拟静态方法。可测试性。下面的例子:

new MockUp<ClassName>() {
    @Mock
    public int doSomething(Input input1, Input input2){
        return returnValue;
    }
};

r
rashedcs

静态方法是Java中无需创建类对象即可调用的方法。它属于类。

当我们不需要使用实例调用方法时,我们使用静态方法。


h
hemanto

静态方法有两个主要目的:

对于不需要任何对象状态的实用程序或辅助方法。由于不需要访问实例变量,因此使用静态方法消除了调用者为了调用方法而实例化对象的需要。对于类的所有实例共享的状态,例如计数器。所有实例必须共享相同的状态。仅使用该状态的方法也应该是静态的。


J
Jake N

我找到了一个很好的描述,何时使用静态方法:

没有硬性规定,编写良好的规则来决定何时使方法成为静态方法,但很少有基于经验的观察,这不仅有助于使方法成为静态方法,而且还教会了何时在 Java 中使用静态方法.您应该考虑在 Java 中将方法设为静态:

如果方法不修改对象的状态,或者不使用任何实例变量。您想在不创建该类的实例的情况下调用方法。如果一个方法只对提供给它的参数起作用,例如 public int factorial(int number){},则该方法是静态的很好候选者,该方法只对作为参数提供的数字起作用。实用方法也是静态的很好的候选方法,例如 StringUtils.isEmpty(String text),这是一个检查字符串是否为空的实用方法。如果方法的函数将在类层次结构中保持静态,例如 equals() 方法不是制作静态的好选择,因为每个类都可以重新定义相等性。

来源是 here


a
ave4496

在 Eclipse 中,您可以启用有助于检测潜在静态方法的警告。 (突出显示的行上方是我忘记突出显示的另一行)

https://i.stack.imgur.com/aQPg0.png


R
Ravindra babu

我想知道什么时候使用静态方法?

静态方法的一个常见用途是访问静态字段。但是您可以拥有静态方法,而无需引用静态变量。没有引用静态变量的辅助方法可以在一些 java 类中找到,例如 java.lang.Math public static int min(int a, int b) { return (a <= b) ?一:乙;另一个用例,我可以想到这些方法结合同步方法是在多线程环境中实现类级锁定。

假设我有一个带有几个 getter 和 setter 的类,一个或两个方法,并且我希望这些方法只能在类的实例对象上调用。这是否意味着我应该使用静态方法?

如果您需要访问类的实例对象上的方法,您的方法应该是非静态的。

Oracle 文档 page 提供了更多详细信息。

并非所有实例和类变量和方法的组合都是允许的:

实例方法可以直接访问实例变量和实例方法。实例方法可以直接访问类变量和类方法。类方法可以直接访问类变量和类方法。类方法不能直接访问实例变量或实例方法——它们必须使用对象引用。此外,类方法不能使用 this 关键字,因为没有实例可供 this 引用。


我们不能通过常规方法访问静态字段吗?那么这个 A common use for static methods is to access static fields. 不是一个参数。
A
Amrit

每当您不想创建对象来调用代码中的方法时,只需将该方法声明为静态。由于静态方法不需要调用实例,但这里的问题并不是所有静态方法都由 JVM 自动调用。此特权仅由 java 中的 main() "public static void main[String...args]" 方法享有,因为在运行时,这是 JVM 寻求的签名 public "static" void main[] 作为入口点的方法开始执行代码。

例子:

public class Demo
{
   public static void main(String... args) 
   {
      Demo d = new Demo();

      System.out.println("This static method is executed by JVM");

     //Now to call the static method Displ() you can use the below methods:
           Displ(); //By method name itself    
      Demo.Displ(); //By using class name//Recommended
         d.Displ(); //By using instance //Not recommended
   }

   public static void Displ()
   {
      System.out.println("This static method needs to be called explicitly");
   }
} 

输出:- 此静态方法由 JVM 执行此静态方法需要显式调用此静态方法需要显式调用此静态方法需要显式调用


A
Amitesh Bharti

无论何时,您都应该使用静态方法,

该方法中的代码不依赖于实例创建,也不使用任何实例变量。一段特定的代码将由所有实例方法共享。不应更改或覆盖方法的定义。您正在编写不应更改的实用程序类。

https://www.tutorialspoint.com/When-to-use-static-methods-in-Java