同时播放三个声音c#

本文关键字:三个 声音 播放 | 更新日期: 2023-09-27 18:08:09

我想同时播放三个声音,但第二个声音必须在一秒钟后播放,第三个声音在两秒钟后播放。我有这个代码:

private void Play()
        {
            AxWindowsMediaPlayer player1 = new AxWindowsMediaPlayer();
            player1.CreateControl();
            AxWindowsMediaPlayer player2 = new AxWindowsMediaPlayer();
            player2.CreateControl();
            AxWindowsMediaPlayer player3 = new AxWindowsMediaPlayer();
            player3.CreateControl();
            player1.URL = "sounds''1.wav";
            player1.Ctlcontrols.play();
            System.Threading.Thread.Sleep(1000);
            player2.URL = "sounds''2.wav";
            player2.Ctlcontrols.play();
            System.Threading.Thread.Sleep(1000);
            player3.URL = "sounds''3.wav";
            player3.Ctlcontrols.play();

为什么两秒钟后所有这些声音都在一次内播放?

同时播放三个声音c#

我最终使用了SharpDX(可通过NuGet包SharpDX获得和CCD_ 2。

在我的一个GitHub项目中可以找到它的用法示例:2DAI在这个屏幕录音中,你也可以听到各种声音的重叠。

播放声音:

var backgroundMusicSound = new AudioClip(@".'Assets'Sounds'Music'Background.wav" /*Sound path*/, 1.0 /* Volumne*/, true /*Loop Forever?*/)
backgroundMusicSound.Play();

我拼凑的类:

public class AudioClip
{
    private XAudio2 _xaudio = new XAudio2();
    private WaveFormat _waveFormat;
    private AudioBuffer _buffer;
    private SoundStream _soundstream;
    private SourceVoice _singleSourceVoice;
    private bool _loopForever;
    private bool _isPlaying = false; //Only applicable when _loopForever == false;
    private bool _isFading;
    private string _wavFilePath; //For debugging.
    private float _initialVolumne;
    public AudioClip(string wavFilePath, float initialVolumne = 1, bool loopForever = false)
    {
        _loopForever = loopForever;
        _wavFilePath = wavFilePath;
        _initialVolumne = initialVolumne;
        var masteringsound = new MasteringVoice(_xaudio); //Yes, this is required.
        var nativefilestream = new NativeFileStream(wavFilePath,
            NativeFileMode.Open, NativeFileAccess.Read, NativeFileShare.Read);
        _soundstream = new SoundStream(nativefilestream);
        _waveFormat = _soundstream.Format;
        _buffer = new AudioBuffer
        {
            Stream = _soundstream.ToDataStream(),
            AudioBytes = (int)_soundstream.Length,
            Flags = BufferFlags.EndOfStream
        };
        if (loopForever)
        {
            _buffer.LoopCount = 100;
        }
    }
    public void Play()
    {
        lock (this)
        {
            if (_loopForever == true)
            {
                if (_isPlaying)
                {
                    if (_isFading)
                    {
                        _isFading = false;
                        _singleSourceVoice.SetVolume(_initialVolumne);
                    }
                    return;
                }
                _singleSourceVoice = new SourceVoice(_xaudio, _waveFormat, true);
                _singleSourceVoice.SubmitSourceBuffer(_buffer, _soundstream.DecodedPacketsInfo);
                _singleSourceVoice.SetVolume(_initialVolumne);
                _singleSourceVoice.Start();
                _isPlaying = true;
                return;
            }
        }
        var sourceVoice = new SourceVoice(_xaudio, _waveFormat, true);
        sourceVoice.SubmitSourceBuffer(_buffer, _soundstream.DecodedPacketsInfo);
        sourceVoice.SetVolume(_initialVolumne);
        sourceVoice.Start();
    }
    public void Fade()
    {
        if (_isPlaying && _isFading == false)
        {
            _isFading = true;
            (new Thread(FadeThread)).Start();
        }
    }
    private void FadeThread()
    {
        float volumne;
        _singleSourceVoice.GetVolume(out volumne);
        while (_isFading && volumne > 0)
        {
            volumne -= 0.25f;
            volumne = volumne < 0 ? 0 : volumne;
            _singleSourceVoice.SetVolume(volumne);
            Thread.Sleep(100);
        }
        Stop();
    }
    public void Stop()
    {
        if (_loopForever == true)
        {
            if (_singleSourceVoice != null && _isPlaying)
            {
                _singleSourceVoice.Stop();
            }
            _isPlaying = false;
            _isFading = false;
        }
        else
        {
            throw new Exception("Cannot stop overlapped audio.");
        }
    }
}

还应该注意的是,加载声音可能是一个繁重的过程,所以如果你经常这样做,那么你可能想像我一样缓存它们:

private Dictionary<string, AudioClip> _audioClips { get; set; } = new Dictionary<string, AudioClip>();
public AudioClip GetSoundCached(string wavFilePath, float initialVolumne, bool loopForever = false)
{
    lock (_audioClips)
    {
        AudioClip result = null;
        wavFilePath = wavFilePath.ToLower();
        if (_audioClips.ContainsKey(wavFilePath))
        {
            result = _audioClips[wavFilePath];
        }
        else
        {
            result = new AudioClip(wavFilePath, initialVolumne, loopForever);
            _audioClips.Add(wavFilePath, result);
        }
        return result;
    }
}