您的位置:首页 > 其它

解决Win7下使用waveIn系列函数录音时插拔麦克风时的死锁问题

2014-11-19 15:34 155 查看
最近由于项目需要,使用waveIn系列函数进行录音,程序一直工作得很好。直到有一次启动程序后发现没有插入麦克风,在没有关闭程序的情况下插入麦克风,之后程序就无法退出了。后来测试了在程序运行时拔掉麦克风的情况,也无法退出。

先说一下程序,waveInOpen函数支持多种方式获取数据,例如回调函数、线程、窗口等。回调函数的方式无疑是最简单的,本程序就是采用的这种方式。通过调试程序发现在程序运行过程中插拔麦克风后会阻塞在回调函数中的任何waveIn调用中,也即发生了死锁。

为什么会发生死锁,由于无法看到Winmm的源码,不好妄下结论。但是有一点可以确定,使用回调方式录音,回调函数是在录音线程中执行的。而在录音线程中执行回调函数是不太安全的,例如录音线程使用Mutex锁。Mutex在一个线程和多个线程中的表现是不同的,当一个线程获取到Mutex后,那么该线程后续任意多次调用WaitForSingleObject都将立即返回。作者以前就曾遇到过这个问题,当时也是弄了好久,后来改用Semaphore解决的。

基于上面的思考,作者新开了一个线程,在回调函数中将所有信息使用PostThreadMessage传递给新线程处理,问题果然解决了。其实这种方式还有点绕,可以直接在waveInOpen中传入线程ID,省掉一个回调函数。作者对这两种方法都进行了测试,都可以解决问题,并且完美支持麦克风切换,中间会稍微卡顿一两秒中,waveInOpen传入的参数是WAVE_MAPPER。

PS:作者使用的是笔记本电脑,拥有内置和外置共两个麦克风,操作系统为Win7 32位。无法测试只有一个麦克风的情况,另外也没有测试使用窗口的情况,不过理论上说使用窗口也处于录音线程意外的线程中,应该也不会出现问题。最后,作者水平有限,关于文中错误的地方,欢迎斧正。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: