Java媒体框架(JMF)使你能够编写出功能强大的多媒体程式,却不用关心底层复杂的实现细节。JMF API的使用相对比较简单,不过能够满足几乎所有多媒体编程的需求。在这篇文章中,我将向你介绍怎么用非常少的代码就编写出多媒体程式。
Java多媒体框架(JMF)中包含了许多用于处理多媒体的API。他是个相当复杂的系统,完全了解这个系统可能需要花上几周的时间,不过这篇文章将主要介绍JMF的几个核心接口和类,然后通过一个简单的例子向你展示怎么利用该接口进行编程。
JMF目前的最新版本是2.1,Sun通过他向Java中引入处理多媒体的能力。下面是JMF所支持的功能的一个概述:
● 能在Java Applet和应用程式中播放各种媒体文件,例如AU、AVI、MIDI、MPEG、QuickTime和WAV等文件。
● 能播放从互连网上下载的媒体流。
● 能利用麦克风和摄像机一类的设备截取音频和视频,并保存成多媒体文件。
● 处理多媒体文件,转换文件格式。
● 向互连网上传音频和视频数据流。
● 在互连网上广播音频和视频数据。
JMF的结构
为了更好地说明JMF的结构,让我们用立体声音响做一个简单的比喻。当你CD机播放CD唱片的时候,CD唱片向系统提供音乐信号。这些数据是在录音棚中用麦克风和其他类似的设备记录下来的。CD播放机将音乐信号传送到系统的音箱上。在这个例子中,麦克风就是个音频截取设备,CD唱片是数据源,而音箱是输出设备。
JMF的结构和立体声音响系统非常相似,在后面的文章中,你会遇见下面的这些术语:
● 数据源(Data source)
● 截取设备(Capture Device,包括视频和音频截取设备)
● 播放器(Player)
● 处理器(Processor)
● 数据格式(Format)
● 管理器(Manager)
下面让我们来看一看这些术语到底代表什么意思。
1.数据源
就像CD中保存了歌曲相同,数据源中包含了媒体数据流。在JMF中,DataSource对象就是数据源,他能是个多媒体文件,也能是从互连网上下载的数据流。对于DataSource对象,一旦你确定了他的位置和类型,对象中就包含了多媒体的位置信息和能够播放该多媒体的软件信息。当创建了DataSource对象后,能将他送入Player对象中,而Player对象不必关心DataSource中的多媒体是怎么获得的,及格式是什么。
在某些情况下,你需要将多个数据源合并成一个数据源。例如当你在制作一段录像时,你需要将音频数据源和视频数据源合并在一起。JMF支持数据源合并,在后面的例子中我们将提到这一点。
2.截取设备
截取设备指的是能截取到音频或视频数据的硬件,如麦克风、摄像机等。截取到的数据能被送入Player对象中进行处理。
3.播放器
在JMF中对应播放器的接口是Player。Player对象将音频/视频数据流作为输入,然后将数据流输出到音箱或屏幕上,就像CD播放机读取CD唱片中的歌曲,然后将信号送到音箱上相同。Player对象有多种状态,JMF中定义了JMF的六种状态,在正常情况下Player对象需要经历每个状态,然后才能播放多媒体。下面是对这些状态的说明。
● Unrealized:在这种状态下,Player对象已被实例化,不过并不知道他需要播放的多媒体的所有信息。
● Realizing:当调用realize()方法时,Player对象的状态从Unrealized转变为Realizing。在这种状态下,Player对象正在确定他需要占用哪些资源。
● Realized:在这种状态下Player对象已确定了他需要哪些资源,并且也知道需要播放的多媒体的类型。
● Prefetching:当调用prefectch()方法时,Player对象的状态从Realized变为Prefetching。在该状态下的Player对象正在为播放多媒体做一些准备工作,其中包括加载多媒体数据,获得需要独占的资源等。这个过程被称为预取(Prefetch)。
● Prefetched:当Player对象完成了预取操作后就到达了该状态。
● Started:当调用start()方法后,Player对象就进入了该状态并播放多媒体。
4.处理器
处理器对应的接口是Processor,他一种播放器。在JMF API中,Processor接口继承了Player接口。 Processor对象除了支持支持Player对象支持的所有功能,还能控制对于输入的多媒体数据流进行何种处理及通过数据源向其他的Player对象或Processor对象输出数据。
除了在播放器中提到了六种状态外,Processor 对象还包括两种新的状态,这两种状态是在Unrealized状态之后,不过在Realizing状态之前。
● Configuring:当调用configure()方法后,Processor对象进入该状态。在该状态下,Processor对象连接到数据源并获取输入数据的格式信息。
● Configured:当完成数据源连接,获得输入数据格式的信息后,Processor对象就处于Configured状态。
5.数据格式
Format对象中保存了多媒体的格式信息。该对象中本身没有记录多媒体编码的相关信息,不过他保存了编码的名称。Format的子类包括AudioFormat和VideoFormat类,ViedeoFomat又有六个子类:H261Format、H263Format、IndexedColorFormat、JPEGFormat、RGBFormat和YUVFormat类。
6.管理器
JMF提供了下面四种管理器:
● Manager:Manager相当于两个类之间的接口。例如当你需要播放一个DataSource对象,你能通过使用Manager对象创建一个Player对象来播放他。使用Manager对象能创建Player、Processor、DataSource和DataSink对象。
● PackageManager:该管理器中保存了JMF类注册信息。
● CaptureDeviceManager:该管理器中保存了截取设备的注册信息。
● PlugInManager:该管理器中保存了JMF插件的注册信息。
创建一个Player对象
在JMF编程中,最常见的工作就是创建一个Player对象。你能通过Manager类的createPlayer()方法创建Player对象。Manager对象使用多媒体的URL或MediaLocator对象来创建Player对象。当你获得了一个Player对象后,你能通过调用getVisualComponent()方法得到Player对象的图像部件(Visual Component,在图像部件上能播放多媒体的图像)。然后将图像部件加入到应用程式或Applet的界面上。Player对象还包含一个控制面板,在上面能控制媒体的播放、停止和暂停等。
Player类中的非常多方法只有在Player对象处于Realized的状态下才会被调用。为了确保Player对象已到达了该状态,你需要使用Manager的createRealizePlayer()方法来获得Player对象。不过对于start()方法来说,你能在Player对象到达Prefetched状态之前调用他,他能自动将Player的状态转换到Started状态。
截取多媒体数据
多媒体数据的截取是JMF程式中另一个非常重要的功能。你能按照下面的步骤截取数据:
● 通过查询CaptureDevieceManager获得你希望使用的截取设备。
● 获得设备对应的CaptureDeviceInfo对象。
● 从CaptureDeviecInfo对象中获得MediaLocator对象,然后用他创建一个DataSource对象。
● 使用DataSource对象创建Player对象或Processor对象。
● 调用start()方法,开始截取多媒体数据。
你能使用CaptureDeviceManager对象获得系统中可用的视频和音频截取设备。通过调用getDeviceList()方法你能获得设备的列表。每个设备都对应一个CaptrueDeviceInfo对象。也能通过调用CaptureDevieceManager对象的getDevice()方法来获得特定的CaptureDeviceInfo对象。在使用设备截取多媒体数据前,还需要从CaptureDeviceInfo对象中获得设备对应的MediaLocator对象。然后你能直接使用MediaLocator来构造Player或Processor的实例,也能用MediaLocator构造一个DataSource对象,然后将DataSource对象送入Player或Processor对象中。最后调用start()方法来截取多媒体数据。
一个JMF例子
当你使用JMF进行编程以前,你需要安装JMF。同时在硬件上也有一些需求。由于本文的代码是在视窗系统 2000下编写和测试,因此文章中提到的操作系统需要的软件都是和视窗系统有关的。虽然Java是跨平台的,不过JMF是个例外――并不是所有的平台上都实现了JMF。
硬件和软件需求
硬件方面你需要和SoundBlaster兼容的声卡,芯片最佳使用奔腾III以上的芯片。内存最佳不小于64MB。同时你需要安装下面的软件:
● 视窗系统95/98,视窗系统 NT 4.0, 视窗系统2000或 视窗系统XP。
● JDK1.1.6或以上的视窗系统版本。
● JMF类和动态库
在视窗系统下安装JMF2.1
当下载了JMF2.1以后,运行jmf-2_1_1b-windows-i586.exe。该程式会将JMF2.1安装到你指定的目录下。当安装成功后,你需要确认一下安装程式正确设定了CLASSPATH和PATH环境变量。在CLASSPATH中需要包含jmf.jar和sound.jar;在PATH中需要包含JMF动态库的路径。
JMFRegistry
如果你希望使用视频和音频截取的设备,你需要确认安装了这些设备的驱动程式。除此之外,你还需要运行JMFRegistry应用程式。JMFRegistry能向JMF注册新的数据源、媒体处理器、插件、视频和音频截取设备,然后你才能够在你的程式中使用他们。你只需要运行一次JMFRegistry就能注册系统中所有的视频和音频截取设备。
当你运行了JMFRegistry后,会弹出图一所示的窗口:
图一 通过JMFRegistry注册视频和音频截取设备
选择“Capture Devices”标签,然后按下“Detect Capture Devices”按钮,程式将自动检测出系统中的视频和音频截取设备。在左边的类表框中会列出所有检测到的设备的名称。在图一中我们看到JMFRegsitery发现了JavaSound audio capture、vfw:Logitech USB Video Camera:0和vfw:Microsoft WDM Image Capture (Win32):1。单击某个设备能看到该设备支持的视频或音频格式。如果JMFRegistry无法检测到设备,有可能是没有正常安装设备的驱动程式。
例子程式
由于JMF2.1比较复杂,我不可能在在例子中包含JMF2.1支持的所有功能。因此我选择了下面几个在JMF中比较常用的功能:播放多媒体、注册音频和视频截取设备、截取视频和音频。
1.播放多媒体
在JMF.java中有一个play()方法。该方法能播放用户选择的多媒体文件。当播放多媒体文件时,你需要一个Player对象。在例子中,dualPlayer就是Player接口的实现对象。
Player dualPlayer;
在Play()方法中,通过使用FileDialog获得媒体文件的路径和文件名,并保存在filename中。
try { FileDialog fd = new FileDialog(this, "Select File", FileDialog.LOAD); fd.show(); String filename = fd.getDirectory() + fd.getFile(); ... } catch (Exception e) { System.out.println(e.toString()); }
然后你需要通过媒体管理器Manager间接创建一个Player对象。你能使用Manager类的createPlayer()方法或createProcessor()方法来获得一个Player对象或Processor对象。在play()方法中,我使用的是createPlayer()方法。
dualPlayer = Manager.createPlayer (new MediaLocator("file:///" + filename));
有时你需要使用一个Player对象来控制多个其他的Player和Controller对象,我们把这个Player对象称为主对象,并把这些对象组成一个组。通过调用主对象中的start()、stop()、setMediaTime()等方法就能激活组中所有成员的相应方法。主对象控制所有的状态变化和事件发布。然后使用addControllerListerner()方法来将一个ControllerListener对象绑定到Player对象上,Controller对象将向该ControllerListener对象发送事件消息。
dualPlayer.addControllerListener(this);
最后需要调用start()方法来启动Player对象。start()方法将Player对象的状态设置为Started。如果Player没有被实体化(Realize)或预取(Prefetch),start()方法会自动执行这些操作。
dualPlayer.start();
由于JMF类实现了ControllerLister接口,因此需要实现该接口中的controllerUpdate()方法,该方法在Controller对象产生一个事件时被调用。
public synchronized void controllerUpdate(ControllerEvent event) { if (event instanceof RealizeCompleteEvent) { Component comp; if ((comp = dualPlayer.getVisualComponent()) != null) add ("Center", comp); if ((comp = dualPlayer.getControlPanelComponent()) != null) add("South", comp); validate(); } }
当JMF类产生了一个RealizeCompleteEvent事件后,controllerUpdate()方法在界面上增加两个Component对象,一个用于播放媒体,一个用于放置控制按钮,例如播放、停止等。
在运行程式的过程中,程式会产生下面的输出。
Starting player ...javax.media.TransitionEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78, previous=Unrealized, current=Realizing, target=Started] Open log file: C:\test\Java\JMF\JMF\jmf.log javax.media.DurationUpdateEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78,duration= javax.media.Time@2a37a6 javax.media.RealizeCompleteEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78, previous=Realizing, current=Realized, target=Started] Adding visual component Adding control panel javax.media.TransitionEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78, previous=Realized, current=Prefetching, target=Started] javax.media.PrefetchCompleteEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78, previous=Prefetching, current=Prefetched,target=Started] javax.media.StartEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78, previous=Prefetched, current=Started, target=Started, mediaTime=javax.media.Time@56a05e,timeBaseTime= javax.media.Time@3a8602] javax.media.EndOfMediaEvent [source=com.sun.media.content.video.mpeg.Handler@71bb78, previous=Started, current=Prefetched, target=Prefetched, mediaTime=javax.media.Time@1d332b]
前面提到,当调用start()方法的时候,Player会转换到Started状态。从上面列出的信息中能看到Player对象的状态从Unrealized变成了Started。当EndOfMedia事件被激活时(这时Player对象完成了媒体文件的播放),状态从Started变成了Prefetched。图二显示了程式正在播放多媒体文件时的情况。
图二 程式正在播放媒体文件
2.注册音频和视频截取设备
在例子中,注册音频和视频截取设备的方法只在程式的内部注册这些设备,在程式外则不起作用。该方法的作用是当用户的计算机上存在多和音频和视频截取设备时,告诉程式因该使用哪个设备和这些设备支持的音频和视频格式。因此在进行截取处理之前需要获得设备的设置信息。在例子中,当在Configure菜单上按下Capture Device命令后,会弹出CaptureDeviceDialog对话框。如果在截取音频或视频前没有设定设备的设置,也会弹出该对话框。图三显示了该对话框。
图三 设备注册对话框
让我们来看一下CaptureDeviceDialog类中的init()方法:在初始化了界面之后,通过调用CaptureDeviceManager类的getDeviceList()方法:
devices = CaptureDeviceManager.getDeviceList ( null );
CaptureDeviceManager类使用查询机制和一个注册表来定位设备,然后将设备的信息放入CaptureDeviceInfo对象中返回。我们还能利用CaptureDeviceManager类来注册新的设备。通过调用getDeviceList()方法程式获取了一个支持指定格式的设备的列表。在例子中,我将格式参数设定为null,这意味着设备能使用所有格式。返回值被放入device变量中。如果getDeviceList()方法返回的是个非空值,程式会将包含在其中的音频设备名称和视频设备名称分别放入两个下拉列表中中,不过到目前为止我们还不知道哪些设备是音频设备,哪些是视频设备。
我们能通过CaptureDeviceInfo的getFormat()方法获得Format对象组数,在Format对象中保存了设备支持的媒体格式。Format类间接被AudioFormat和VideoFormat类所继承。因此我们能利用设备支持的格式类型来区分设备的类型:
if (devices!=null &#38;&#38; devices.size()>0) { int deviceCount = devices.size(); audioDevices = new Vector(); videoDevices = new Vector(); Format[] formats; for ( int i = 0; i < deviceCount; i++ ) { cdi = (CaptureDeviceInfo) devices.elementAt ( i ); formats = cdi.getFormats(); for ( int j=0; j<formats.length; j++ ) { if ( formats[j] instanceof AudioFormat ) { audioDevices.addElement(cdi); break; } else if (formats[j] instanceof VideoFormat ) { videoDevices.addElement(cdi); break; } } } . . . }
上面的程式运行后,audioDevices()中将包含所有的音频设备,videoDevices()中将保存所有的视频设备。其中cdi是CaptureDeviceInfo对象。然后将设备名称填入下拉列表中:
// 将音频设备显示在下拉列表中 for (int i=0; i<audioDevices.size(); i++) { cdi = (CaptureDeviceInfo) audioDevices.elementAt(i); audioDeviceCombo.addItem(cdi.getName()); } // 将视频设备显示在下拉列表中 for (int i=0; i<videoDevices.size(); i++) { cdi = (CaptureDeviceInfo) videoDevices.elementAt(i); videoDeviceCombo.addItem(cdi.getName()); }
然后程式显示出当前选中的设备支持的格式:
displayAudioFormats(); displayVideoFormats();
下一步需要获取用户选中的音频设备和视频设备及他们支持的格式,相关的方法是JMF类中的getAudioDevice()、getVideoDevice()、getAudioFormat()和getVideoFormat()方法。然后将获取的对象分别保存到audioCDI,videoCDI,audioFormat和videoFormat中:
audioCDI = cdDialog.getAudioDevice(); if (audioCDI!=null) { audioDeviceName = audioCDI.getName(); System.out.println("Audio Device Name: " + audioDeviceName); } videoCDI = cdDialog.getVideoDevice(); if (videoCDI!=null) { videoDeviceName = videoCDI.getName(); System.out.println("Video Device Name: " + videoDeviceName); } // 获得选中的多媒体格式 videoFormat = cdDialog.getVideoFormat(); audioFormat = cdDialog.getAudioFormat();
3.截取视频和音频
使用capture()方法能截取音频和视频数据。不过在使用该方法前需要确定是否已选中了视频和音频截取设备:
if (audioCDI==null &#38;&#38; videoCDI==null) registerDevices();
和play()方法类似,能通过使用Manger类中的静态方法createPlayer()创建一个Player对象,该对象能播放一个DataSource对象中的数据流。
Player createPlayer(MediaLocator sourceLocator)
在例子中,我首先通过调用audioCDI和videoCDI的getLocator()方法来获得MediaLocator对象,然后利用Manager类的createPlayer()方法创建Player对象。最后将一个ControllerListener对象绑定到视频Player对象上并开始播放。
videoPlayer = Manager.createPlayer(videoCDI.getLocator()); audioPlayer = Manager.createPlayer(audioCDI.getLocator()); videoPlayer.addControllerListener(this); videoPlayer.start(); audioPlayer.start();
使用这种方法导致最后获得了两个Player对象。我们也能使用Manager类中的createDataSource()方法从视频和音频CaptureDeviceInfo对象(audioCID和videoCDI)中获得视频和音频数据源(DataSource对象),然后调用createMergingDataSource()方法将两个数据源合并成一个数据源(ds):
DataSource[] dataSources = new DataSource[2]; dataSources[0] = Manager.createDataSource(audioCDI.getLocator()); dataSources[1] = Manager.createDataSource(videoCDI.getLocator()); DataSource ds = Manager.createMergingDataSource(dataSources);
然后能使用ds作为createPlayer()方法的参数来获得一个Player对象dualPlayer。调用addControllerListener()就能进行播放了。
dualPlayer = Manager.createPlayer(ds); dualPlayer.addControllerListener(this); dualPlayer.start();
小结
Java多媒体框架是个非常好的多媒体编程工具。在这篇文章中我只是简单介绍了JMF的一些基本功能。如果有兴趣的话能仔细阅读一下Sun公司的Java网站上提供的JMStudio的例子。在JMStudio中不仅实现了简单的播放和视频/音频截取功能,还实现了从互连网下载和向互连网上传多媒体数据流的功能。而且他还包含了JMFRegistry的原始码,将相应的代码移植到你的应用程式中后,你就不必在运行程式前运行JMFRegistry来向JMF注册设备了。
作者简介:冯睿,2000年毕业于美国Northern Illinois大学电气工程系,获硕士学位。随后在New Monics软件公司工作了一年,其间参加了Java虚拟机的研发和优化工作。目前在国内一家GIS公司担任项目经理,主要从事应急指挥系统的交通GIS系统的研发。
分享到:
相关推荐
总的来说,JMF为Java开发者提供了一套强大的多媒体编程工具,通过理解和掌握其核心概念和API,可以轻松创建各种多媒体应用程序,满足不同需求。然而,由于JMF的复杂性,开发者需要投入一定的时间学习和实践,以便...
标题“jmf多媒体播放.zip”和描述中的信息表明,这个压缩包包含与Java Media Framework (JMF) 相关的资源,用于多媒体播放。JMF是Java平台上的一个开源框架,它允许开发者处理和播放音频、视频等多种媒体格式。下面...
### Java中利用JMF的多媒体编程 #### 摘要 Java作为一种广泛应用于互联网环境下的编程语言,在多媒体处理方面有着独特的优势。Java Media Framework(JMF)为开发者提供了一个强大的平台,使得开发者能够轻松地创建...
在“java利用JMF框架实现视频传输的完整代码”这个资源中,我们主要会探讨以下几个核心知识点: 1. **Java Media Framework (JMF)** JMF 是Sun Microsystems为Java开发的多媒体框架,它提供了处理音频、视频和实时...
你可以学习如何利用JMF实现P2P中的媒体数据交换和同步。 **学习资源** 在"www.pudn.com.txt"这个文件中,可能包含了更多关于JMF的资源链接,如相关的论坛讨论、其他教程或者示例代码,这些都可以作为深入学习的...
尽管JMF有一定的学习曲线,但它仍然是理解Java多媒体编程的一个重要切入点。开发者需要熟悉JMF的API,理解如何查找、控制和处理来自摄像头的数据,才能成功实现这样的程序。同时,也要注意随着技术的发展,选择更...
包含文件:1.Java Media Framework 基础教程.doc;2.java的媒体架构(JMF).docx;3.jmf基础.docx;4.Java中利用JMF的多媒体编程.pdf;5.jmf-2_1_1e-windows-i586.exe;6.JMFAPI_CN.html(不错的)
JMF API的使用相对比较简单,但是能够满足几乎所有多媒体编程的需求。在这篇文章中,我将向你介绍如何用很少的代码就编写出多媒体程序。 Java多媒体框架(JMF)中包含了许多用于处理多媒体的API。它是一个相当复杂...
开发者可以参考此源码来学习如何利用JMF进行多媒体处理,也可以在此基础上进行扩展和优化。 【描述】JMF多媒体播放器源码实现了多种功能,包括但不限于: 1. **文件格式支持**:JMF能够处理多种多媒体文件格式,如...
### Java中利用JMF编写摄像头拍照程序实例 #### 背景与介绍 在现代软件开发中,多媒体处理是一项非常重要的技术。Java Media Framework (JMF) 是Sun Microsystems推出的一个强大的多媒体开发框架,它允许开发者在...
”意味着开发者只需将 JMF 的 `.jar` 文件添加到项目的类路径(classpath)中,就能利用 JMF 的功能进行多媒体开发。这里的 "lib" 通常是指库文件夹,开发者将 JMF 的 `.jar` 放入此文件夹或指定其路径,以便编译器...
采用JAVA 编程语言,用eclipse开发平台实现。用到多媒体技术(JAVA 多媒体框架JMF),视频和音频用TRP协议传输,用到Socket编程技术...利用JAVA的这个组件,我们可以方便的处理多媒体文件,并基于RTP协议进行实时传输。
通过学习这些资料,开发者可以掌握如何利用JMF创建复杂的多媒体应用程序,如在线视频会议、媒体播放器、监控系统等。理解JMF的工作原理和API,以及如何利用其强大的媒体处理能力,将对Java平台上的多媒体编程带来极...
Java Media Framework(JMF)是Java平台上用于处理多媒体数据的一个框架,它允许开发人员创建、播放和捕获音频和视频内容。...通过深入学习和实践,开发者可以克服初始的困难,充分利用JMF的特性来开发多媒体应用程序。
JMF  API的使用相对比较简单,但是能够满足几乎所有多媒体编程的需求。在这篇文章中,我将向你介绍如何用很少的代码就编写出多媒体程序。  Java多媒体框架(JMF)中包含了许多用于处理多媒体的API。它是一个...
开发者只需要将.jar文件导入到他们的项目类路径中,就可以利用JMF提供的API来实现各种多媒体功能。这包括播放音频和视频文件,支持不同的编码格式,以及进行媒体的捕获和处理。 JMF的核心组件包括: 1. **Media ...
#### 利用JMF进行多媒体编程 ##### 创建Player对象 在实际开发中,创建`Player`对象是最常见的任务之一。这通常通过`Manager`类的`createPlayer()`方法完成。该方法接受多媒体的URL或`MediaLocator`对象作为参数。...
在这个“java jmf 例子”中,我们将探讨如何利用JMF进行多媒体编程。 首先,JMF的安装至关重要。在开始任何编程之前,你需要从官方或可靠的第三方源下载JMF的安装包,并按照指示完成安装过程。安装后,JMF的库文件...
在工程中,开发者可以利用JMF API读取、播放和控制视频流。源码中可能包含了示例,演示如何加载视频文件,创建播放器,以及控制播放速度、音量等参数。 **5. 开发流程** - **创建JMF Player**: 使用`Manager....