热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

FMS入门教程

原文地址:http:ld-21.iteye.comblog802816FMS的最新版本已经到FMS3了,本入门教程使用FlashMediaServer2演示,但你可以很容易的升到3

FMS的最新版本已经到FMS 3了 ,本入门教程使用Flash Media Server 2 演示,但你可以很容易的升到 3,几乎没有什么区别
我把我以前的那些FMS学习笔记整理了一下,做了下修改,成了一页的FMS教程,方便想学FMS的新手来看,如果还有错误或者看不懂的地方欢迎留言,或email偶(nshen121gmail.com),偶会进行更改
什么是Flash Media Server ?
Flash大家庭里的一员,这个东东以前叫Flash Communication Server,传说中的FCS就是这个,现在改叫FMS了...
不见不知道哦,一见真可怕,adobe还有这么cool的东东,自从见了她,让我做些小东东的兴趣全没了,一心想研究这个
这东东能做什么?
视频录制啊,flash聊天室啊,在线视频会议啊啊, 网络游戏?
不管怎样先安个试试吧
先到这里下载免费的开发者版本吧
http://www.macromedia.com/software/flashmediaserver/
然后安装,安装时记得用户名和密码不要瞎填自己要记住,如果你已经瞎填了那就到 安装目录下\conf\fms.ini 里找吧 
偶装到了c盘,找到目录
C:\Program Files\Macromedia\Flash Media Server 2 
 
先要打开服务器哦 
你在本机装的fms,本机就是服务器了哦!记得现在你的机器既是客户端又是服务端哦
开始===程序===Macromedia===Flash Media Server 2
有两个start****,都要打开,关时候两个都要关,如果你怕麻烦直接找tool文件夹里的批处理文件StartServerService.bat吧,双击他,他会帮你搞定地,想关就找StopServerService.bat。~ 
如果想设置默认开机时候fms自动打开还是需要手动打开,可以在 开始菜单里--运行--输入services.msc ,就会打开服务,找到Flash Media Server项,设置为手动或者自动,这里也可以随时关闭服务或打开服务
applications文件夹
你可以在里边建一个文件夹例如叫 FirstApp,这就建了一个Application,以后所有此项目服务器端的flv文件啊,共享文件啊都会在这里边,有时服务器端需要写程序的时候里边会有个main.asc 文件,这个就是服务器端程序,服务器端程序目前只支持as1写,是在服务器上执行的,你也可以用trace调试,怎么trace管理服务器时候你会看到,你可能会建若干个app,在一台服务器上同时运行,比如录象的app,聊天室的app。。。这些所有的app都通过fms自带的fms2_console.swf管理,现在就去看看如何管理服务器吧!
管理服务器~
服务器打开后就要管理了哦,点开fms2_console.swf,别看他只是个swf,后台管理就靠他了,输入你安装时候的密码和用户,服务器地址写 localhost 就好看到了吧,熟悉熟悉他吧
 
以后我们写程序最常用的就是这个View Applications ,每当有客户端连接服务器时,左边就会列出连接的是哪个app,有多少连接 下图为连接applications文件夹下的bs文件夹(bs文件夹,因为每个文件夹对应一个app,就是bs app),连接数为1,458是实例名(实例名默认为 _definst_ ,每个文件夹可以有若干实例,互相不影响,这个特性可以用来做聊天是的房间,以后再说) 
 
选中某个app后,或者客户端有连接,会看到当前打开的app的状态,这个
 
Live Log 服务器端的trace就这里看了 ,右边依次是客户端情况 , 共享对象,流,执行的情况(占内存,cpu等),后边两个小按钮,调试时候常用哦,reload和unload! 
每当服务器端main.asc修改后一定记得reload或者unload一下,否则不会生效,unload会把窗口关闭,有客户端连的时候还会自动打开 
总有人告诉我,他的fms经常会出现诡异现象,比如连接不上,代码已经删了还会执行,一生气连文件夹都删了,还会执行,怀疑是自己的rp有问题。。 
那不是rp问题,记住出现问题就reload !实在不行就去tools文件夹点StopServerService.bat
conf文件夹
还有重要的是conf文件夹了,里边是一些服务器端的配置文件,以后可能会用,先不用动。。。。 
自此,安装部分结束,go on... 
有一种协议叫rtmp
客户端和服务器端通信是使用协议rtmp的
现在在服务器端applications文件夹(当然偶的客户端和服务器端是一台机器了)里建个test1文件夹,你的地址就为
rtmp:/test1 或者 rtmp://localhost/test1
注意两个地址中的 "/ "符号
打开flash
与服务器通信首先要建个NetConnection() 
nc = new NetConnection(); 
连接 
nc.connect("rtmp://localhost/test1"); 
怎么知道连没连上呢?顺利连接服务器后会触发一个onStatus事件,自己trace一下info.code

Java代码  技术分享
  1. nc.onStatus = function(info) {  
  2. //trace(info)  
  3. //trace(info.code)  
  4. for (i in info) {  
  5. trace(i+": "+info[i]);  
  6. }  
  7. };   

完整代码:

Java代码  技术分享
  1. nc = new NetConnection();  
  2. nc.onStatus = function(info) {  
  3. trace(info.code);  
  4. if (info.code == "NetConnection.Connect.Success") {  
  5. trace("接通");  
  6. }  
  7. };  
  8. nc.connect("rtmp://localhost/test1");   

注意:默认情况下服务器是允许你连接的,但只是默认,如果服务器拒绝你连接的话,上述代码就不好用了。怎么回事?我们看看连接的过程吧
连接过程
每当客户端试图连接服务器,一个NetConnection.connect(),服务器将会调用application.onConnect 来鉴定是不是允许客户端连接,onConnect()方法返回null 或不返回则将进入未决状态,直到onConnect方法中返回true或执行acceptConnection(client)则允许,返回false或执行rejectConnection(client)则拒绝,如图(从左往右看)
 
服务器文件是以.asc形式存在的,可以在test1文件夹里建一个main.asc

Java代码  技术分享
  1. application.OnConnect=function(client){  
  2. this.rejectConnection(client);  
  3. //this.acceptConnection(client)  
  4. }   

 这样就拒绝连接了,动手试试,别忘了,服务器端改动的话,别忘了到fms2_console.swf去reload !不知道按哪个的到上边找,每当有客户端连接,那个reload按钮的界面就会出来哦。
检查是否uri错误
如果你的rtmp地址是从其他什么地方传过来的,可以顺便检查一下rtmp是否错误,下边代码如果地址是错误的就会trace出来

Java代码  技术分享
  1. mync = new NetConnection();  
  2. mync.onStatus = function(info) {  
  3. if (info.code == "NetConnection.Connect.Success") {  
  4. trace("连接成功");  
  5. }  
  6. };  
  7. //正确的uri  
  8. //uri = "rtmp://localhost/connect";  
  9. //错误的uri  
  10. uri = "rtmpppppp://localhost/connect";  
  11. if (mync.connect(uri, "N神")) {  
  12. trace("尝试连接服务器中。。");  
  13. else {  
  14. trace("没有尝试连接服务器~是uri错误???");  
  15. }  

ok 了,现在我们深入一点点。。看看连接上的一些细节问题
info.code:
连接后info.code会告诉你连接的状态,上边看的都是NetConnection.Connect.Success, 还有一些其他值,和这些值是什么意思,自己看看。
值得注意的是。NetConnection.Connect.Rejected,收到这条消息的时候说明服务器端拒绝了你,接着马上你会收到另一条,NetConnection.Connect.Closed,连接就关闭了~~~

Java代码  技术分享
  1. mync = new NetConnection();  
  2. mync.onStatus = function(info) {  
  3. switch (info.code) {  
  4. case "NetConnection.Connect.Success" :  
  5. trace("连接成功");  
  6. break;  
  7. case "NetConnection.Connect.Failed" :  
  8. //关掉服务器的情况  
  9. trace("尝试连接失败,服务器有可能挂掉了 -_-b");  
  10. break;  
  11. case "NetConnection.Connect.Rejected" :  
  12. //注意这里,服务器拒绝你的情况,如果遭到拒绝,将会调用两次mync.onStatus,  
  13. //第一次"NetConnection.Connect.Rejected"第2次"NetConnection.Connect.Closed"  
  14. trace("遭到服务器拒绝");  
  15. trace("服务器返回信息:"+info.application.msg);  
  16. break;  
  17. case "NetConnection.Connect.Closed" :  
  18. trace("连接关闭");  
  19. break;  
  20. }  
  21. };  
  22. mync.connect("rtmp://localhost/connect", "N神");  
  23. //mync.connect("rtmp://localhost/connect","小新")   

服务器端拒绝连接?好象见过。。。回头找找。。。。。。。。哦在这里
application.OnConnect= function(client) {
this.rejectConnection(client);
}
我不能所有人都拒绝了。。我要把讨厌的人拒绝了。。。
传给服务器一个人名~~
mync.connect("rtmp://localhost/connect", "N神");
服务器看看是不是讨厌的人。。

Java代码  技术分享
  1. application.onConnect = function(client, name) {  
  2. trace(name);  
  3. if (name == "N神") {  
  4. //拒绝连接,并返回个错误对象{msg:"服务器不想"+name+"进去,哈哈~"},包含错误消息  
  5. application.rejectConnection(client, {msg:"服务器不想"+name+"进去,哈哈~"});  
  6. else {  
  7. application.acceptConnection(client);  
  8. //成功不能返回客户端信息  
  9. }  
  10. };   

 看客户端的代码。。

Java代码  技术分享
  1. case "NetConnection.Connect.Rejected" :  
  2. //注意这里,服务器拒绝你的情况,如果遭到拒绝,将会调用两次mync.onStatus,  
  3. //第一次"NetConnection.Connect.Rejected"第2次"NetConnection.Connect.Closed"  
  4. trace("遭到服务器拒绝");  
  5. trace("服务器返回信息:"+info.application.msg);  
  6. break;   

 遭到服务器拒绝后会trace出服务器返回的错误消息,这是一个最基础的与服务器交互的例子,以后还会有很多滴 
下边进入新一层次。。。 
视频,流 
这个比较重要,但超简单,网上播放电影,在线录制,在线播放,视频会议,视频电话,全靠他了 ,下边我们先做一个最简单 录制和播放 
录制视频 
打开flash,新建一个fla,Ctrl + L 打开library,右键新建视频,确定。 
拉到舞台上起个名叫my_video 
第一帧开始加代码

Java代码  技术分享
  1. //从麦和设像头显示视频显示在my_video上   
  2. my_video.attachVideo(Camera.get());  
  3. my_video.attachAudio(Microphone.get());   
  4. //像以前一样连接   
  5. nc = new NetConnection();  
  6. nc.connect("rtmp://localhost/aaaa"); //注意这里,Applications文件夹里要有aaaa文件夹哦!   
  7. //可以理解为在nc连接上绑一个流   
  8. nsOut = new NetStream(nc);   
  9. //在流上加麦克风和视频头   
  10. nsOut.attachVideo(Camera.get());  
  11. nsOut.attachAudio(Microphone.get());   
  12. //发布 2.flv  
  13. nsOut.publish("2", "record");   

第一个参数是文件名,后一个参数要"record"才是录制 
把fla发布一下, 录一会儿,把视频关掉,打开你的
叉盘:\Program Files\Macromedia\Flash Media Server 2\applications\aaaa\
是不是多了个streams\_definst_
打开C:\Program Files\Macromedia\Flash Media Server 2\applications\aaaa\streams\_definst_
看见2.flv了吧。。 
这里下载原文件 : 
http://www.nshen.net/blog/attachments/200601/25_163617_c1.fla
播放flv 
用fms播放的flv目前是无法下载的,这可以保护你的版权 :) 
打开flash,新建一个fla ,Ctrl + L 打开library,右键新建视频,确定。 
拉到舞台上,这回起个名叫view ,我们来播放你刚才录的那个视频

Java代码  技术分享
  1. nc = new NetConnection();  
  2. nc.connect("rtmp://localhost/aaaa");  
  3. res = new NetStream(nc);   
  4. //view元件要加载res流   
  5. view.attachVideo(res);  
  6. view.attachAudio(res);   
  7. //播放   
  8. res.play("2");   

 原文件 : 
http://www.nshen.net/blog/attachments/200602/13_223604_p1.fla

现场流 
上边做的都是先录制好了视频,然后才播放,网上的实时视频会议,视频电话是怎么做的呢?总不能先录好再播放吧? 
这要用到现场流,现场流是指你连接到服务器后,你在发布的同时,其他人就可以实时的看到你 
很难吗?看看吧,把上边录制视频的例子拿下来 
nsOut.publish("2", "record"); 这句改成 nsOut.publish("2", "live"); 
把"record" 改成"live"后,就不会生成flv了,取而代之的是一个看不到的实时的视频流 
ok 了,发布,这就是直播端了,同时再发布上边那个播放端,已经可以实时看见你了吧。现在你是用本机测试,等你有了服务器。其他人也能同时看见你了哦 
到这里你已经可以自己做一个网页上的直播了,发布端不要让别人看到,让别人看你的播放端就好了 :) 至于为什么要用两个swf,因为目前为止你还不知道怎么样跟服务器之间传递消息,这样做可以避免这些东西,等你把后边的东西全都学完就可以在一个swf里,写个视频会议之类的东东了
远程共享 
远程共享?
共享的概念就是让每个连接到服务器的swf都能实时的得到服务器端共享的数据。
一个人更改了这些数据,其他人都会看得到通知。可以想象聊天室里的发言,一个人发了以后其他人都可以看到。
 
共享对象
共享对象,说英文大概你比较熟ha~ SharedObject, 恩flash中有两种sharedObject,local sharedobject (LSO) 和 remote sharedobject (RSO),也就是本地共享和远程共享,偶们讨论远程的,不过之前你最好先去了解了解本地的,对你有好处...
RSO在服务器端是以文件形式存储的,扩展名是.fso,为什么不是.rso?....我也想问呢- -b
代码
初始化RSO需要先与服务器建立一个连接,续上节 ,我们先与服务器建立一个连接

Java代码  技术分享
  1. //初始化远程共享要利用nc通道  
  2. var myNC = new NetConnection();  
  3. myNC.onStatus = function(info) {  
  4. if (info.code == "NetConnection.Connect.Success") {  
  5. //成功则利用此nc初始化rso  
  6. initRSO(this);  
  7. }  
  8. };   

 跟以前的代码一样,只是连接成功后多了一个initRSO()函数,看不懂的回去重头再看一下。。。
下边是initRSO了,跟连接结构差不多

Java代码  技术分享
  1. function initRSO(NC) {  
  2. //在服务器上建立myRSO.fso文件,第2个参数指定nc通道,第3个指定文件在服务器上持久保留,即使服务器重启了,还是有  
  3. my_rso = SharedObject.getRemote("myRSO", NC.uri, true);  
  4. my_rso.onSync = function(list) {  
  5. //list 是一个对象数组 ,类似这种[{name:"x",code:"success"},{name:"y",code:"success"}] ,下边会详细讲  
  6. //初始成功  
  7. };  
  8. my_rso.connect(NC); //连接  
  9. }   

 了解了吧,看一个完整的例子
画一个mc起名叫mc,在第一帧上写代码,

Java代码  技术分享
  1. //初始化远程共享要利用nc通道  
  2. var myNC = new NetConnection();  
  3. myNC.onStatus = function(info) {  
  4. if (info.code == "NetConnection.Connect.Success") {  
  5. //成功则利用此nc初始化rso  
  6. initRSO(this);  
  7. }  
  8. };  
  9. myNC.connect("rtmp://localhost/test1"); //不会不知道要建test1文件夹吧   
  10. function initRSO(NC) {  
  11. my_rso = SharedObject.getRemote("myRSO", NC.uri, true);   
  12. //onSync是回调函数,每次服务器端so数据有改变,这里都会有反映!这里的意思每当有人按鼠标,这里都会有反映,我们读取so的data下的值就可以了  
  13. my_rso.onSync = function() {  
  14. mc._x=this.data.x  
  15. mc._y=this.data.y  
  16. };  
  17. my_rso.connect(NC);  
  18. }   
  19. onMouseDown = function () {   
  20. //改变so的数据  
  21. my_rso.data.x = _root._xmouse  
  22. my_rso.data.y = _root._ymouse  
  23. };   

 然后发布设置中设置只允许网络,发布看看
现在你可以开多个播放器窗口,点其中一个,看看其他的窗口变不变
源文件: 
http://www.nshen.net/blog/attachments/200601/11_145320_test3.fla

连接流程
 
再写一个,看起来很像在做网游~

Java代码  技术分享
  1. mync = new NetConnection();  
  2. mync.onStatus = function(info) {  
  3. if (info.code == "NetConnection.Connect.Success") {  
  4. initRSO();  
  5. }  
  6. if (info.code == "NetConnection.Connect.Closed") {  
  7. trace("关闭");  
  8. }  
  9. };   
  10. function initRSO() {  
  11. my_RSO = SharedObject.getRemote("myRSO", mync.uri, true);  
  12. trace(my_RSO);  
  13. my_RSO.onSync = function() {  
  14. mc._x = this.data.hero.x;  
  15. };  
  16. my_RSO.connect(mync);  
  17. }  
  18. mync.connect("rtmp:/my_app/test1");  
  19. mc.onEnterFrame = function() {  
  20. my_RSO.data.hero.x = this._x;  
  21. if (Key.isDown(Key.LEFT)) {  
  22. this._x -= 5;  
  23. }  
  24. if (Key.isDown(Key.RIGHT)) {  
  25. this._x += 5;  
  26. }  
  27. };   

写个简单的聊天室 
很简单的东西,基本上就是共享对象的运用,没有用到服务器端,大型聊天室可能不会这么做,这个只适用于初学者 :/
注释很详细,不说多了,可以直接下载原文件
http://www.nshen.net/blog/attachments/200602/smallchat.fla

代码:

Java代码  技术分享
  1. //用户名  
  2. myname="游客"  
  3. //建立连接  
  4. var myNC = new NetConnection();  
  5. myNC.connect("rtmp://localhost/smallchat");  
  6. //搞到rso  
  7. Talk_SO = SharedObject.getRemote("Talk", myNC.uri, false);  
  8. Talk_SO.onSync = function() {  
  9. //先把聊天文本框清空  
  10. remoteText.text = "";  
  11. //把聊天列表显示出来,talklist的格式就是[谁谁说:啊啊啊,谁谁谁说:2222]  
  12. var t = this.data.talklist;  
  13. for (var i = 0; i=5) {  
  14. Talk_SO.data.talklist.shift();  
  15. }  
  16. //把消息装到so里  
  17. Talk_SO.data.talklist.push(myname+"说:"+meText.text);  
  18. meText.text = "";  
  19. }  
  20. //文字显示,换行  
  21. function writeln(msg) {  
  22. remoteText.text += msg+"\n";  
  23. remoteText.vPosition =remoteText.maxVPosition  
  24. }  
  25. //-----------------------------------------------  
  26. Btn.onRelease = function() {  
  27. post();  
  28. };  
  29. this.onKeyDown = function() {  
  30. if (Key.isDown(Key.ENTER)) {  
  31. post();  
  32. }  
  33. };  
  34. Key.addListener(this);   

