织梦CMS - 轻松建站从此开始!

罗索

在局域网内实现图像的实时传输

jackyhwei 发布于 2011-02-20 12:47 点击:次 
节点与服务器的交互采用C/S模式,使用控件ClientSocket/ServerSocket,节点间传输信息,由于需要实现接收和发送两个功能,所以在每个节点都要有两个控件,一个用来接收,一个用来发送,该系统中使用了NMStrm/NMStrmServ控件。
TAG:

作者:bat603  网址:http://blog.csdn.net/bat603/(本文可以随意转载及修改并可用于任何用处,但须注明作者和网址)

在局域网内实现图像的实时传输(实现环境bcb6.0+MSSQL)

本来要去睡觉的,但是为了整理一下该文档,还是坐在了电脑旁。明天睡个大头觉。之所以使用bcb6.0,是因为用它开发速度快。但是一个前辈曾经给我说过,bcb开发的软件没有商业价值,当然他说的有点偏颇,不过也反映了bcb的境遇。不过如果开发较小的项目,使用它的优势还是明显的。
转入正题,在网络传输信息,有两种模型:C/S、P2P。C/S模式需要在网络内有个服务器,客户端传输信息都要通过服务器进行转发。当传输信息量较小的文本信息时,采用该方法因为其实现较简单,方便控制信息的传输,所以可以使用,以前的QQ就是采用这个方法,但是如果传输信息量较大的图像信息时,显然当客户端较多时,服务器的负担会呈级数增加,显然是不合适的。只有使用P2P模式,即节点之间传输,这样可以把负载平衡到各个节点,效率很高。但是它的缺点也是很明显的,每个节点都要负责维护其他节点的状态信息,实现起来比较复杂。这个也是现在的研究热点,并有成品出现,但是也是应用于较小的网络。
在做这个网络协作系统时,由于人力资源较弱,所以我采用了这种的方法。就是把这两者结合,在图像信息传输的时候采用P2P模式,但是需要在网络内运行一个服务器来负责维护各个节点的状态信息。这样可以极大的减少服务器的负荷,同时节点间传输的实现也比较容易。
实现方法:服务器运行,监视各个节点的状态。当一个节点登陆网络时,需要向服务器报告自己的状态,并请求得到它希望传输图像信息的状态信息。当得到信息时,便不再与服务器进行交互,而是自己把图像向节点传输。当该节点退出网络时要向服务器报告。可见服务器的工作就是一个索引服务器,而负载已经平衡到了各个节点。
应用背景:在网络协作学习系统中实时协作学习电子白板
系统中的电子白板是一个虚拟公共学习区域,在问题求解的过程中它作为主要的问题解决方案编辑的协作空间,而不同于聊天室只承担组内成员沟通和情感维系的工作,以及与教师就学习问题互动的渠道。
 • 实时协作学习电子白板的功能
 ①基本的书写的功能,可以对文字进行协作编辑
 ②对文字可以进行重点标示,以及擦除。
 ③对协作动作进行控制,即在操作前必须首先控制申请,操作结束后解除控制。
 ④可以对文字和标记进行点对点的传输,亦可在组内广播。

代码实现:节点与服务器的交互采用C/S模式,使用控件ClientSocket/ServerSocket,节点间传输信息,由于需要实现接收和发送两个功能,所以在每个节点都要有两个控件,一个用来接收,一个用来发送,该系统中使用了NMStrm/NMStrmServ控件。数据库采用MSSQL,当然要存放在服务器上。有必要强调的是,要把图像的BMP格式转化为JPG格式在进行传输,这样可以大大缩小传输的信息量。

该方法的 缺点:在传输图像时,采用的是传输整个图像方法。虽然经过格式优化,但是还是有很大的信息量,较好的方法是传输在白板上的动作信息,比如画线动作,可以采集关键部位的坐标和画笔颜色信息传输,这样更能减少信息量的传输(仲日给提的建议),但是实现起来较麻烦,考虑到时间问题,没有实现。
部分源代码:(部分代码参考csdn的bcb版和www.ccrun.com)
节点端

#include <vcl.h>
#pragma hdrstop

#include "board.h"
#include "Unit7.h"
#include "Unit1.h"
#include <jpeg.hpp>
#pragma package(smart_init)
#pragma resource "*.dfm"
TWhiteBoard *WhiteBoard;

