记一次 开发语音播报遇到的window环境不稳定的问题
windows环境理论上是比Linux环境不稳定的,linux大法好,但界面显示还是要在window上做的,开发的一个语音播报程序就在一个win7系统中会突然闪退崩溃,其他电脑都没有这个问题,就唯独这台电脑会。
1.首先看window日志,排查具体闪退原因,.net好处就在于window日志比较清晰233,可以清楚地知道问题在哪。
而这次在window日志中就可以看到是SpeechLib的库执行Speak函数时抛出了AccessViolationException 异常,就是非法访问内存,这个我改不动,第三方库抛出的异常,而且传入的参数也没什么问题。
然而.net4.0通常并不能捕捉调用第三方dll所抛出的异常,如果只是throw new AccessViolationException的话是可以捕捉的,要想捕捉就得在捕捉异常的函数上加上[HandleProcessCorruptedStateExceptions] ,想测试捕捉该异常,可以编写一个c++dll
编写一个函数
void TestMethod()
{
int *p=NULL;
*p=10;
}
后调用该dll方法就可以模拟该异常。
然而捕捉了异常,程序是不闪退了,但程序业务在捕捉异常后,再执行下一次Speak依旧没办法执行,甚至导致播报线程阻塞,原理估计得看SpeechLib的Speak函数源码,报错原因是设备正忙,无法播报语音。
解决方法是
SpVoice voice
voice.Stop();
voice = null;
GC.Collect();
voice = new SpVoice();
强制释放旧对象的内存并重置voice对象,这样就能让播报线程正常运作了