方法中返回的对象,一般不会造成内存溢出,但是一下情况就是刚好造成的内容溢出:
如果对象的引用刚好被单例所持有的话,JVM就不会回收该引用。
1、创建对象
Info info = new Info();
复制代码
new Info()的时候,会返回一个地址,并且将地址赋值给引用info,当这个引用被info持有的时候,java虚拟机会认为这个对象是有用的,不会回收。
2、方法中返回一个对象
private UiDataCenter.QrInfo genQrcInfo(QrActionInfo info, String aType, String id)
{
UiDataCenter.QrInfo qrInfo = new UiDataCenter.QrInfo();
if(null != info && Ut.isValidate(id) && StringUtil.isValidate(aType))
{
qrInfo.mainBatchID = info.mainBatchID;
qrInfo.subBatchID = info.subBatchID;
qrInfo.money = info.money;
qrInfo.qrId = id;
qrInfo.aType = aType;
}
return qrInfo;
}
复制代码
此时返回的是对象的引用。如果该对象的引用的被持有数为0的话,该对象的引用会被Java虚拟机回收;否则JVM不会回收该引用。
以上操作都不会造成内存溢出,刚好造成内存溢出的是下方的单例持有该对象的引用。
3、单例持有对象的引用,造成内存溢出
UiDataCenter.QrInfo qrInfo = genQrcInfo(qrActionInfo, qrResInfo.actType,qrId)
UiDataCenter.getInstance().setQrInfo(qrInfo);
复制代码
上述代码中,对象的引用qrInfo刚好被单例所持有。genQrcInfo方法中每new一个对象的引用都被静态类所持有,所以这些对象的引用都不会被JVM回收,最终造成内存溢出。
4、解决方法
讲对象设置为静态成员变量,并在创建对象的时候,判断对象为空时再创建,否则直接修改对象的属性。
private static UiDataCenter.QrInfo qrInfo = null;
private UiDataCenter.QrInfo genQrcInfo(QrActionInfo info, String aType, String id)
{
if (qrInfo == null)
{
qrInfo = new UiDataCenter.QrInfo();
}
if(null != info && Ut.isValidate(id) && StringUtil.isValidate(aType))
{
qrInfo.mainBatchID = info.mainBatchID;
qrInfo.subBatchID = info.subBatchID;
qrInfo.money = info.money;
qrInfo.qrId = id;
qrInfo.aType = aType;
}
return qrInfo;
}
复制代码
此时再将对象设置到单例中,也不会造成内存的溢出,因为没有被频繁的创建对象,单例只会持有一个对象的引用。
#### YunSoul技术分享,扫码关注微信公众号##
- ——只要你学会了之前所不会的东西,只要今天的你强过了昨天的你,那你就一直是在进阶的路上了。