__fastcall TWhiteBoard::TWhiteBoard(TComponent* Owner)
    : TForm(Owner)
{
}

void __fastcall TWhiteBoard::FormCreate(TObject *Sender)
{
    //禁用关闭按钮
    EnableMenuItem(GetSystemMenu(Handle,false), SC_CLOSE,
        MF_DISABLED | MF_BYCOMMAND | MF_GRAYED);

    Button3->Enabled = false;

    Timer1->Enabled = false;
    m_npenFlag = 0;//初始化画笔的功能
    Image->Parent->DoubleBuffered = true;
    void *dsdc;
    void *dxwnd;

    dxwnd=GetDesktopWindow();//取得桌面句柄
    dsdc=GetDC(dxwnd);
    BitBlt(Image->Canvas->Handle,0,0,NULL,NULL,dsdc,0,0,SRCCOPY);
    ReleaseDC(dxwnd,dsdc);
}

void __fastcall TWhiteBoard::ImageMouseMove(TObject *Sender,
      TShiftState Shift, int X, int Y)
{
    //m_oldCursor =  Screen->Cursor;
    if (m_npenFlag == 1)//输入文本
    {

        if (Shift.Contains(ssCtrl))//按下ctrl,要改变文本大小
        {
            int xstart,ystart;

            Screen->Cursor =  crSizeNWSE;
            xstart = myMemo->Left;
            ystart = myMemo->Top;
            myMemo->Width = X - xstart;
            myMemo->Height = Y - ystart;

        }
        else
        {
            Screen->Cursor = crCross;
        }

    }

if (m_npenFlag == 2)//画笔
{
     if(Shift.Contains(ssLeft))
     {
        if(x1==-1&&y1==-1)
        {
            x1=X;
            y1=Y;
            //Image->Canvas->Pen->Color=clRed;
            Image->Canvas->Pen->Width=3;
            Image->Canvas->MoveTo(x1,y1);
            Image->Canvas->LineTo(X,Y);
        }
        else
        {
            //Image->Canvas->Pen->Color=clRed;
            Image->Canvas->Pen->Width=3;
            Image->Canvas->MoveTo(x1,y1);
            Image->Canvas->LineTo(X,Y);
            x1=X;
            y1=Y;
        }
    }
}
    if (m_npenFlag == 3)//橡皮
    {
        if(Shift.Contains(ssLeft))
        {
        if(x1==-1&&y1==-1)
        {
            x1=X;
            y1=Y;
            //Image->Canvas->Pen->Color=clRed;
            Image->Canvas->MoveTo(x1,y1);
            Image->Canvas->LineTo(X,Y);
        }
        else
        {
            //Image->Canvas->Pen->Color=clRed;
            Image->Canvas->MoveTo(x1,y1);
            Image->Canvas->LineTo(X,Y);
            x1=X;
            y1=Y;
        }
        }
    }
}

void __fastcall TWhiteBoard::ImageMouseUp(TObject *Sender,
      TMouseButton Button, TShiftState Shift, int X, int Y)
{
     if(Button==mbLeft)
    {
        x1=-1;
        y1=-1;
    }

}

void __fastcall TWhiteBoard::BitBtn2Click(TObject *Sender)
{
    m_npenFlag = 2;
    if (ColorDialog->Execute())
        Image->Canvas->Pen->Color = ColorDialog->Color;
}

//产生输入文本框
void __fastcall TWhiteBoard::ImageMouseDown(TObject *Sender,
      TMouseButton Button, TShiftState Shift, int X, int Y)
{
    if (m_npenFlag == 1)
    {

        if (Button==mbLeft )
        {
            if (myMemo != NULL)
            {
                delete myMemo;
                myMemo = NULL;
            }
            myMemo = new TMemo(Owner);
            myMemo->Parent = Panel1;
            myMemo->Left = X;
            myMemo->Top = Y;
            myMemo->Width = 500;
            myMemo->Height = 20;
            myMemo->Ctl3D = false;
            myMemo->OnMouseMove = Memo1MouseMove;
        }
    }
    else
        if (m_npenFlag == 2)
        {
            x1 = X;
            y1 = Y;
        }

        if (m_npenFlag == 3)
        {
            Image->Canvas->Pen->Color =  clCaptionText;
            Image->Canvas->Pen->Width = 13;
            Image->Canvas->Rectangle(X-1, Y-1, X, Y);

        }
}

