- Creating Objects
- Declaring a Variable to Refer to an Object
- Instantiating a Class
- Initializing an Object
- Создание объекта Java
- Использование ключевого слова «new»
- Синтаксис
- Пример
- Использование метода clone()
- Пример
- Вывод
- Использование метода newInstance() класса Class
- Пример
- Использование десериализации
- Использование метода newInstance() класса Constructor
- Создание и удаление объектов Java — основы
- Способы создания объекта Java
- Динамическое создание объектов Java
Creating Objects
As you know, a class provides the blueprint for objects; you create an object from a class. Each of the following statements taken from the CreateObjectDemo program creates an object and assigns it to a variable:
Point originOne = new Point(23, 94); Rectangle rectOne = new Rectangle(originOne, 100, 200); Rectangle rectTwo = new Rectangle(50, 100);
The first line creates an object of the Point class, and the second and third lines each create an object of the Rectangle class.
Each of these statements has three parts (discussed in detail below):
- Declaration: The code set in bold are all variable declarations that associate a variable name with an object type.
- Instantiation: The new keyword is a Java operator that creates the object.
- Initialization: The new operator is followed by a call to a constructor, which initializes the new object.
Declaring a Variable to Refer to an Object
Previously, you learned that to declare a variable, you write:
This notifies the compiler that you will use name to refer to data whose type is type. With a primitive variable, this declaration also reserves the proper amount of memory for the variable.
You can also declare a reference variable on its own line. For example:
If you declare originOne like this, its value will be undetermined until an object is actually created and assigned to it. Simply declaring a reference variable does not create an object. For that, you need to use the new operator, as described in the next section. You must assign an object to originOne before you use it in your code. Otherwise, you will get a compiler error.
A variable in this state, which currently references no object, can be illustrated as follows (the variable name, originOne , plus a reference pointing to nothing):
Instantiating a Class
The new operator instantiates a class by allocating memory for a new object and returning a reference to that memory. The new operator also invokes the object constructor.
Note: The phrase «instantiating a class» means the same thing as «creating an object.» When you create an object, you are creating an «instance» of a class, therefore «instantiating» a class.
The new operator requires a single, postfix argument: a call to a constructor. The name of the constructor provides the name of the class to instantiate.
The new operator returns a reference to the object it created. This reference is usually assigned to a variable of the appropriate type, like:
Point originOne = new Point(23, 94);
The reference returned by the new operator does not have to be assigned to a variable. It can also be used directly in an expression. For example:
int height = new Rectangle().height;
This statement will be discussed in the next section.
Initializing an Object
Here’s the code for the Point class:
This class contains a single constructor. You can recognize a constructor because its declaration uses the same name as the class and it has no return type. The constructor in the Point class takes two integer arguments, as declared by the code (int a, int b). The following statement provides 23 and 94 as values for those arguments:
Point originOne = new Point(23, 94);
The result of executing this statement can be illustrated in the next figure:
Here’s the code for the Rectangle class, which contains four constructors:
public class Rectangle < public int width = 0; public int height = 0; public Point origin; // four constructors public Rectangle() < origin = new Point(0, 0); >public Rectangle(Point p) < origin = p; >public Rectangle(int w, int h) < origin = new Point(0, 0); width = w; height = h; >public Rectangle(Point p, int w, int h) < origin = p; width = w; height = h; >// a method for moving the rectangle public void move(int x, int y) < origin.x = x; origin.y = y; >// a method for computing the area of the rectangle public int getArea() < return width * height; >>
Each constructor lets you provide initial values for the rectangle’s origin, width, and height, using both primitive and reference types. If a class has multiple constructors, they must have different signatures. The Java compiler differentiates the constructors based on the number and the type of the arguments. When the Java compiler encounters the following code, it knows to call the constructor in the Rectangle class that requires a Point argument followed by two integer arguments:
Rectangle rectOne = new Rectangle(originOne, 100, 200);
This calls one of Rectangle ‘s constructors that initializes origin to originOne . Also, the constructor sets width to 100 and height to 200. Now there are two references to the same Point objectan object can have multiple references to it, as shown in the next figure:
The following line of code calls the Rectangle constructor that requires two integer arguments, which provide the initial values for width and height. If you inspect the code within the constructor, you will see that it creates a new Point object whose x and y values are initialized to 0:
Rectangle rectTwo = new Rectangle(50, 100);
The Rectangle constructor used in the following statement doesn’t take any arguments, so it’s called a no-argument constructor:
Rectangle rect = new Rectangle();
All classes have at least one constructor. If a class does not explicitly declare any, the Java compiler automatically provides a no-argument constructor, called the default constructor. This default constructor calls the class parent’s no-argument constructor, or the Object constructor if the class has no other parent. If the parent has no constructor ( Object does have one), the compiler will reject the program.
Создание объекта Java
Java – это объектно-ориентированный язык. Другими словами, почти все в Java рассматривается как объект.
Использование ключевого слова «new»
При программировании вы наверняка сталкивались с ключевым словом «new», используемым для создания объекта, которому динамически выделяется память, т.е. память этим объектам выделяется во время выполнения. И это динамическое распределение требуется большую часть времени при создании объектов. Следовательно, этот метод используется чаще, чем другие.
Синтаксис
ClassName ObjectName = new classConstructor();
Пример
public class ObjectCreation < String FirstString = "Hello World"; public static void main(String[] args) < ObjectCreation obj = new ObjectCreation(); System.out.println(obj.FirstString); >>
Этот метод создания объектов в Java может использоваться с любым конструктором требуемого класса, если класс имеет более одного конструктора.
Использование метода clone()
Что если объект, который мы хотим создать, должен быть копией уже существующего ? В этом случае мы можем использовать метод clone(), он – часть класса Object, но его нельзя использовать напрямую, поскольку является защищенным методом.
Метод clone() можно использовать только после реализации интерфейса Cloneable и обработки исключения CloneNotSupportedException.
Пример
class Message implements Cloneable < String FirstString; Message() < this.FirstString = "Hello World"; >public Object clone() throws CloneNotSupportedException < return super.clone(); >> public class ObjectCreation < public static void main(String[] args) throws CloneNotSupportedException < Message FirstObj = new Message(); System.out.println(FirstObj.FirstString); Message SecondObj = (Message)FirstObj.clone(); System.out.println(SecondObj.FirstString); SecondObj.FirstString = "Welcome to the world of programming"; System.out.println(SecondObj.FirstString); System.out.println(FirstObj.FirstString); >>
Вывод
Welcome to the world of programming
В приведенной выше программе мы создали копию нашего уже существующего объекта. Чтобы обе переменные не указывали на одну и ту же ячейку памяти, было важно изменить значение «FirstString» для второго объекта, а затем распечатать его значение для обоих объектов.
Использование метода newInstance() класса Class
Этот метод не часто используется для создания объектов, в случаях, когда мы знаем имя класса, а конструктор по умолчанию является публичным по своей природе. Чтобы использовать его, нам нужно обработать 3 исключения
- ClassNotFoundException – это исключение происходит, если JVM не может найти класс, который передается в качестве аргумента.
- InstantiationException – это исключение возникает, если данный класс не содержит конструктор по умолчанию.
- IllegalAccessException – это исключение происходит, если у нас нет доступа к указанному классу.
Пример
class ObjectCreation < String FirstString = "Hello World"; public static void main(String[] args) < try < Class Message = Class.forName("ObjectCreation"); ObjectCreation obj = (ObjectCreation) Message.newInstance(); System.out.println(obj.FirstString); >catch (ClassNotFoundException e) < e.printStackTrace(); >catch (InstantiationException e) < e.printStackTrace(); >catch (IllegalAccessException e) < e.printStackTrace(); >> >
Использование десериализации
В Java сериализация используется для преобразования текущего состояния объекта в поток байтов. десериализация является полной противоположностью, поскольку мы воссоздаем объект, используя поток байтов. Для процесса сериализации нам необходимо реализовать интерфейс Serializable.
Обработка исключений должна быть сделана для создания объектов с использованием этого метода.
ObjectInputStream objectInputStream = new ObjectInputStream(inputStream); Classname object = (classname) objectInputStream.readObject();
Использование метода newInstance() класса Constructor
Мы увидели метод newInstance класса Class, который мы использовали для создания объекта. Точно так же конструктор класса также состоит из метода newInstance(), который можно использовать для создания объектов. Другие могут использовать конструкторы Java по умолчанию с помощью этого метода, мы также можем вызвать параметризованные конструкторы.
import java.lang.reflect.*; public class ObjectCreation < private String FirstString = "Hello World"; ObjectCreation() < >public void changeMessage(String message) < this.FirstString = message; >public static void main(String[] args) < try < Constructorconstructor = ObjectCreation.class.getDeclaredConstructor(); ObjectCreation objectCreation = constructor.newInstance(); objectCreation.changeMessage("Welcome to the world of programming"); System.out.println(objectCreation.FirstString); > catch (Exception e) < e.printStackTrace(); >> >
Welcome to the world of programming.
Это 5 различных способов создания объектов в Java, которые используются чаще других. Каждый метод имеет свои преимущества и недостатки. В конце концов, выбор за вами.
Создание и удаление объектов Java — основы
В объектно-ориентированном программировании при создании объекта вызывается конструктор. Это блок команд, который инициализирует объект: присваивает значения его параметрам и выполняет начальные действия для подготовки объекта к работе. Конструктор может принимать параметры явно или использовать прописанные в классе значения по умолчанию.
Способы создания объекта Java
Чтобы создать объект, используют ключевое слово new. Вы его наверняка помните по созданию массивов, которые в Java тоже являются объектами. После new указывают тип создаваемого объекта, т.е. класс. В Java, как и в других ООП-языках, можно использовать встроенные классы и определять пользовательские.
Класс — это шаблон, по которому создаётся экземпляр — объект. Класс определяет свойства и поведение будущего объекта. Команда «Изготовь шестигранную гайку» имеет смысл, только если исполнитель понимает, что такое гайка. Именно класс описывает свойства «гайки» и указывает, что с ней можно делать.
Допустим, у нас есть класс Item («предмет»), где прописано, что каждый предмет имеет габариты и свойство moveable, которое определяет, можно ли предмет двигать:
class Item < int height = 5; int width = 5; int depth = 5; boolean isMoveable = false;
// А вот конструктор Item(int userHeight, int userWidth, int userDepth, boolean itemIsMoveable) < // присваиваем пользовательские значения width = userHeight; height = userWidth; depth = userDepth; moveable = itemIsMoveable; >>
Чтобы создать экземпляр toolbox («ящик с инструментами») класса Item, заведём переменную для хранения ссылки на объект, а затем запустим конструктор:
Item toolbox; toolbox = new Item (7, 2, 4, true);
При создании ящика мы переопределили его параметры. Если бы нас устраивали значения по умолчанию, мы создали бы ящик без параметров:
Динамическое создание объектов Java
Объекты создаются и исчезают во время работы приложений. Вы заказали новый юнит в игре — он появился, отправили его сражаться — он погиб. У каждого объекта свой жизненный цикл: от создания до уничтожения.
Память для хранения данных об объекте выделяется и высвобождается. Причем специально удалять неиспользуемый объект не нужно — для этого в Java есть сборщик мусора. Он автоматически запускается и стирает объекты, которые уже не планируется вызывать. Как он их находит? По отсутствию дальнейших обращений к этим объектам (ссылок на них).
В Джаве нет деструктора в привычном понимании. Есть метод finalize — он вызывается, когда за объектом приходит сборщик мусора. Но есть у этого метода особенности, из-за которых его редко используют на практике:
- неизвестно, в какой момент он будет вызван и будет ли вызван вообще — программа может завершиться раньше;
- он замедляет работу сборщика мусора с объектом более чем в 400 раз;
- объекты утилизируются не сразу, а попадают в очередь на финализацию и продолжают занимать память неопределённое время.
Зачем же этот метод существует в языке?
- Для составления «описи» объектов, которые не используются, но и не удалены — через поиск и логирование.
- Для высвобождения ресурсов при работе с нативным кодом: например, С.
Более толковый помощник в борьбе за свободные ресурсы — интерфейс AutoCloseable — позволяет закрывать потоки, файлы, соединения с БД и др. Чтобы воспользоваться интерфейсом, нужно реализовать его единственный метод, сlose(), в классе, объекты которого вы планируете закрывать:
public class SampleClass implements AutoCloseable < //указали, что наш класс реализует интерфейс AutoCloseable public void do_something() < // метод класса, который делает что-то нужное System.out.println("Вызван метод класса."); > @Override public void close() throws Exception / реализуем close() System.out.println("Объект отработал, высвобождаем ресурсы."); > >
Вызвать метод можно в конструкции try-с-ресурсами:
private static void SampleClass() throws Exception < try(SampleClass sc1 = new SampleClass())< sc1.do_something(); > >
Открыли ресурс, запустили выполнение задачи, завершили выполнение, закрыли ресурс.
Также рекомендуем посмотреть вебинар по созданию консольных приложений на Java, где будут рассматриваться объекты и классы.