欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java 調(diào)用本地?fù)P聲器的步驟

 更新時(shí)間:2021年05月14日 14:22:52   作者:青衫染紅塵  
博主的畢設(shè)系統(tǒng)在做一個(gè)餐廳的點(diǎn)餐管理系統(tǒng),在進(jìn)行移動(dòng)端頁(yè)面開(kāi)發(fā)的時(shí)候突發(fā)奇想做一個(gè)呼叫功能,因此就有了這篇文章

實(shí)現(xiàn)方式

接下來(lái)就對(duì)這個(gè)小功能進(jìn)行分析和實(shí)現(xiàn)。先寫(xiě)一個(gè)Demo。

1.首先,我們需要一個(gè)dll作為輔助。這里解釋一下dll的含義(DLL(Dynamic Link Library)文件為動(dòng)態(tài)鏈接庫(kù)文件,又稱(chēng)“應(yīng)用百程序拓展”,是軟件文件類(lèi)型。在Windows中,許多應(yīng)用程序并不是一個(gè)度完整的可執(zhí)行文件,它們被分割成一些相知對(duì)獨(dú)立的動(dòng)態(tài)鏈接庫(kù),即DLL文件,放置于道系統(tǒng)中。當(dāng)我們執(zhí)行某一個(gè)程序時(shí),相應(yīng)的版DLL文件就會(huì)被調(diào)用。一個(gè)應(yīng)用程序可使用權(quán)多個(gè)DLL文件,一個(gè)DLL文件也可能被不同的應(yīng)用程序使用,這樣的DLL文件被稱(chēng)為共享DLL文件)。

需要把jacob-1.17-M2-x64.dll復(fù)制到C:\Windows\System32\目錄下。我們也能看到目錄下有很多的.dll文件。

這里的文件大家自己百度下,很好找的。

2.使用maven項(xiàng)目導(dǎo)入坐標(biāo)。

<!-- https://mvnrepository.com/artifact/net.sf.jacob-project/jacob -->
<dependency>
	<groupId>net.sf.jacob-project</groupId>
	<artifactId>jacob</artifactId>
	<version>1.14.3</version>
</dependency>

3.測(cè)試類(lèi)代碼。

/**
 * 文字轉(zhuǎn)語(yǔ)音測(cè)試 jdk bin文件中需要導(dǎo)入jacob-1.17-M2-x64.dll
 * 注意導(dǎo)包哈
 * @date: 2020年2月25日 上午10:05:21
 */
public class Jacobtest {


    public static void main(String[] args) {
        textToSpeech("工作人員請(qǐng)注意,桌號(hào)8001顧客正在尋求幫助!!");
    }