void __fastcall TWhiteBoard::BitBtn1Click(TObject *Sender)
{
    m_npenFlag = 1;
}

void __fastcall TWhiteBoard::Panel1MouseMove(TObject *Sender,
      TShiftState Shift, int X, int Y)
{
    if (m_npenFlag == 1)
    {
        if(X<0||X>Panel1->Width||Y<0||Y>Panel1->Height)
        {
                ReleaseCapture();
                Screen->Cursor = crArrow;
               // Label1->Caption="Leave";//鼠标离开事件
        }
        else
        {
                if(Panel1->Handle!=GetCapture())
                {
                        SetCapture(Panel1->Handle);
                        Screen->Cursor = crCross;
                       // Label1->Caption="Enter";//鼠标进入事件
                }
        }
    }
    if (m_npenFlag == 2)
    {
        if(X<0||X>Panel1->Width||Y<0||Y>Panel1->Height)
        {
                ReleaseCapture();
                Screen->Cursor = crArrow;
               // Label1->Caption="Leave";//鼠标离开事件
        }
        else
        {
                if(Panel1->Handle!=GetCapture())
                {
                        SetCapture(Panel1->Handle);
                        Screen->Cursor = crHandPoint;
                       // Label1->Caption="Enter";//鼠标进入事件
                }
        }
    }

     Label1->Caption = IntToStr(X)+","+IntToStr(Y);

}

void __fastcall TWhiteBoard::Memo1MouseMove(TObject *Sender,
      TShiftState Shift, int X, int Y)
{
    //if (Shift.Contains(ssCtrl)
   // {
    Label1->Caption = IntToStr(X)+","+IntToStr(Y);
     Screen->Cursor = crIBeam;

}

void __fastcall TWhiteBoard::FormMouseMove(TObject *Sender,
      TShiftState Shift, int X, int Y)
{

    Screen->Cursor = crArrow;
}



void __fastcall TWhiteBoard::BitBtn3Click(TObject *Sender)
{
    Image->Canvas->TextOutA(myMemo->Left,myMemo->Top,myMemo->Text);
    if (myMemo != NULL)
    {
        delete myMemo;
        myMemo = NULL;
    }
}

void __fastcall TWhiteBoard::BitBtn4Click(TObject *Sender)
{
    Screen->Cursor = crNoDrop;
    m_npenFlag = 3;
}

void __fastcall TWhiteBoard::NMStrmServ1MSG(TComponent *Sender,
      const AnsiString sFrom, TStream *strm)
{
    TMemoryStream *ImageStream;
    ImageStream = new TMemoryStream;
    strm->Seek(0,soFromBeginning);
    TJPEGImage *jpeg;  // 定义JPEG图象
    try
    {
        jpeg = new TJPEGImage;  // 分配内存
        // 从数据流中载入图象
        jpeg->LoadFromStream(strm);
        // 显示图象
        Image->Picture->Bitmap->Assign(jpeg);
        //MessageBeep(MB_OK);  // 发出提示声音
    }
    __finally {
        delete jpeg;  // 释放资源
    }
    delete ImageStream;
    ImageStream = NULL;
}

void __fastcall TWhiteBoard::Button2Click(TObject *Sender)
{
    WorkRoom->ClientSocket->Socket->SendText("4:"+NBCL->m_sUserID+":Request");//申请白板控制权
    Button2->Enabled = false;
    Button3->Enabled = true;
}