深入onSync 
onSync有个list参数,这个开始有些难度了。不想动脑的可以跳过没影响,只是以后写出的程序效率会低点 :(
看代码:
my_rso = SharedObject.getRemote("myRSO", NC.uri, true);
my_rso.OnSync= function(list) {//.......};
my_rso.connect(NC); //连接 
在onSync回调中我们可以知道我们的my_rso被改变了,但my_rso里具体什么改变了呢? 我们就要分析这个 list 参数 了
list参数其实是一个对象数组 ,首先它是一个数组,里边装了很多对象(Object),每一个对象都包括了SharedObject中一个插槽(slot)的改动信息。我暂时给他起名叫插槽信息对象。。。这名字太猥亵了。。但我就这么叫了。。
插槽信息对象包含两个属性,name 和 code,偶尔还会有个oldValue?我不太常用,不说它
name 描述被改变的属性名
code 描述该属性的改变方式 ,有可能为以下几种值:"success" , "change" , "delete" , "reject" , "clear" ,具体含义后边说
说白了这个插槽信息对象大概就是这么个样子:
{name:"x",code:"success"} 
表示x属性被修改成功
要得到这些插槽信息对象就要for in 这个list参数
for (var i in list) {
list[i] 就是插槽信息对象
}
要分析具体so哪改变了,就是分析list[i],比如
if(list[i].code=="change") trace("list[i].name"+被+"change了")
if(list[i].code=="delete") trace("list[i].name"+被+"delete")
“change”是啥?“delete”是啥?
"success" , "change" , "delete" , "reject" , "clear" 具体含义:
success : 表示当前影片修改so的插槽获得了成功
change : 表示so的插槽被别人修改,或填加
也就是说,你修改so的某个属性成功了会收到 "success" ,与此同时其他影片会收到 "change"
reject : 拒绝修改
例如发生在两个或多个客户端同时要修改一个so的插槽,这时候fms会只让一个client修改,并返回"success" 其他的会收到"reject"
delete , clear : 这个好理解,一个是删除,一个是清空,看例子:
比如服务器端删除某个so
so = SharedObject.get("某个so");
so.lock( );
var names = so.getPropertyNames( );
for (i in names) {
so.setProperty(names[i], null);
}
so.unlock( ); 
这样client端会收到 若干个插槽信息对象,所有的code都为"delete",表示若干个item被删除
然而这样:
so = SharedObject.get("某个so");
so.clear( ); 
client端就只会收到一个插槽信息对象,code属性为“clear”。 
client端与server端直接交互 
看完了SO,看一下client与server端如何直接进行交互的 
原文件在这里: 
http://www.nshen.net/blog/attachments/200602/25_152648_csc.rar
1. 客户端呼叫服务器
fla: 
//客户端呼叫server端msgfromclient函数,并将返回值trace出来
mync = new NetConnection();
mync.connect("rtmp://localhost/connect");
//返回值接收对象
var resObj = new Object();
resObj.OnResult= function(val):Void {
trace("val"+val);
}; 
/* 
我们用mync去call服务器端的msgfromclient函数,resObj是返回接收对象,当服务器有返回值后,会自动直接调用这个对象的onResult处理函数,后边可以传递给server无数个参数,这里只传一个字符串 
*/
mync.call("msgfromclient", resObj, "第一个call"); 
服务器端代码是放在main.asc里的,你可以到你的application下的connect目录下建一个main.asc,写代码
main.asc: 
//要把函数定义到Client上!!
application.OnConnect= function(client) {
/* 在这里定义也可以,在Client.prototype里定义也是可以的 
client.msgfromclient=function(what){
trace(what+"进来了")
var aa="呼叫成功并返回结果"
return aa
}
*/
application.acceptConnection(client);
};
Client.prototype.msgfromclient=function(what){
trace(what+"进来了")
var aa="呼叫成功并返回结果"
return aa


现在去试一下吧。。。成功了的话,再继续....... 
2. 服务器端呼叫指定的客户端
fla:

Java代码  技术分享
  1. //server呼叫client端  
  2. //要把函数定义到nc上!!  
  3. //  
  4. mync = new NetConnection();  
  5. mync.onStatus = function(info) {  
  6. if (info.code == "NetConnection.Connect.Success") {  
  7. trace("连接成功");  
  8. }  
  9. };  
  10. mync.connect("rtmp://localhost/connect");  
  11. mync.msgfromserver = function(msg) {  
  12. trace(msg);  
  13. };   
  14. main.asc:   
  15. application.onConnect = function(client) {   
  16. application.acceptConnection(client);   
  17. //这里呼叫刚连线成功的客户   
  18. //跟client呼叫server基本一样,服务器一般很少让client端返回值所以第2个参数设为null  
  19. client.call("msgfromserver",null,"服务器叫你啊")  
  20. };   

3. 服务器端呼叫所有的客户端(广播)
有些时候需要服务器广播数据给所有连接上的客户端,这里就用到了广播的概念 
广播其实SharedObject的时候已经讲过了一种实现,就是把数据放到remote SharedObject中,当数据改变了,自然所有客户端都会onSync 
这里再讲一种用call来实现的: 
下边是很常见的一个情况,当某人下线了的时候要通知所有客户端,某某已经下线了 
server端:
application.OnDisconnect=function(newClient){ 
//遍历客户端列表,分别call他们
for(var i=0;i
}
Client端:
nc.client_fun=function(myvar){....... } 
这个自己完善一下吧,这里就不贴fla了 
还有: 
关于广播,不只有服务器端广播给所有客户,还有可能某一个客户端对所有客户端直接进行广播,当然上边的例子你如果都看懂了的话,你已经可以自己做某一个客户端对所有客户端的广播了。怎么做? 
第1步 某一个客户端呼叫服务器 
第2步 服务器广播给所有客户端 
这样就形成了,某客户端对所有客户端的广播,当然如果你能细心的耐心的看看帮助的话,你会发现Shared Object 和 NetStream都有send方法就是做这件事的,而且更为简洁,服务端不用写代码 :) 
好了,看到这里,fms常用的大部分概念都说到了,这个教程也就基本结束了,但请注意,现在你只是入门阶段.想学更多的,你可能需要多翻翻手册,多找找教程,英文有不少很好的教程进阶. 
本文原地址:
http://www.nshen.net/blog/?id=509
,转载请保留链接 
其他需要注意的问题: 
* 中文编码: 
有些时候我们用flash去读取外部的php,asp.....文件里的中文显示在flash里会出现乱码的情况,为了解决在flash里显示中文很多教程里通常都直接加了一句System.useCodepage=true 
问题就在这,显示不了外部中文是因为flash内默认用Unicode编码,外部的文件大多都是gb2312,加上这句System.useCodepage=true代表强制flash使用系统默认的gb2312,这样flash就显示正确了,但fms服务器端默认也是用unicode的,这样客户端跟服务器端不同编码有时就会出错了,搜了一下server字典好象没有System.useCodepage=true了。。。所以解决办法就是去掉System.useCodepage=true,在外部php或asp中把编码转成utf-8,至于怎么转,不知道,问你的asp或php程序员吧 ,另外不要用记事本编辑你的asc文件。。即使编辑最后要一定另存为utf-8格式。
* 判断影片播放结束
ns.OnStatus=function(info){
if(info.code=="NetStream.Play.Stop")trace("结束")
}
乍看好象没错,但是如果设置了缓冲以后(setBufferTime)就不好用了,仔细研究了一下原因4这样的
监视onStatus(info) ,info.code:
开始播放
NetStream.Play.Start (其实还没播放)
然后缓冲(根据setBufferTime设置的秒数缓。。)
NetStream.Buffer.Full (缓冲装满了,这才开始播放)
然后播放完了
NetStream.Play.Stop (其实还没播放完)
注意了,然后还要播放缓冲 - -b
NetStream.Buffer.Empty (缓冲空了,这才播放完了。。)
群里的kinglong兄比较聪明~,先Stop的时候做个记号,然后再满足Empty才算播放完,也就是两个条件,因为网速慢也会Empty。。。好办法
但我看了一下帮助,好象有一个专门的事件通知播放结束
ns.OnPlayStatus=function(info){
if(info.code=="NetStream.Play.Complete")trace("感谢观看帮助")
}
* 防火墙,端口
默认安装的话默认端口是1935,管理是1111端口,记得防火墙要把1935和1111端口打开。

FMS入门教程


推荐阅读
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 知识图谱——机器大脑中的知识库
    本文介绍了知识图谱在机器大脑中的应用,以及搜索引擎在知识图谱方面的发展。以谷歌知识图谱为例,说明了知识图谱的智能化特点。通过搜索引擎用户可以获取更加智能化的答案,如搜索关键词"Marie Curie",会得到居里夫人的详细信息以及与之相关的历史人物。知识图谱的出现引起了搜索引擎行业的变革,不仅美国的微软必应,中国的百度、搜狗等搜索引擎公司也纷纷推出了自己的知识图谱。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 后台获取视图对应的字符串
    1.帮助类后台获取视图对应的字符串publicclassViewHelper{将View输出为字符串(注:不会执行对应的ac ... [详细]
  • 《数据结构》学习笔记3——串匹配算法性能评估
    本文主要讨论串匹配算法的性能评估,包括模式匹配、字符种类数量、算法复杂度等内容。通过借助C++中的头文件和库,可以实现对串的匹配操作。其中蛮力算法的复杂度为O(m*n),通过随机取出长度为m的子串作为模式P,在文本T中进行匹配,统计平均复杂度。对于成功和失败的匹配分别进行测试,分析其平均复杂度。详情请参考相关学习资源。 ... [详细]
  • 本文介绍了通过ABAP开发往外网发邮件的需求,并提供了配置和代码整理的资料。其中包括了配置SAP邮件服务器的步骤和ABAP写发送邮件代码的过程。通过RZ10配置参数和icm/server_port_1的设定,可以实现向Sap User和外部邮件发送邮件的功能。希望对需要的开发人员有帮助。摘要长度:184字。 ... [详细]
  • 如何使用Java获取服务器硬件信息和磁盘负载率
    本文介绍了使用Java编程语言获取服务器硬件信息和磁盘负载率的方法。首先在远程服务器上搭建一个支持服务端语言的HTTP服务,并获取服务器的磁盘信息,并将结果输出。然后在本地使用JS编写一个AJAX脚本,远程请求服务端的程序,得到结果并展示给用户。其中还介绍了如何提取硬盘序列号的方法。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 1,关于死锁的理解死锁,我们可以简单的理解为是两个线程同时使用同一资源,两个线程又得不到相应的资源而造成永无相互等待的情况。 2,模拟死锁背景介绍:我们创建一个朋友 ... [详细]
  • 本文介绍了RPC框架Thrift的安装环境变量配置与第一个实例,讲解了RPC的概念以及如何解决跨语言、c++客户端、web服务端、远程调用等需求。Thrift开发方便上手快,性能和稳定性也不错,适合初学者学习和使用。 ... [详细]
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
author-avatar
mobiledu2502863347
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有