Singleton beans in Spring and classes based on Singleton design pattern are quite different.
The Java singleton is scoped by the Java class loader, the Spring singleton is scoped by the container context.
Which basically means that, in Java, you can be sure a singleton is a truly a singleton only within the context of the class loader which loaded it. Other class loaders should be capable of creating another instance of it (provided the class loaders are not in the same class loader hierarchy), despite of all your efforts in code to try to prevent it. In Spring, if you could load your singleton class in two different contexts and then again we can break the singleton concept. So, in summary, Java considers something a singleton if it cannot create more than one instance of that class within a given class loader, whereas Spring would consider something a singleton if it cannot create more than one instance of a class within a given container/context.
Here is some example:
spring-config.xml
The Java singleton is scoped by the Java class loader, the Spring singleton is scoped by the container context.
Which basically means that, in Java, you can be sure a singleton is a truly a singleton only within the context of the class loader which loaded it. Other class loaders should be capable of creating another instance of it (provided the class loaders are not in the same class loader hierarchy), despite of all your efforts in code to try to prevent it. In Spring, if you could load your singleton class in two different contexts and then again we can break the singleton concept. So, in summary, Java considers something a singleton if it cannot create more than one instance of that class within a given class loader, whereas Spring would consider something a singleton if it cannot create more than one instance of a class within a given container/context.
Here is some example:
spring-config.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.2.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.2.xsd">
<bean id="a" class="com.pkg.Singleton" scope="singleton" />
</beans>
Bean Singleton public class Singleton{
private String text;
public String getText() {
return text;
}
public void setText(String text) {
this.text = text;
}
}
Test public class Test {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-config.xml");
Singleton a1 = ctx.getBean("a", Singleton.class);
a1.setText("text A1");
Singleton a2 = ctx.getBean("a", Singleton.class);
a2.setText("text A2");
System.out.println("a1: " + a1.getText());
System.out.println("a2: " + a2.getText());
}
}
Output: a1: text A2
a2: text A2
And now let's create another one ApplicationContext: public class Test {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("spring-config.xml");
ApplicationContext ctx2 = new ClassPathXmlApplicationContext("spring-config.xml");
Singleton a1 = ctx.getBean("a", Singleton.class);
a1.setText("text A1");
Singleton a2 = ctx2.getBean("a", Singleton.class);
a2.setText("text A2");
System.out.println("a1: " + a1.getText());
System.out.println("a2: " + a2.getText());
// both ctx and ctx2 have same classloaders
System.out.println("context1 classloader: " + ctx.getClassLoader());
System.out.println("context2 classloader: " + ctx2.getClassLoader());
}
}
Output: a1: text A1
a2: text A2
context1 classloader: sun.misc.Launcher$AppClassLoader@5284e9
context2 classloader: sun.misc.Launcher$AppClassLoader@5284e9
No comments:
Post a Comment