    /**
     * 語(yǔ)音轉(zhuǎn)文字并播放
     *
     * @param text
     */
    public static void textToSpeech(String text) {
        ActiveXComponent ax = null;
        try {
            ax = new ActiveXComponent("Sapi.SpVoice");

            // 運(yùn)行時(shí)輸出語(yǔ)音內(nèi)容
            Dispatch spVoice = ax.getObject();
            // 音量 0-100
            ax.setProperty("Volume", new Variant(100));
            // 語(yǔ)音朗讀速度 -10 到 +10
            ax.setProperty("Rate", new Variant(0));
            // 執(zhí)行朗讀
            Dispatch.call(spVoice, "Speak", new Variant(text));

           /* // 下面是構(gòu)建文件流把生成語(yǔ)音文件

            ax = new ActiveXComponent("Sapi.SpFileStream");
            Dispatch spFileStream = ax.getObject();

            ax = new ActiveXComponent("Sapi.SpAudioFormat");
            Dispatch spAudioFormat = ax.getObject();

            // 設(shè)置音頻流格式
            Dispatch.put(spAudioFormat, "Type", new Variant(22));
            // 設(shè)置文件輸出流格式
            Dispatch.putRef(spFileStream, "Format", spAudioFormat);
            // 調(diào)用輸出 文件流打開(kāi)方法,創(chuàng)建一個(gè).wav文件
            Dispatch.call(spFileStream, "Open", new Variant("./text.wav"), new Variant(3), new Variant(true));
            // 設(shè)置聲音對(duì)象的音頻輸出流為輸出文件對(duì)象
            Dispatch.putRef(spVoice, "AudioOutputStream", spFileStream);
            // 設(shè)置音量 0到100
            Dispatch.put(spVoice, "Volume", new Variant(100));
            // 設(shè)置朗讀速度
            Dispatch.put(spVoice, "Rate", new Variant(-2));
            // 開(kāi)始朗讀
            Dispatch.call(spVoice, "Speak", new Variant(text));

            // 關(guān)閉輸出文件
            Dispatch.call(spFileStream, "Close");
            Dispatch.putRef(spVoice, "AudioOutputStream", null);

            spAudioFormat.safeRelease();
            spFileStream.safeRelease();*/
            spVoice.safeRelease();
            ax.safeRelease();

        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

4.從測(cè)試類(lèi)可以看出,這個(gè)方法既可以發(fā)聲還能輸出后綴為.wav的文件,這是一個(gè)標(biāo)準(zhǔn)的多媒體文件。上述代碼注釋很清晰,就不解釋了,自己看哈。

5.測(cè)試成功,現(xiàn)在集成到自己的項(xiàng)目中。

另述

這里說(shuō)到了調(diào)用揚(yáng)聲器發(fā)聲,不放還可以想一下如何調(diào)用麥克風(fēng)收音。

public class EngineeCore {
    String filePath = "E:\\voice\\voice_cache.wav";
    AudioFormat audioFormat;
    TargetDataLine targetDataLine;
    boolean flag = true;
    
	private void stopRecognize() {
        flag = false;
        targetDataLine.stop();
        targetDataLine.close();
    }
    private AudioFormat getAudioFormat() {
        float sampleRate = 16000;
        // 8000,11025,16000,22050,44100
        int sampleSizeInBits = 16;
        // 8,16
        int channels = 1;
        // 1,2
        boolean signed = true;
        // true,false
        boolean bigEndian = false;
        // true,false
        return new AudioFormat(sampleRate, sampleSizeInBits, channels, signed, bigEndian);
    }// end getAudioFormat


    private void startRecognize() {
        try {
            // 獲得指定的音頻格式
            audioFormat = getAudioFormat();
            DataLine.Info dataLineInfo = new DataLine.Info(TargetDataLine.class, audioFormat);
            targetDataLine = (TargetDataLine) AudioSystem.getLine(dataLineInfo);
            // Create a thread to capture the microphone
            // data into an audio file and start the
            // thread running. It will run until the
            // Stop button is clicked. This method
            // will return after starting the thread.
            flag = true;
            new CaptureThread().start();
        } catch (Exception e) {
            e.printStackTrace();
        } // end catch
    }// end captureAudio method

    class CaptureThread extends Thread {
        public void run() {
            AudioFileFormat.Type fileType = null;
            File audioFile = new File(filePath);

            fileType = AudioFileFormat.Type.WAVE;
            //聲音錄入的權(quán)值
            int weight = 2;
            //判斷是否停止的計(jì)數(shù)
            int downSum = 0;

            ByteArrayInputStream bais = null;
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            AudioInputStream ais = null;
            try {
                targetDataLine.open(audioFormat);
                targetDataLine.start();
                byte[] fragment = new byte[1024];

                ais = new AudioInputStream(targetDataLine);
                while (flag) {

                    targetDataLine.read(fragment, 0, fragment.length);
                    //當(dāng)數(shù)組末位大于weight時(shí)開(kāi)始存儲(chǔ)字節(jié)(有聲音傳入),一旦開(kāi)始不再需要判斷末位
                    if (Math.abs(fragment[fragment.length-1]) > weight || baos.size() > 0) {
                        baos.write(fragment);
                        System.out.println("守衛(wèi):"+fragment[0]+",末尾:"+fragment[fragment.length-1]+",lenght"+fragment.length);
                        //判斷語(yǔ)音是否停止
                        if(Math.abs(fragment[fragment.length-1])<=weight){
                            downSum++;
                        }else{
                            System.out.println("重置奇數(shù)");
                            downSum=0;
                        }
               //計(jì)數(shù)超過(guò)20說(shuō)明此段時(shí)間沒(méi)有聲音傳入(值也可更改)
                        if(downSum>20){
                            System.out.println("停止錄入");
                            break;
                        }

                    }
                }

                //取得錄音輸入流
                audioFormat = getAudioFormat();
                byte audioData[] = baos.toByteArray();
                bais = new ByteArrayInputStream(audioData);
                ais = new AudioInputStream(bais, audioFormat, audioData.length / audioFormat.getFrameSize());
                //定義最終保存的文件名
                System.out.println("開(kāi)始生成語(yǔ)音文件");
                AudioSystem.write(ais, AudioFileFormat.Type.WAVE, audioFile);
                downSum = 0;
                stopRecognize();

            } catch (Exception e) {
                e.printStackTrace();
            } finally {
                //關(guān)閉流

                try {
                    ais.close();
                    bais.close();
                    baos.reset();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }// end run
    }// end inner class CaptureThread

這個(gè)測(cè)試沒(méi)測(cè)試,偷個(gè)懶找的“哈哈”。

還有一點(diǎn)是Java操作語(yǔ)音文件.wav先不要研究了 :laugh and cry:,這里涉及到了語(yǔ)音識(shí)別,但是有百度那么些api,有興趣的試試吧!

好了,在這里就結(jié)束了

更新

博主把自己的畢設(shè)項(xiàng)目打包放到自己的服務(wù)器上,這個(gè)揚(yáng)聲器出現(xiàn)了新的問(wèn)題。

本來(lái)所有的基礎(chǔ)都是在本地運(yùn)行的,通過(guò)調(diào)用本地dll文件實(shí)現(xiàn)揚(yáng)聲器發(fā)聲,現(xiàn)在部署到centOS上將會(huì)失去這個(gè)dll的支持,目前所存在的問(wèn)題是如何不使用dll文件實(shí)現(xiàn)這個(gè)功能,中間借助了.wav后綴的音視頻文件。

如何在Linux上生成.wav的文件。
如何獲取這個(gè)文件并輸出。(解釋一下,用餐顧客點(diǎn)擊手機(jī)網(wǎng)頁(yè)的菜單,然后再餐廳的主機(jī)來(lái)播放這個(gè)聲音)
如何在輸出主機(jī)不進(jìn)行任何操作就能播放這個(gè)聲音或者能夠恢復(fù)之前的工作狀態(tài)。

現(xiàn)在的臨時(shí)解決辦法是本地跑一個(gè)呼叫服務(wù)的接口,當(dāng)需要這個(gè)功能的時(shí)候遠(yuǎn)程服務(wù)器調(diào)用本地跑的接口,進(jìn)而實(shí)現(xiàn)餐廳主機(jī)發(fā)聲。

這個(gè)和上面描述的并無(wú)差別,不一樣的是存在了兩臺(tái)主機(jī)的調(diào)用(當(dāng)然兩臺(tái)主機(jī)都應(yīng)該鏈接網(wǎng)絡(luò),能夠互相通信)

先寫(xiě)到這了,當(dāng)有解決辦法的時(shí)候再更新吧!

以上就是java 調(diào)用本地?fù)P聲器的步驟的詳細(xì)內(nèi)容,更多關(guān)于java 調(diào)用本地?fù)P聲器的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 淺析Java中comparator接口與Comparable接口的區(qū)別

    淺析Java中comparator接口與Comparable接口的區(qū)別

    本文要來(lái)詳細(xì)分析一下Java中Comparable和Comparator接口的區(qū)別,兩者都有比較的功能,那么究竟有什么區(qū)別呢,感興趣的Java開(kāi)發(fā)者繼續(xù)看下去吧
    2016-10-10
  • Java中的CopyOnWriteArrayList容器解析

    Java中的CopyOnWriteArrayList容器解析

    這篇文章主要介紹了Java中的CopyOnWriteArrayList容器解析,CopyOnWriteArrayList容器允許并發(fā)讀,讀操作是無(wú)鎖的,性能較高。至于寫(xiě)操作,比如向容器中添加一個(gè)元素,則首先將當(dāng)前容器復(fù)制一份,然后在新副本上執(zhí)行寫(xiě)操作,需要的朋友可以參考下
    2023-12-12
  • SpringCloud負(fù)載均衡實(shí)現(xiàn)定向路由詳情

    SpringCloud負(fù)載均衡實(shí)現(xiàn)定向路由詳情

    這篇文章主要介紹了SpringCloud負(fù)載均衡實(shí)現(xiàn)定向路由詳情,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-08-08
  • Java中接口的深入詳解

    Java中接口的深入詳解

    在Java語(yǔ)言中,接口由類(lèi)來(lái)實(shí)現(xiàn)以便使用接口中的方法,這篇文章主要給大家介紹了關(guān)于Java中接口的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2021-11-11
  • 教你怎么用Java通過(guò)關(guān)鍵字修改pdf

    教你怎么用Java通過(guò)關(guān)鍵字修改pdf

    此方法只適合通過(guò)關(guān)鍵字位置,在pdf上添加字符直接上代碼,代碼比較長(zhǎng),大部分自己的理解都在代碼注釋中了,需要的朋友可以參考下
    2021-05-05
  • Mybatis中resultMap的使用總結(jié)

    Mybatis中resultMap的使用總結(jié)

    resultmap是mybatis中最復(fù)雜的元素之一,它描述如何從結(jié)果集中加載對(duì)象,主要作用是定義映射規(guī)則、級(jí)聯(lián)的更新、定制類(lèi)型轉(zhuǎn)化器。今天通過(guò)本文給大家介紹Mybatis中resultMap的使用,感興趣的朋友參考下吧
    2021-06-06
  • log4j2的異步使用及添加自定義參數(shù)方式

    log4j2的異步使用及添加自定義參數(shù)方式

    這篇文章主要介紹了log4j2的異步使用及添加自定義參數(shù)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-12-12
  • Java8 中使用Stream 讓List 轉(zhuǎn) Map使用問(wèn)題小結(jié)

    Java8 中使用Stream 讓List 轉(zhuǎn) Map使用問(wèn)題小結(jié)

    這篇文章主要介紹了Java8 中使用Stream 讓List 轉(zhuǎn) Map使用總結(jié),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-06-06
  • 關(guān)于Java語(yǔ)法糖以及語(yǔ)法糖的原理和用法

    關(guān)于Java語(yǔ)法糖以及語(yǔ)法糖的原理和用法

    這篇文章主要介紹了關(guān)于Java什么是語(yǔ)法糖以及語(yǔ)法糖的種類(lèi),也稱(chēng)糖衣語(yǔ)法,是由英國(guó)計(jì)算機(jī)學(xué)家?Peter.J.Landin?發(fā)明的一個(gè)術(shù)語(yǔ),指在計(jì)算機(jī)語(yǔ)言中添加的某種語(yǔ)法,這種語(yǔ)法對(duì)語(yǔ)言的功能并沒(méi)有影響,但是更方便程序員使用,需要的朋友可以參考下
    2023-05-05
  • java排序高級(jí)之選擇排序?qū)崿F(xiàn)方法

    java排序高級(jí)之選擇排序?qū)崿F(xiàn)方法

    這篇文章主要介紹了java排序高級(jí)之選擇排序?qū)崿F(xiàn)方法,較為全面的分析了選擇排序的原理與具體實(shí)現(xiàn)技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下
    2015-02-02

最新評(píng)論