阅读更多
问题:
public class DaoZ {
public static Dao dao() { // 每当需要使用dao的时候就取一次
Ioc ioc = new NutIoc(new JsonLoader("dao.js"));
return ioc.get(Dao.class);
}
}
----------------------------------------------------------------------------
Wedal解析:
注意, java.sql.DataSource只是个接口.但下面讨论的DataSource均指代数据库连接池实现.
根本原因在于: Nutz的Ioc容器不是全局的,一旦没有被引用,就随时被GC.
题目中的NutIoc实例,在方法返回之后,处于无引用的状态, 即满足了GC的条件,随时会被GC.
题目中默认为2个基本ioc对象, DataSource和Dao, 其中Dao默认指代NutDao
NutIoc被gc时,会触发depose事件, 从而调用各ioc对象的depose/close方法
一般来说DataSource都会配置成depose事件触发close方法,即关闭连接池. 这样就导致DataSource对象处于关闭状态, 而NutDao的数据库连接均由DataSource提供, 直接导致NutDao实例在操作任意数据库方法时报错.
注意, GC的时机是几乎不可控,不可预测的, 这就意味着上述的ioc容器被gc, depose方法被调用,DataSource被关闭的时机也是不可控及不可预测的. 这会导致程序有时候是好的(GC在Dao方法执行完成之后),有时候会挂的(GC在Dao执行完成之前)
错误的解决思路:
1. 去掉depose事件. 首先是连接池都有初始大小和最小活跃数,这意味着DataSource被gc前总持有那么几个数据库连接. 并发上去了,gc来不及, 很快就数据库连接耗尽.
2. 那,去掉depose事件,连接池初始化大小和最小活跃数都设置为0呢. 初始化大小为0, 但你总会使用一次NutDao的方法吧, 那么就建立了一个连接, 然后最小活跃数是0, 将连接数从1降为0那么, 总有那么一个时间差,然后, 呵呵, 一个坑就摆在那了, 哪天碰巧没能快速关闭,也没被gc, 爆数据库连接数
3. 好吧,那我不用连接池,可以了吧? 可以,性能掉渣
解决方法:
1 在Mvc中, 你几乎100%会用到@IocBy, 也就是NutMvc来持有ioc, 那么你就不别自行new NutIoc了. 在Setup接口中,你可以通过config.getIoc()来获取Ioc容器(常见于启动时的初始化,加载全局变量,计划任务什么的), 在任何不可能直接得到Ioc容器的地方, 通过 Mvcs.ctx().getDefaultIoc() 获取 .注意, Mvcs.ctx()总是非常规方法.
2 非Mvc环境, 应使用一个工具类管理, 而且必须持有Ioc容器
3
public class DaoZ { // 供参考
private static Ioc ioc; // 静态属性,确保了Ioc容器不会被gc
public static synchronized Ioc ioc() { // 同步,用你喜欢的方法就好啦
if (ioc == null) {
try {
ioc = new NutIoc(new ComboIocLoader("*js","ioc/", "*annotation", "net.wendal.iot"));
}
catch (ClassNotFoundException e) {
throw Lang.warp(e);
}
}
return ioc;
}
public static Dao dao() {
return ioc.get(Dao.class);
}
}