//定时传输图像,以达到实时传输目的
void __fastcall TWhiteBoard::Timer1Timer(TObject *Sender)
{

        TMemoryStream  *imgstream;
        imgstream = new TMemoryStream;
        Graphics::TBitmap *bBitmap;  // 定义位图变量
     try {
      bBitmap = new Graphics::TBitmap(); // 创建位图

      // 拷贝屏幕的指定区域到位图
            bBitmap->Assign(Image->Picture->Bitmap);
   TJPEGImage *jpeg;
   try {
    jpeg = new TJPEGImage;  // 创建JPEG图象
    jpeg->Assign(bBitmap);  // 将位图转化为JPEG格式
    jpeg->SaveToStream(imgstream);  // 保存JPEG图象信息
   }
    __finally {
    delete jpeg;  // 释放资源
   }
   }
 __finally {
  delete bBitmap;  // 释放资源
 }
    //向服务器发送图像信息
   /* AnsiString shostName = WorkRoom->ClientSocket->Socket->RemoteHost;
    try
    {   imgstream->Position = 0;
        NMStrm1->Host=shostName; //指定主机名
        NMStrm1->PostIt(imgstream); //发送的文件
    }
    catch(...){}
 */

    //找到其他组成员
    vFindOtherHost();
    for (int i=0; i<m_nHostNum; i++)
    try
    {   imgstream->Position = 0;
        NMStrm1->Host=m_sOtherHost[i]; //指定主机名
        NMStrm1->PostIt(imgstream); //发送的文件
    }
    catch(...){}
   delete imgstream;
   imgstream = NULL;

}

void __fastcall TWhiteBoard::Button3Click(TObject *Sender)
{
    WorkRoom->ClientSocket->Socket->SendText("4:"+NBCL->m_sUserID+":Giveup");//放弃白板控制权
    Button2->Enabled = true;
    Button3->Enabled = false;
    Timer1->Enabled = false;
    WorkRoom->Memo->Lines->Add("我放弃了白板控制权");

}




void __fastcall TWhiteBoard::FormHide(TObject *Sender)
{
    Timer1->Enabled = false;
    if (Button3->Enabled)
        WorkRoom->ClientSocket->Socket->SendText("4:"+NBCL->m_sUserID+":Giveup");//放弃白板控制权
    if (WorkRoom->Showing)
        WorkRoom->Hide();
}

//得到其他节点状态
void TWhiteBoard::vFindOtherHost()
{
    //找到同组在线人的计算机名称
    //读取本组在线成员信息表
    AnsiString SQL;
    AnsiString sUserID,sUserName;
    AnsiString sHost,sIP;
    SQL = "  declare @teamid int ";
    SQL += " select @teamid=TeamID from StudentGroupInfo where UserID='"+NBCL->m_sUserID+"'";
    SQL += " select R.UserID,UserName,O.Host,O.IP from StudentRegisterInfo as R inner join StudentGroupInfo as G on

R.UserID=G.UserID and G.TeamID=@teamid inner join OnlineInfo as O on O.UserID=G.UserID";
    NBCL->MainQuery->Close();
    NBCL->MainQuery->SQL->Clear();
    NBCL->MainQuery->SQL->Add(SQL);
    NBCL->MainQuery->Open();
    m_nHostNum = 0;
    while (!NBCL->MainQuery->Eof)
    {
        sUserID = NBCL->MainQuery->FieldByName("UserID")->AsString;
        if (sUserID != NBCL->m_sUserID)
        {
            sUserName = NBCL->MainQuery->FieldByName("UserName")->AsString;
            sHost = NBCL->MainQuery->FieldByName("Host")->AsString;
            sIP = NBCL->MainQuery->FieldByName("IP")->AsString;

            m_sOtherHost[m_nHostNum++] = sHost;
        }
        NBCL->MainQuery->Next();
    }
    //m_nHostNum--;//保存本组主机名的个数
}
void __fastcall TWhiteBoard::Button1Click(TObject *Sender)
{
    if (Button3->Enabled)
    {
        ShowMessage("请放弃控制白板控制后再关闭该窗口");
        return;
    }

    Hide();
}
}
服务器端较简单,不做介绍。若有不妥之处,敬请指点。

(bat603)
本站文章除注明转载外,均为本站原创或编译欢迎任何形式的转载,但请务必注明出处,尊重他人劳动,同学习共成长。转载请注明:文章转载自:罗索实验室 [http://www.rosoo.net/a/201102/10925.html]
本文出处:CSDN博客 作者:bat603
顶一下
(0)
0%
踩一下
(0)
0%
------分隔线----------------------------
发表评论
请自觉遵守互联网相关的政策法规,严禁发布色情、暴力、反动的言论。
评价:
表情:
用户名: 验证码:点击我更换图片
栏目列表
将本文分享到微信
织梦二维码生成器
推荐内容