Java使用Jacob调用SAPI合成语音

来源:互联网 发布:丁丁长体验知乎 编辑:程序博客网 时间:2024/05/08 11:39


Java使用Jacob调用SAPI合成语音:
1.配置Jacob:我这里使用的jacob为jacob-1.18-M2,jacob-1.18-M2下载下来之后,解压缩文件,
里面的文件为docs、jacob.jar、jacob-1.18-M2-x64.dll、jacob-1.18-M2-x86.dll、LICENSE.TXT、README.txt;
把jacob.jar复制到项目的lib目录下面,jacob-1.18-M2-x64.dll和jacob-1.18-M2-x86.dll是两个动态库文件,
它们指JDK的安装版本,如果JDK是64位的,那么需要把jacob-1.18-M2-x64.dll复制到JRE BIN目录下面,因为
我装的JDK是32位的,所以我把jacob-1.18-M2-x86.dll复制到JRE BIN目录下面,
我的JRE路径为G:\Program Files (x86)\Java\jre7\bin。
2.代码如下

package com.jacob;import com.jacob.activeX.ActiveXComponent;import com.jacob.com.ComThread;import com.jacob.com.Dispatch;import com.jacob.com.Variant;/** * 调用MSTTS将字符串转换成语音信息 * 调用windows speech API(SAPI) * @author suyunlong * */public class MSTTSSpeech{private int volume=100;// 声音:1到100private int rate=0;// 频率:-10到10private int voice=0;// 语音库序号private int audio=0;// 输出设备序号private ActiveXComponent ax=null;private Dispatch spVoice=null;// 声音对象private Dispatch spFileStream=null;// 音频文件输出流对象,在读取或保存音频文件时使用private Dispatch spAudioFormat=null;// 音频格式对象private Dispatch spMMAudioOut=null;// 音频输出对象private int formatType=22;// 音频的输出格式,默认为:SAFT22kHz16BitMonopublic MSTTSSpeech() {ComThread.InitSTA();if(ax==null){ax=new ActiveXComponent("Sapi.SpVoice");spVoice=ax.getObject();}}/** * 改变语音库 * @param voice 语音库序号 */public void changeVoice(int voice){if(this.voice != voice){this.voice=voice;}try{Dispatch voiceItems=Dispatch.call(spVoice,"GetVoices").toDispatch();int count=Integer.valueOf(Dispatch.call(voiceItems,"Count").toString());if(count>0){Dispatch voiceItem=Dispatch.call(voiceItems,"Item",new Variant(this.voice)).toDispatch();Dispatch.put(spVoice,"Voice",voiceItem);}}catch(Exception e){System.out.println(e.getMessage());e.printStackTrace();}}/** * 改变音频输出设备 * @param audio 音频设备序号 */public void changeAudioOutput(int audio){if(this.audio != audio){this.audio=audio;}try{Dispatch audioOutputs=Dispatch.call(spVoice,"GetAudioOutputs").toDispatch();int count=Integer.valueOf(Dispatch.call(audioOutputs,"Count").toString());if(count > 0){Dispatch audioOutput=Dispatch.call(audioOutputs,"Item",new Variant(this.audio)).toDispatch();Dispatch.put(spVoice,"AudioOutput",audioOutput);}}catch(Exception e){System.out.println(e.getMessage());e.printStackTrace();}}/** * 播放语音  * @param text 要转换成语音的文本 */public void speak(String text){this.speak(text,0);}/** * 停止播放语音 */public void stop(){// this.speak("", 1);Dispatch.call(spVoice,"Pause");}/** * 播放语音 * @param text 要转换成语音的文本 * @param type 类型0:播放,1:停止 */private void speak(String text,int type){switch(type) {case 0:try{// 调整音量和读的速度Dispatch.put(spVoice,"Volume",new Variant(this.volume));// 设置音量Dispatch.put(spVoice,"Rate",new Variant(this.rate));// 设置速率// 设置音频格式类型if(spAudioFormat==null){ax=new ActiveXComponent("Sapi.SpAudioFormat");spAudioFormat=ax.getObject();ax=new ActiveXComponent("Sapi.SpMMAudioOut");spMMAudioOut=ax.getObject();}Dispatch.put(spAudioFormat,"Type",new Variant(this.formatType));Dispatch.putRef(spMMAudioOut,"Format",spAudioFormat);Dispatch.put(spVoice,"AllowAudioOutputFormatChangesOnNextSet",new Variant(false));Dispatch.putRef(spVoice,"AudioOutputStream",spMMAudioOut);// 开始朗读Dispatch.call(spVoice,"Speak",new Variant(text));}catch(Exception e){System.out.println(e.getMessage());e.printStackTrace();}break;case 1:try{Dispatch.call(spVoice,"Speak",new Variant(text),new Variant(2));}catch(Exception e){System.out.println(e.getMessage());e.printStackTrace();}break;default:break;}}/** * 获取系统中所有的语音库名称数组  * @return String[] */public String[] getVoices(){String[] voices=null;try{Dispatch voiceItems=Dispatch.call(spVoice,"GetVoices").toDispatch();int count=Integer.valueOf(Dispatch.call(voiceItems,"Count").toString());if(count > 0){voices=new String[count];for(int i=0;i<count;i++){Dispatch voiceItem=Dispatch.call(voiceItems,"Item",new Variant(i)).toDispatch();String voice=Dispatch.call(voiceItem,"GetDescription").toString();voices[i]=voice;}}}catch(Exception e){System.out.println(e.getMessage());e.printStackTrace();}return voices;}/** * 获取音频输出设备名称数组 * @return String[] */public String[] getAudioOutputs(){String[] result=null;try{Dispatch audioOutputs=Dispatch.call(spVoice,"GetAudioOutputs").toDispatch();int count=Integer.valueOf(Dispatch.call(audioOutputs,"Count").toString());if(count > 0){result=new String[count];for(int i=0;i<count;i++){Dispatch voiceItem=Dispatch.call(audioOutputs,"Item",new Variant(i)).toDispatch();String voice=Dispatch.call(voiceItem,"GetDescription").toString();result[i]=voice;}}}catch(Exception e){System.out.println(e.getMessage());e.printStackTrace();}return result;}/** * 将文字转换成音频信号,然后输出到.WAV文件 * @param text 文本字符串 * @param filePath 输出文件路径 */public void saveToWav(String text,String filePath){// 创建输出文件流对象ax=new ActiveXComponent("Sapi.SpFileStream");spFileStream=ax.getObject();// 创建音频流格式对象if(spAudioFormat==null){ax=new ActiveXComponent("Sapi.SpAudioFormat");spAudioFormat=ax.getObject();}// 设置音频流格式类型Dispatch.put(spAudioFormat,"Type",new Variant(this.formatType));// 设置文件输出流的格式Dispatch.putRef(spFileStream,"Format",spAudioFormat);// 调用输出文件流对象的打开方法,创建一个.wav文件Dispatch.call(spFileStream,"Open",new Variant(filePath),new Variant(3),new Variant(true));// 设置声音对象的音频输出流为输出文件流对象Dispatch.putRef(spVoice,"AudioOutputStream",spFileStream);// 调整音量和读的速度Dispatch.put(spVoice,"Volume",new Variant(this.volume));// 设置音量Dispatch.put(spVoice,"Rate",new Variant(this.rate));// 设置速率// 开始朗读Dispatch.call(spVoice,"Speak",new Variant(text));// 关闭输出文件流对象,释放资源Dispatch.call(spFileStream,"Close");Dispatch.putRef(spVoice,"AudioOutputStream",null);}/** * @return the volume */public int getVolume(){return volume;}/** * @param volume * the volume to set */public void setVolume(int volume){this.volume = volume;}/** * @return the rate */public int getRate(){return rate;}/** * @param rate * the rate to set */public void setRate(int rate){this.rate = rate;}/** * @return the voice */public int getVoice(){return voice;}/** * @param voice * the voice to set */public void setVoice(int voice){this.voice = voice;}/** * @return the audio */public int getAudio(){return audio;}/** * @param audio * the audio to set */public void setAudio(int audio){this.audio=audio;}/** * @return the ax */public ActiveXComponent getAx(){return ax;}/** * @param ax * the ax to set */public void setAx(ActiveXComponent ax){this.ax=ax;}/** * @return the formatType */public int getFormatType(){return formatType;}/** * 设置音频输出格式类型<br> * SAFTDefault = -1<br> * SAFTNoAssignedFormat = 0<br> * SAFTText = 1<br> * SAFTNonStandardFormat = 2<br> * SAFTExtendedAudioFormat = 3<br> * // Standard PCM wave formats<br> * SAFT8kHz8BitMono = 4<br> * SAFT8kHz8BitStereo = 5<br> * SAFT8kHz16BitMono = 6<br> * SAFT8kHz16BitStereo = 7<br> * SAFT11kHz8BitMono = 8<br> * SAFT11kHz8BitStereo = 9<br> * SAFT11kHz16BitMono = 10<br> * SAFT11kHz16BitStereo = 11<br> * SAFT12kHz8BitMono = 12<br> * SAFT12kHz8BitStereo = 13<br> * SAFT12kHz16BitMono = 14<br> * SAFT12kHz16BitStereo = 15<br> * SAFT16kHz8BitMono = 16<br> * SAFT16kHz8BitStereo = 17<br> * SAFT16kHz16BitMono = 18<br> * SAFT16kHz16BitStereo = 19<br> * SAFT22kHz8BitMono = 20<br> * SAFT22kHz8BitStereo = 21<br> * SAFT22kHz16BitMono = 22<br> * SAFT22kHz16BitStereo = 23<br> * SAFT24kHz8BitMono = 24<br> * SAFT24kHz8BitStereo = 25<br> * SAFT24kHz16BitMono = 26<br> * SAFT24kHz16BitStereo = 27<br> * SAFT32kHz8BitMono = 28<br> * SAFT32kHz8BitStereo = 29<br> * SAFT32kHz16BitMono = 30<br> * SAFT32kHz16BitStereo = 31<br> * SAFT44kHz8BitMono = 32<br> * SAFT44kHz8BitStereo = 33<br> * SAFT44kHz16BitMono = 34<br> * SAFT44kHz16BitStereo = 35<br> * SAFT48kHz8BitMono = 36<br> * SAFT48kHz8BitStereo = 37<br> * SAFT48kHz16BitMono = 38<br> * SAFT48kHz16BitStereo = 39<br> * <br> * // TrueSpeech format<br> * SAFTTrueSpeech_8kHz1BitMono = 40<br> * // A-Law formats<br> * SAFTCCITT_ALaw_8kHzMono = 41<br> * SAFTCCITT_ALaw_8kHzStereo = 42<br> * SAFTCCITT_ALaw_11kHzMono = 43<br> * SAFTCCITT_ALaw_11kHzStereo = 4<br> * SAFTCCITT_ALaw_22kHzMono = 44<br> * SAFTCCITT_ALaw_22kHzStereo = 45<br> * SAFTCCITT_ALaw_44kHzMono = 46<br> * SAFTCCITT_ALaw_44kHzStereo = 47<br> * <br> * // u-Law formats<br> * SAFTCCITT_uLaw_8kHzMono = 48<br> * SAFTCCITT_uLaw_8kHzStereo = 49<br> * SAFTCCITT_uLaw_11kHzMono = 50<br> * SAFTCCITT_uLaw_11kHzStereo = 51<br> * SAFTCCITT_uLaw_22kHzMono = 52<br> * SAFTCCITT_uLaw_22kHzStereo = 53<br> * SAFTCCITT_uLaw_44kHzMono = 54<br> * SAFTCCITT_uLaw_44kHzStereo = 55<br> * SAFTADPCM_8kHzMono = 56<br> * SAFTADPCM_8kHzStereo = 57<br> * SAFTADPCM_11kHzMono = 58<br> * SAFTADPCM_11kHzStereo = 59<br> * SAFTADPCM_22kHzMono = 60<br> * SAFTADPCM_22kHzStereo = 61<br> * SAFTADPCM_44kHzMono = 62<br> * SAFTADPCM_44kHzStereo = 63<br> * <br> * // GSM 6.10 formats<br> * SAFTGSM610_8kHzMono = 64<br> * SAFTGSM610_11kHzMono = 65<br> * SAFTGSM610_22kHzMono = 66<br> * SAFTGSM610_44kHzMono = 67<br> * // Other formats<br> * SAFTNUM_FORMATS = 68<br> *  * @param formatType *            音频输出格式类型 */public void setFormatType(int formatType){this.formatType=formatType;}public static void main(String[] args){MSTTSSpeech speech=new MSTTSSpeech();String text="这是我的测试,物理内存至少需要512MB,建议2GB以上,虚拟内存是主机物理内存的两倍,不要设到系统盘,硬盘空闲空间大于4.77GB.";speech.setFormatType(6);// speech.setRate(-1);speech.saveToWav(text,"E://syl//C//test.wav");//speech.speak(text);SoundUtil.wavToPcm("E://syl//C//test.wav");}}

3.把wav语音文件转换为pcm,其中用的Jar包为sound.jar,代码如下:

package com.jacob;import java.io.File;import java.io.IOException;import javax.sound.sampled.AudioFileFormat;import javax.sound.sampled.AudioFormat;import javax.sound.sampled.AudioInputStream;import javax.sound.sampled.AudioSystem;import javax.sound.sampled.UnsupportedAudioFileException;/** * 与声音有关的工具类 * @author suyunlong * */public class SoundUtil{/** * 将WAV格式的音频文件转换为PCM格式的文件 * @param wavFilePath WAV文件路径 * @throws UnsupportedAudioFileException * @throws IOException */public static void wavToPcm(String wavFilePath){try{File wavFile=new File(wavFilePath);AudioInputStream sourceAudioInputStream=AudioSystem.getAudioInputStream(wavFile);// 此处的转换必需是16bit的音频文件AudioInputStream targetAudioInputStream=AudioSystem.getAudioInputStream(AudioFormat.Encoding.ULAW,sourceAudioInputStream);String newFilePath=wavFilePath.substring(0,wavFilePath.lastIndexOf(".")+1)+"pcm";AudioSystem.write(targetAudioInputStream,AudioFileFormat.Type.WAVE,new File(newFilePath));}catch(UnsupportedAudioFileException e){System.out.println(e.getMessage());e.printStackTrace();}catch(IOException e){System.out.println(e.getMessage());e.printStackTrace();}}/** * 获取音频文件的编码格式  * @param wavFilePath 音频文件格式 * @return String */public static String getWavFormat(String wavFilePath){File wavFile=new File(wavFilePath);AudioInputStream ais;String result="";try{ais=AudioSystem.getAudioInputStream(wavFile);AudioFormat af=ais.getFormat();result=af.toString();}catch(UnsupportedAudioFileException e){System.out.println(e.getMessage());e.printStackTrace();}catch(IOException e){System.out.println(e.getMessage());e.printStackTrace();}return result;}public static void main(String[] args){System.out.println(SoundUtil.getWavFormat("E://syl//C//test.wav"));SoundUtil.wavToPcm("E://syl//C//test.wav");}}

用到的架包可以从这里下载:

jacob-1.18-M2架包

java sound架包

0 0
原创粉丝点击