public double getLength() {
return 11;
}
@Override
public String getShape() {
return “Rim,S”;
}
}
/**
-
.22 Long
-
@author skyline
*/
public class Long22 extends BulletSpecification {
@Override
public double getCaliber() {
return 5.59;
}
@Override
public double getLength() {
return 15;
}
@Override
public String getShape() {
return “Rim,S”;
}
}
/**
-
.22 LR
-
@author skyline
*/
public class LR22 extends BulletSpecification {
@Override
public double getCaliber() {
return 5.7;
}
@Override
public double getLength() {
return 15;
}
@Override
public String getShape() {
return “Rim,S”;
}
}
/**
-
.22 Short
-
@author skyline
*/
public class Short22 extends BulletSpecification {
@Override
public double getCaliber() {
return 5.59;
}
@Override
public double getLength() {
return 11;
}
@Override
public String getShape() {
return “Rim,S”;
}
}
/**
-
.22 Stinger
-
@author skyline
*/
public class Stinger22 extends BulletSpecification {
@Override
public double getCaliber() {
return 5.59;
}
@Override
public double getLength() {
return 18;
}
@Override
public String getShape() {
return “Rim,S”;
}
}
/**
-
.22 Win
-
@author skyline
*/
public class Win22 extends BulletSpecification {
@Override
public double getCaliber() {
return 5.59;
}
@Override
public double getLength() {
return 27;
}
@Override
public String getShape() {
return “Rim,S”;
}
}
=======================================================================
/**
-
子弹
-
@author skyline
*/
public class Bullet {
private final String id;
private final BulletSpecification specification;
public Bullet(BulletSpecification specification) {
this.id = UUID.randomUUID().toString();
this.specification = specification;
}
public String getId() {
return id;
}
public BulletSpecification getSpecification() {
return specification;
}
@Override
public String toString() {
return “Bullet{” +
“id='” + id + ‘’’ +
“, specification=” + specification +
‘}’;
}
}
/**
-
子弹工厂
-
@author skyline
*/
public class BulletFactory {
public Bullet createBullet(Class<? extends BulletSpecification> specificationClazz) {
Bullet bullet = null;
try {
BulletSpecification bulletSpecification = specificationClazz.newInstance();
bullet = new Bullet(bulletSpecification);
} catch (InstantiationException | IllegalAccessException e) {
e.printStackTrace();
}
return bullet;
}
}
====================================================================
public class FlyWeightMain {
public static void main(String[] args) {
BulletFactory factory = new BulletFactory();
Class<? extends BulletSpecification>[] types = new Class[]{BBCap22.class, CBCap22.class, Long22.class, LR22.class, Short22.class, Win22.class, Stinger22.class};
List bullets = new ArrayList<>();
for (Class<? extends BulletSpecification> type : types) {
for (int i = 0; i < 100000; i++) {
Bullet bullet = factory.createBullet(type);
bullets.add(bullet);
System.out.println(bullet.toString());
}
}
}
}
==================================================================
上面的这一套代码搞下来其实从程序运行的角度看,没啥问题。直接运行一下也可以正常结束。为了突出问题,我给程序加了一些限制-Xmx115M -Xms115M
,然后我们再看看。
从截图中可以看出来GC的非常疯狂,最终GC总耗时大约是5.3秒。那么有没有优化空间呢?
==================================================================
我们从对象上分析后发现,每个子弹对象都包含两部分,如下图:
一个是子弹的id,这个是每个子弹都不一样的。还有一个是子弹的规格信息specification
。这个规格信息中包含了很多数据,而且其实没必要每个子弹都创建一个新的规格信息,相同规格的子弹共用一个规格信息对象即可。这样我们就可以省下很多内存空间。
==================================================================
按照上面的思想我对程序作了改造,改造后程序如下:
相同规格的子弹使用的规格信息被缓存到specificationMap
中,每次创建子弹时先从specificationMap
中获取规格信息,如果规格信息不存在,那再创建。让我们看下改造之后的GC情况:
从GC上看明显比第一版程序快了很多。这就是享元模式给我们带来的好处。
====================================================================
享元模式主要用于减少创建对象的数量,以减少内存占用和提高性能。这种类型的设计模式属于结构模式,它提供了减少对象数量从而改善应用所需的对象结构的方式。
享元模式尝试重用现有的同类对象,如果未找到匹配的对象,则创建新的对象。
======================================================================
JVM层面最典型的享元模式的应用就是字符串常量池了。String之所以能够作为常量是因为String本身是final的,同时String中的char[]也是final的,也就是说String一旦new出来,就不可能再发生变化了。
接下来我们看下下面这段经典代码:
/**
- 字符串常量池
*/
public class StringMain {
public static void main(String[] args) {
String str1 = new String(“a”);
String str2 = “a”;
String str3 = str1.intern();
String str4 = new String(“a”).intern();
System.out.println(str1 == str2); //false
System.out.println(str2 == str3); //true
System.out.println(str3 == str4); //true
System.out.println(str4 == str1); //false
最后
本人也收藏了一份Java面试核心知识点来应付面试,借着这次机会可以送给我的读者朋友们
目录:
Java面试核心知识点
一共有30个专题,足够读者朋友们应付面试啦,也节省朋友们去到处搜刮资料自己整理的时间!
Java面试核心知识点
已经有读者朋友靠着这一份Java面试知识点指导拿到不错的offer了
System.out.println(str1 == str2); //false
System.out.println(str2 == str3); //true
System.out.println(str3 == str4); //true
System.out.println(str4 == str1); //false
最后
本人也收藏了一份Java面试核心知识点来应付面试,借着这次机会可以送给我的读者朋友们
目录:
[外链图片转存中…(img-YjqSPxr2-1721172739671)]
Java面试核心知识点
一共有30个专题,足够读者朋友们应付面试啦,也节省朋友们去到处搜刮资料自己整理的时间!
[外链图片转存中…(img-SY9IUdal-1721172739672)]
Java面试核心知识点
已经有读者朋友靠着这一份Java面试知识点指导拿到不错的offer了
[外链图片转存中…(img-RK4wHnku-1721172739672)]