nhibernate1.2支持access数据库。但是我一直在用sqlserver开发。现在已经有了一点样子,于是我心血来潮,将数据库切换为Access,看看执行的效果如何。
碰,出错了。Sql语句语法错误。我靠,果然有错误,改改,再试?还是出错。郁闷了。打断点开始调试,确定我的代码没有错。我没错那谁错了?nhibernate??不会吧?不过事已至此,又不可能重新写,只好硬着头皮上了,呵呵。将nhibernate的引用移除,直接引用nhibernate的源码开始调试,没想到啊没想到,居然还真的被我给找出bug来了。
nhibernate处理access时,使用的是jetdriver这个方言。由于access的特殊性(sql语句做连接时,必须要用括号),于是在jetdriver里,对sql语句进行了处理。代码在此:jetdriver.cs的96行处有这么一段
private SqlString FinalizeJoins(SqlString sqlString)
{
if (_queryCache.Contains(sqlString))
{
return (SqlString) _queryCache[sqlString];
}
// fix wxy
int beginOfFrom = sqlString.IndexOfCaseInsensitive(" from ");
int endOfFrom = sqlString.IndexOfCaseInsensitive(" where ");
if (beginOfFrom < 0)
{
return sqlString;
}
if (endOfFrom < 0)
{
endOfFrom = sqlString.Length;
}
string fromClause = sqlString.Substring(beginOfFrom, endOfFrom - beginOfFrom).ToString();
string transformedFrom = TransformFromClause(fromClause);
//put it all together again
SqlStringBuilder final = new SqlStringBuilder(sqlString.Count + 1);
final.Add(sqlString.Substring(0, beginOfFrom));
final.Add(transformedFrom);
final.Add(sqlString.Substring(endOfFrom));
SqlString ret = final.ToSqlString();
_queryCache[sqlString] = ret;
return ret;
}
这段的代码的意思呢,是取出from后面的部分,以对join语句进行access语法的特别加工。原来bug就在这里,
他在查找关键字时,是按照"from","where"这两个,但是,from,where这两个词必须要在头尾加上空格才能保证不会和其他单词混淆。原文是没有空格的。这个好像也算是低级错误呀,呵呵,测试不仔细!BS一下再说
改成我这样写的,再试,OK!这个问题没了。没高兴一会,碰!又出错了。再一跟,OMG, jetdriver.cs里的bug还真不少,又来一个:132行处,返回的字符串居然把 from 关键字给吃掉了,晕,他们到底测试过没啊??改成像我这样就好了
private string TransformFromClause(string fromClause)
{
string transformed;
//"from ".Length
const int fromLength = 5;
fromClause = fromClause.Substring(fromLength, fromClause.Length - fromLength);
string[] blocks = fromClause.Split(',');
if (blocks.Length > 1)
{
for (int i = 0; i < blocks.Length; i++)
{
string tr = TransformJoinBlock(blocks[i]);
if (tr.IndexOf(" join ") > -1)
{
blocks[i] = "(select * from " + tr + ") as jetJoinAlias" + i.ToString();
}
else
{
blocks[i] = tr;
}
}
// wxy fixed
transformed = " from " + string.Join(",", blocks);
}
else
{
transformed = " from " + TransformJoinBlock(blocks[0]);
}
return transformed;
}
OK!,一切顺利,我的系统在sql,access两个数据库下都顺利的跑起来了!庆祝一下呵呵.