воскресенье, 7 апреля 2013 г.

Паттерн проектирования: Singleton / Одиночка



Тип паттерна: порождающий

Один из самых легких и полезных паттернов является синглтон. С него можно начинать изучать паттерны проектирования сразу после изучения основ ООП.

Его поведение крайне простое: не допустить существование в программе двух объектов одного класса. Для класса, реализующего синглтон, может существовать только один объект. И класс предоставляет глобальную точку доступа к нему.

Глобальный объект, конечно, усложняет масштабирование системы. И это минус. Но для небольших приложений вполне применим.

Например, часто бывает нужно настроить коннект к базе и потом использовать его из разных модулей. Здесь может прийти на помощь синглтон.

В чем принцип работы?
Во-первых скрываются конструкторы класса модификатором private.
Во-вторых у класса добавляется статическое поле - экземпляр класса, недоступное извне (private).
В-третьих добавляется статический метод возвращающий этот экземпляр getInstance()

Пример простого синглтона:

public class Singleton { private static Singleton instance; private Singleton (){ } public static Singleton getInstance(){ if (instance == null){ instance = new Singleton(); } return instance; } }

К сожалению, здесь нет учета работы в многопоточной среде. Поэтому, чтобы не возникло гонки, можно реализовать синглтон так:

public class Singleton { private static Singleton instance; private Singleton (){ } public static Singleton getInstance(){ synchronized (this){ if (instance == null){ instance = new Singleton(); } } return instance; } }
Или даже так:

public class Singleton { private static Singleton instance = new Singleton(); private Singleton (){ } public static Singleton getInstance(){ return instance; } }

суббота, 6 апреля 2013 г.

Зачем нужен Class.forName при работе с БД

Это статический метод, который возвращает экземпляр типа Class. Но используется он не для получения объекта. Дело в том, что при работе с базами данных нам нужно "загрузить jdbc драйвер". То-есть нам нужно, чтобы экземпляр класса драйвера был загружен в jvm.

Да, именно экземпляр класса, а не объекта. Класс тоже представляет собой некую структуру, которая занимает оперативную память. Более того у класса может быть static-секция, которая инициализирует статические данные класса.

Например для HSQLDB, загрузка класса может выглядеть так:

Class.forName("org.hsqldb.jdbcDriver");

Класс может быть автоматически  загружен при создании экземпляра класса с помощью конструктора. Например так:

new org.hsqldb.jdbcDriver();

Но этот код не скажет нам о проблемах при загрузке класса. А Class.forName может кинуть исключение, которое можно будет обработать.


private boolean loadDriver() { try { Class.forName("org.hsqldb.jdbcDriver"); } catch (ClassNotFoundException e) { System.out.println("Драйвер не найден"); e.printStackTrace(); return false; } return true; }