¿Cómo puedo llamar a un constructor de otro en Java?

¿Es posible llamar a un constructor desde otro (dentro de la misma clase, no desde una subclase)? Si es así, ¿cómo? ¿Y cuál podría ser la mejor manera de llamar a otro constructor (si hay varias formas de hacerlo)?

Sí, es posible:

public class Foo { private int x; public Foo() { this(1); } public Foo(int x) { this.x = x; } } 

Para encadenar a un constructor de superclase particular en lugar de uno en la misma clase, use super lugar de this . Tenga en cuenta que solo puede encadenar a un constructor , y tiene que ser la primera instrucción en su cuerpo constructor .

Ver también esta pregunta relacionada , que trata sobre C # pero donde se aplican los mismos principios.

Usando this(args) . El patrón preferido es trabajar desde el constructor más pequeño hasta el más grande.

 public class Cons { public Cons() { // A no arguments constructor that sends default values to the largest this(madeUpArg1Value,madeUpArg2Value,madeUpArg3Value); } public Cons(int arg1, int arg2) { // An example of a partial constructor that uses the passed in arguments // and sends a hidden default value to the largest this(arg1,arg2, madeUpArg3Value); } // Largest constructor that does the work public Cons(int arg1, int arg2, int arg3) { this.arg1 = arg1; this.arg2 = arg2; this.arg3 = arg3; } } 

También puede usar un enfoque defendido más recientemente de valueOf o simplemente “de”:

 public class Cons { public static Cons newCons(int arg1,...) { // This function is commonly called valueOf, like Integer.valueOf(..) // More recently called "of", like EnumSet.of(..) Cons c = new Cons(...); c.setArg1(....); return c; } } 

Para llamar a una súper clase, use super(asdf) . La llamada a super debe ser la primera llamada en el constructor o obtendrá un error de comstackción.

[ Nota: solo quiero agregar un aspecto, que no vi en las otras respuestas: cómo superar las limitaciones del requisito de que esto () tiene que estar en la primera línea). ]

En Java, otro constructor de la misma clase puede ser llamado desde un constructor a través de this() . Sin embargo, tenga en cuenta que this tiene que estar en la primera línea.

 public class MyClass { public MyClass(double argument1, double argument2) { this(argument1, argument2, 0.0); } public MyClass(double argument1, double argument2, double argument3) { this.argument1 = argument1; this.argument2 = argument2; this.argument3 = argument3; } } 

Que this tenga que aparecer en la primera línea parece una gran limitación, pero puedes construir los argumentos de otros constructores a través de métodos estáticos. Por ejemplo:

 public class MyClass { public MyClass(double argument1, double argument2) { this(argument1, argument2, getDefaultArg3(argument1, argument2)); } public MyClass(double argument1, double argument2, double argument3) { this.argument1 = argument1; this.argument2 = argument2; this.argument3 = argument3; } private static double getDefaultArg3(double argument1, double argument2) { double argument3 = 0; // Calculate argument3 here if you like. return argument3; } } 

Cuando necesito llamar a otro constructor desde dentro del código (no en la primera línea), suelo usar un método de ayuda como este:

 class MyClass { int field; MyClass() { init(0); } MyClass(int value) { if (value<0) { init(0); } else { init(value); } } void init(int x) { field = x; } } 

Pero la mayoría de las veces trato de hacerlo al revés llamando a los constructores más complejos desde los más simples en la primera línea, en la medida de lo posible. Para el ejemplo anterior

 class MyClass { int field; MyClass(int value) { if (value<0) field = 0; else field = value; } MyClass() { this(0); } } 

Dentro de un constructor, puede usar la palabra clave this para invocar a otro constructor en la misma clase. Hacerlo se llama una invocación de constructor explícita .

Aquí hay otra clase Rectangle, con una implementación diferente a la de la sección Objetos.

 public class Rectangle { private int x, y; private int width, height; public Rectangle() { this(1, 1); } public Rectangle(int width, int height) { this( 0,0,width, height); } public Rectangle(int x, int y, int width, int height) { this.x = x; this.y = y; this.width = width; this.height = height; } } 

Esta clase contiene un conjunto de constructores. Cada constructor inicializa algunas o todas las variables de miembros del rectángulo.

Como todos ya han dicho, usas this(…) , que se llama una invocación explícita del constructor .

Sin embargo, tenga en cuenta que dentro de una statement de invocación de constructor tan explícita no puede referirse a

  • cualquier variable de instancia o
  • cualquier método de instancia o
  • cualquier clase interna declarada en esta clase o cualquier superclase, o
  • this o
  • super .

Como se establece en JLS (§8.8.7.1).

Te diré una manera fácil

Hay dos tipos de constructores:

  1. Constructor predeterminado
  2. Constructor parametrizado

Explicaré en un ejemplo

 class ConstructorDemo { ConstructorDemo()//Default Constructor { System.out.println("D.constructor "); } ConstructorDemo(int k)//Parameterized constructor { this();//-------------(1) System.out.println("P.Constructor ="+k); } public static void main(String[] args) { //this(); error because "must be first statement in constructor new ConstructorDemo();//-------(2) ConstructorDemo g=new ConstructorDemo(3);---(3) } } 

En el ejemplo anterior, mostré 3 tipos de llamadas

  1. this () llamada a esto debe ser primera instrucción en constructor
  2. Este es Name less Object. esto llama automáticamente al constructor predeterminado. 3. Esto llama al constructor parametrizado.

Nota: esta debe ser la primera statement en el constructor.

Sí, cualquier número de constructores puede estar presente en una clase y pueden ser llamados por otro constructor usando this() [No confunda this() llamada de constructor this() con this palabra clave]. this() o this(args) debería ser la primera línea en el constructor.

Ejemplo:

 Class Test { Test() { this(10); // calls the constructor with integer args, Test(int a) } Test(int a) { this(10.5); // call the constructor with double arg, Test(double a) } Test(double a) { System.out.println("I am a double arg constructor"); } } 

Esto se conoce como sobrecarga de constructor.
Tenga en cuenta que para el constructor, solo el concepto de sobrecarga es aplicable y no herencia o anulación.

Puede usar un constructor de otro constructor de la misma clase usando la palabra clave “this”. Ejemplo –

 class This1 { This1() { this("Hello"); System.out.println("Default constructor.."); } This1(int a) { this(); System.out.println("int as arg constructor.."); } This1(String s) { System.out.println("string as arg constructor.."); } public static void main(String args[]) { new This1(100); } } 

Salida – cadena como constructor arg .. Constructor por defecto .. int como constructor arg ..

Llamando al constructor desde otro constructor

 class MyConstructorDemo extends ConstructorDemo { MyConstructorDemo() { this("calling another constructor"); } MyConstructorDemo(String arg) { System.out.print("This is passed String by another constructor :"+arg); } } 

También puede llamar al constructor padre utilizando la llamada super()

Sí, es posible llamar a un constructor de otro con el uso de this()

 class Example{ private int a = 1; Example(){ this(5); //here another constructor called based on constructor argument System.out.println("number a is "+a); } Example(int b){ System.out.println("number b is "+b); } 

Sí, es posible llamar a un constructor de otro. Pero hay una regla para eso. Si se realiza una llamada de un constructor a otro, entonces

esa nueva llamada de constructor debe ser la primera statement en el constructor actual

 public class Product { private int productId; private String productName; private double productPrice; private String category; public Product(int id, String name) { this(id,name,1.0); } public Product(int id, String name, double price) { this(id,name,price,"DEFAULT"); } public Product(int id,String name,double price, String category){ this.productId=id; this.productName=name; this.productPrice=price; this.category=category; } } 

Por lo tanto, algo como lo siguiente no funcionará.

 public Product(int id, String name, double price) { System.out.println("Calling constructor with price"); this(id,name,price,"DEFAULT"); } 

Además, en el caso de la herencia, cuando se crea el objeto de la subclase, primero se llama al constructor de la superclase.

 public class SuperClass { public SuperClass() { System.out.println("Inside super class constructor"); } } public class SubClass extends SuperClass { public SubClass () { //Even if we do not add, Java adds the call to super class's constructor like // super(); System.out.println("Inside sub class constructor"); } } 

Por lo tanto, en este caso también se declara primero otra llamada de constructor antes que cualquier otra instrucción.

La palabra clave esto se puede usar para llamar a un constructor desde un constructor, cuando se escriben varios constructores para una clase, hay momentos en los que le gustaría llamar a un constructor de otro para evitar el código duplicado.

Bellow es un enlace que explico otro tema sobre constructor y getters () y setters () y usé una clase con dos constructores. Espero que las explicaciones y los ejemplos te ayuden.

Setter métodos o constructores

Hay patrones de diseño que cubren la necesidad de una construcción compleja; si no se puede hacer de manera sucinta, cree un método de fábrica o una clase de fábrica.

Con la última versión de Java y la adición de lambdas, es fácil crear un constructor que pueda aceptar cualquier código de inicialización que desee.

 class LambdaInitedClass { public LamdaInitedClass(Consumer init) { init.accept(this); } } 

Llámalo con …

  new LambdaInitedClass(l -> { // init l any way you want }); 

Bastante simple

 public class SomeClass{ int number; String someString; public SomeClass(){ number = 0; } public SomeClass(int number){ this(); //set the class to 0 this.setNumber(number); } public SomeClass(int number, String someString){ this(number); //call public SomeClass( int number ) } public void setNumber(int number){ this.number = number; } public void setString(String someString){ this.someString = someString; } //.... add some accessors } 

ahora aquí hay un pequeño crédito extra:

 public SomeOtherClass extends SomeClass { public SomeOtherClass(int number, String someString){ super(number, someString); //calls public SomeClass(int number, String someString) } //.... Some other code. } 

Espero que esto ayude.

Sé que hay tantos ejemplos de esta pregunta, pero lo que encontré aquí es para compartir mi Idea. hay dos formas de encadenar constructor. En la misma clase, puede usar esta palabra clave. en Herencia, necesitas usar la palabra clave super.

  import java.util.*; import java.lang.*; class Test { public static void main(String args[]) { Dog d = new Dog(); // Both Calling Same Constructor of Parent Class ie 0 args Constructor. Dog cs = new Dog("Bite"); // Both Calling Same Constructor of Parent Class ie 0 args Constructor. // You need to Explicitly tell the java compiler to use Argument constructor so you need to use "super" key word System.out.println("------------------------------"); Cat c = new Cat(); Cat caty = new Cat("10"); System.out.println("------------------------------"); // Self s = new Self(); Self ss = new Self("self"); } } class Animal { String i; public Animal() { i = "10"; System.out.println("Animal Constructor :" +i); } public Animal(String h) { i = "20"; System.out.println("Animal Constructor Habit :"+ i); } } class Dog extends Animal { public Dog() { System.out.println("Dog Constructor"); } public Dog(String h) { System.out.println("Dog Constructor with habit"); } } class Cat extends Animal { public Cat() { System.out.println("Cat Constructor"); } public Cat(String i) { super(i); // Calling Super Class Paremetrize Constructor. System.out.println("Cat Constructor with habit"); } } class Self { public Self() { System.out.println("Self Constructor"); } public Self(String h) { this(); // Explicitly calling 0 args constructor. System.out.println("Slef Constructor with value"); } } 

Puede llamar a otro constructor a través de la palabra clave this(...) (cuando necesita llamar a un constructor de la misma clase) o la palabra clave super(...) (cuando necesita llamar a un constructor desde una superclase).

Sin embargo, tal llamada debe ser la primera statement de su constructor. Para superar esta limitación, use esta respuesta .

Se llama Telescoping Constructor anti-pattern o constructor encadenamiento. Sí, definitivamente puedes hacerlo. Veo muchos ejemplos arriba y quiero agregar que si sabes que solo necesitas dos o tres constructores, podría estar bien. Pero si necesita más, intente utilizar un patrón de diseño diferente, como el patrón de Builder. Como por ejemplo:

  public Omar(){}; public Omar(a){}; public Omar(a,b){}; public Omar(a,b,c){}; public Omar(a,b,c,d){}; ... 

Es posible que necesite más. El patrón de construcción sería una gran solución en este caso. Aquí hay un artículo, podría ser útil https://medium.com/@modestofiguereo/design-patterns-2-the-builder-pattern-and-the-telescoping-constructor-anti-pattern-60a33de7522e