為Java程序中添加播放MIDI音樂功能

字號:

Java在多媒體處理方面的確優(yōu)勢不大,但是我們在程序中有些時(shí)候又需要一些音樂做為點(diǎn)綴,如果播放的音樂是wav等波形音頻文件,又挺大,所以背景音樂就是MIDI了,可是網(wǎng)上很多播放MIDI的教程都是簡單的幾句話的例子,并且沒有考慮資源的釋放問題,如果程序長久運(yùn)行的話,就會出現(xiàn)內(nèi)存越耗越多的情況,以至于最后拋出一個(gè)java.lang.OutOfMemoryError,整個(gè)程序就掛了。
    在MIDI的播放中,一個(gè)類是比較重要的,那就是MidiSystem類,它負(fù)責(zé)整個(gè)MIDI播放設(shè)備等的管理,其實(shí)就是Seqencer,它就是一個(gè)MIDI播放設(shè)置,用于播放MIDI序列的,還有一個(gè)類叫Seqence,它就是MIDI的序列,MIDI的序列可以自己由程序生成,也可以從文件中或者URL中讀取。
    下面我們來看一個(gè)例子吧:
    /*
     * Test5.java
     *
     * Created on 2007-9-22, 11:16:22
     *
     * To change this template, choose Tools | Templates
     * and open the template in the editor.
     */
    package test1;
    import java.io.File;
    import java.io.IOException;
    import java.io.InputStream;
    import java.util.Hashtable;
    import java.util.Map;
    import java.util.logging.Level;
    import java.util.logging.Logger;
    import javax.sound.midi.InvalidMidiDataException;
    import javax.sound.midi.MidiSystem;
    import javax.sound.midi.MidiUnavailableException;
    import javax.sound.midi.Sequence;
    import javax.sound.midi.Sequencer;
    /**
     *
     * @author hadeslee
     */
    public class Test5 implements Runnable{
     private Sequencer midi;
     private String[] names={"1.mid","2.mid","3.mid","4.mid","5.mid"};
     private int i;
     private Map map;
     public Test5(){
     initMap();
     new Thread(this).start();
     }
     private void initMap(){
     try {
     map = new Hashtable();
     midi = MidiSystem.getSequencer(false);
     midi.open();
     for (String s : names) {
     try {
     Sequence s1 = MidiSystem.getSequence(new File(s));
     map.put(s, s1);
     } catch (InvalidMidiDataException ex) {
     Logger.getLogger(Test5.class.getName()).log(Level.SEVERE, null, ex);
     } catch (IOException ex) {
     Logger.getLogger(Test5.class.getName()).log(Level.SEVERE, null, ex);
     }
     }
     } catch (MidiUnavailableException ex) {
     Logger.getLogger(Test5.class.getName()).log(Level.SEVERE, null, ex);
     }
     }
     private void createPlayer(String name){
     try {
     Sequence se=map.get(name);
     midi.setSequence(se);
     midi.start();
     }catch (InvalidMidiDataException ex) {
     Logger.getLogger(Test5.class.getName()).log(Level.SEVERE, null, ex);
     }
     }
     public void run(){
     while(true){
     try {
     System.out.println("換文件了."+(++i));
     String name=names[(int)(Math.random()*names.length)];
     createPlayer(name);
     Thread.sleep(10000);
     } catch (InterruptedException ex) {
     Logger.getLogger(Test5.class.getName()).log(Level.SEVERE, null, ex);
     }
     }
     }
     public static void main(String[] args) {
     new Test5();
     }
    }
    在這里有很重要的一點(diǎn),那就是在程序運(yùn)行的時(shí)候,只要一個(gè)Seqencer就可以了,我以前在程序里面每次播放的時(shí)候都生成了一個(gè)Seqencer,因?yàn)槟莻€(gè)時(shí)候我想,我都調(diào)用它的close()方法了,它還能被打開嗎?其實(shí)它還可以再度被打開的,就是這樣一種慣性思維使得程序最終因內(nèi)存溢出而崩潰。
    現(xiàn)在按我這種方式播,哪怕10毫秒換一次MIDI都可以,換個(gè)幾萬次內(nèi)存一點(diǎn)都沒有加,呵呵,真是防不勝防啊。