【开发实例】C#调用SAPI实现语音合成的两种方法

我们都知道现在的语音合成TTS是可以通过微软的SAPI实现的,好处我就不多说了,方便而已,因为在微软的操作系统里面就自带了这个玩意,主要的方式有两种:  1、使用COM组件技术,不管是C++,C#,Delphi都能玩的转,开发出来的东西在XP和WIN7都能跑。(要引入SpeechLib,好像在项目上点引用,然后选到系统COM吧,好久没弄,记不清楚了)  2、使用WIN7的windows api,其实最终还是调用了SAPI,所以开发出来的东西就只能在WIN7上面跑。  其实不管是哪一种,都是调用SAPI,可能后一种代码比较简单,使用已经安装的TTS引擎,现在一般用NeoSpeech,这个就不解释了,太强大了这个发音。。。  COM组件技术:
  1. public class Speach
  2. {
  3. private static Speach _Instance = null ;
  4. private SpeechLib.SpVoiceClass voice =null; //SAPI5.1
  5. private SpeechLib.SpVoice voice = null;//SAPI 5.4
  6. private Speach()
  7. {
  8. BuildSpeach() ;
  9. }
  10. public static Speach instance()
  11. {
  12. if (_Instance == null)
  13. _Instance = new Speach() ;
  14. return _Instance ;
  15. }
  16. private void SetChinaVoice()
  17. {
  18. voice.Voice = voice.GetVoices(string.Empty,string.Empty).Item(0) ;
  19. }
  20. private void SetEnglishVoice()
  21. {
  22. voice.Voice = voice.GetVoices(string.Empty,string.Empty).Item(1) ;
  23. }
  24. private void SpeakChina(string strSpeak)
  25. {
  26. SetChinaVoice() ;
  27. Speak(strSpeak) ;
  28. }
  29. private void SpeakEnglishi(string strSpeak)
  30. {
  31. SetEnglishVoice() ;
  32. Speak(strSpeak) ;
  33. }
  34. public void AnalyseSpeak(string strSpeak)
  35. {
  36. int iCbeg = 0 ;
  37. int iEbeg = 0 ;
  38. bool IsChina = true ;
  39. for(int i=0;i<strSpeak.Length;i++)
  40. {
  41. char chr = strSpeak[i] ;
  42. if (IsChina)
  43. {
  44. if (chr<=122&&chr>=65)
  45. {
  46. int iLen = i - iCbeg ;
  47. string strValue = strSpeak.Substring(iCbeg,iLen) ;
  48. SpeakChina(strValue) ;
  49. iEbeg = i ;
  50. IsChina = false ;
  51. }
  52. }
  53. else
  54. {
  55. if (chr>122||chr<65)
  56. {
  57. int iLen = i - iEbeg ;
  58. string strValue = strSpeak.Substring(iEbeg,iLen) ;
  59. this.SpeakEnglishi(strValue) ;
  60. iCbeg = i ;
  61. IsChina = true ;
  62. }
  63. }
  64. }//end for
  65. if (IsChina)
  66. {
  67. int iLen = strSpeak.Length - iCbeg ;
  68. string strValue = strSpeak.Substring(iCbeg,iLen) ;
  69. SpeakChina(strValue) ;
  70. }
  71. else
  72. {
  73. int iLen = strSpeak.Length - iEbeg ;
  74. string strValue = strSpeak.Substring(iEbeg,iLen) ;
  75. SpeakEnglishi(strValue) ;
  76. }
  77. }
  78. private void BuildSpeach()
  79. {
  80. if (voice == null)
  81. voice = new SpVoiceClass() ;
  82. }
  83. public int Volume
  84. {
  85. get
  86. {
  87. return voice.Volume ;
  88. }
  89. set
  90. {
  91. voice.SetVolume((ushort)(value)) ;
  92. }
  93. }
  94. public int Rate
  95. {
  96. get
  97. {
  98. return voice.Rate ;
  99. }
  100. set
  101. {
  102. voice.SetRate(value) ;
  103. }
  104. }
  105. private void Speak(string strSpeack)
  106. {
  107. try
  108. {
  109. voice.Speak(strSpeack,SpeechVoiceSpeakFlags.SVSFlagsAsync) ;
  110. }
  111. catch(Exception err)
  112. {
  113. throw(new Exception("发生一个错误:"+err.Message)) ;
  114. }
  115. }
  116. public void Stop()
  117. {
  118. voice.Speak(string.Empty,SpeechLib.SpeechVoiceSpeakFlags.SVSFPurgeBeforeSpeak) ;
  119. }
  120. public void Pause()
  121. {
  122. voice.Pause() ;
  123. }
  124. public void Continue()
  125. {
  126. voice.Resume() ;
  127. }
  128. }//end class

在 private SpeechLib.SpVoiceClass voice =null;这里,我们定义个一个用来发音的类,并且在第一次调用该类时,对它用BuildSpeach方法进行了初始化。 
我们还定义了两个属性Volume和Rate,能够设置音量和语速。 
我们知道,SpVoiceClass 有一个Speak方法,我们发音主要就是给他传递一个字符串,它负责读出该字符串,如下所示。

  1. private void Speak(string strSpeack)
  2. {
  3. try
  4. {
  5. voice.Speak(strSpeack,SpeechVoiceSpeakFlags.SVSFlagsAsync) ;
  6. }
  7. catch(Exception err)
  8. {
  9. throw(new Exception("发生一个错误:"+err.Message)) ;
  10. }
  11. }

第二种使用.NET类库和系统API的代码如下:

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Linq;
  4. using System.Text;
  5. using System.Speech.Synthesis;
  6. using System.Speech;
  7. namespace StudyBeta
  8. {
  9. public class SRead
  10. {
  11. public SpeechSynthesizer synth; //语音合成对象
  12. public SRead()
  13. {
  14. synth = new SpeechSynthesizer();
  15. }
  16. public SRead(int m, int n)
  17. {
  18. //使用 synth 设置朗读音量 [范围 0 ~ 100]
  19. synth.Volume = m;
  20. //使用 synth 设置朗读频率 [范围 -10 ~ 10]
  21. synth.Rate = n;
  22. }
  23. public void SpeakChina(string ggg)
  24. {
  25. //SpVoice Voice = new SpVoice();
  26. synth.SelectVoice("Microsoft Lili");
  27. //Voice.Speak(ggg, SpFlags);
  28. synth.SpeakAsync(ggg);
  29. //String speechPeople = synth.Voice;
  30. //使用 synth 设置朗读音量 [范围 0 ~ 100]
  31. // synth.Volume = 80;
  32. //使用 synth 设置朗读频率 [范围 -10 ~ 10]
  33. //      synth.Rate = 0;
  34. //使用synth 合成 wav 音频文件:
  35. //synth.SetOutputToWaveFile(string path);
  36. }
  37. public void SpeakEnglish(string ggg)
  38. {
  39. //SpVoice Voice = new SpVoice();
  40. synth.SelectVoice("VW Julie");
  41. synth.Speak(ggg); //ggg为要合成的内容
  42. }
  43. public int m
  44. {
  45. get
  46. {
  47. return synth.Volume;
  48. }
  49. set
  50. {
  51. synth.Volume = value;
  52. }
  53. }
  54. public int n
  55. {
  56. get
  57. {
  58. return synth.Rate;
  59. }
  60. set
  61. {
  62. synth.Rate = value;
  63. }
  64. }
  65. }
 
上一篇:Azure SQL Data Warehouse


下一篇:.NET NLog 详解 (三) - LayoutRender