如果把Java媒體框架(http://java.sun.com/jmf)看作客戶端安裝的媒體播放機的話,那麼這種框架並不是非常激動人心。它只支持少量媒體類型,並且事實上這些媒體類型都被象Windows媒體播放機以及QuickTime等其它的應用程序支持。
但是從內容供應商的觀點來看,下面這個事實更加有意義:JMF在所有的Java類型中都可用,這使在客戶端上不必要有任何特定的媒體技術就可以部署媒體--所需要的只是J2S
E Java運行時間。
更重要的是它能利用.jar文件格式的一些性能把解碼器和媒體組成到一個文件中,創建一個"自我播放電影",使用了象WinZip和StuffIt這樣的壓縮應用程序大致相同的方式來創建自我展開文檔。
我們把此分為三個階段:
1.使JMF能夠播放.jar文件中的媒體。
2.只使用需要播放本地媒體所需的JMF的一部分創建一個輕量級.jar文件。
3.將代碼和媒體放入.jar文件,並且創建一個合適的manifest文檔以使它可以雙擊。
基本的JMF播放的關鍵是獲得Player,能解碼和渲染你的媒體。一般來說,這通過使用Manager獲得合適的DataSource來完成,在一個播放實例中提供了媒體流和流元數據,例如媒體格式。Manager然後找到一個Player來處理DataSource。在兩種情況下,Manager結合一個帶有程序包前綴列表(象javax.media、com.ibm.media等等)的反射方案來試圖尋找合適的類,例如如果一個計劃的播放程序不能接受提供給它的DataSource,那麼它就會拋出異常。
Manager沒有太多事要做,只是察看一下URL中的協議和文件擴展名,所以雖然它可以很容易的知道如何處理file:///Users/cadamson/mymp3stash/some.mp3,但是它不知道如何處理象jar:file:/Users/cadamson/dev/jmftests/spmovie-old/src/gatsbymovie.jar!/movie/themovie.mov這樣的URL。
為了緩和這種情況,我們可以寫一個DataSource,說得更准確一點就是一個PullDataSource,擔負為Manager描述細節的責任。JarEntryDataSource中沒有一個方法特別難;我們提供的PullSourceStream還需要實現幾個超級接口。
它有些不夠完美,但是這個類依靠文件擴展名來返回"內容類型"。這多多少少有點象一個MIME類型,除了使用句號代替斜線符號來格式化以外,所以它可被用於程序包名稱(例如video/mpeg MIME類型變成video.mpeg,所以Manager可以找到com.sun.media.codec.video.mpeg包)。下面是我們的簡單的實現:
public String getContentType() {
try {
URL url = getLocator().getURL();
String urlFile = url.getFile();
if (urlFile.endsWith(".mov"))
return "video.quicktime";
else if (urlFile.endsWith(".mpg"))
return "video.mpeg";
else if (urlFile.endsWith(".avi"))
// Manager needs ´_´ insted of ´-´
return "video.x_msvideo";
else
return "unknown";
} catch (MalformedURLException murle) {
return "unknown";
}
}
另一個煩惱是JMF源代碼表明如果提供的流是Seekable的話(這是一個提供隨機存取seek()方法的接口),默認Player只能播放一個QuickTime DataSource。如果尋找點在流的非常上流的地方,那麼JarEntryDataSource中的策略是使用InputStream.skip()。如果尋找點在當前讀取點之後(調用tellPoint,因為它的值通過Seekable.tell ()方法返回),然後它必須關閉InputStream,再重新打開,跳到尋找點。它使用一個內部的thoroughSkip()方法來確定我們實際上停在哪裡。
public long seek (long position) {
try {
if (position > tellPoint) {
thoroughSkip (position - tellPoint);
} else {
close();
open();
thoroughSkip (position);
}
return tellPoint;
} catch (IOException ioe) {
return 0; // bogus...
}
}
使用這個類,Manager可以找到一個可用的Player用於播放.jar文件中的.mov或者.avi文件。我們的示例的TinyPlayer使用ClassLoader.getResource ()方法來在類路徑中尋找movie/themovie.mov或者movie/themovie.avi。當類路徑只包含.jar文件的時候,我們將對其進行設置。