在很多业务场景下,需要监视显示屏画面。在实时性要求不高的情况下,可以通过定时对显示屏进行截图及回传实现。

本文通过C#中提供的TCP通信功能,对该功能的实现进行简单描述。

首先,该功能的实现分为客户端和服务端。其中客户端发送显示屏截图请求;服务端接收截图请求后,进行截图并回传;客户端收到服务端回传的截图后,进行显示处理。

1. 客户端

1)建立对目标主机的TCP连接

2)向目标主机上的服务端发送截图指令

3)接收目标主机上的服务端发回的内存流(代表截图),并解析为截图

具体代码如下:

// 目标主机网络端点:hostIP表示目标主机IP,hostPort表示端口号
var hostEndPoint = new IPEndPoint(IPAddress.Parse(hostIP), hostPort);

// 连接目标主机的Socket
Socket clientSocket = null;
try
{
    // 创建Socket
    clientSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
    clientSocket.SetSocketOption(SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);

// 连接目标主机
    clientSocket.Connect(hostEndPoint);
}
catch (Exception ex)
{
    // 错误日志
}

// 建立Socket连接失败
if (clientSocket == null || !clientSocket.Connected)
return null;

// 向目标主机发送截图命令,截图命令为字符串常量"GetScreenSnap"
clientSocket.Send(Encoding.ASCII.GetBytes("GetScreenSnap"), "GetScreenSnap".Length, SocketFlags.None);

// 接收并解析截图
MemoryStream snapMemoryStream = null;
Image screenSnap = null;
try
{

// 每次接收1024字节数据(服务端每次发送1024字节)
    byte[] by = new byte[1024];

// 重试次数
    int retryCount = 0;

// 前1024字节数据代表的字符串
    string s = "";
    while (true)
    {

// 成功接收到前1024字节数据
        if ((clientSocket.Receive(by)) > 0)
        {

// 转化为字符串
            s = Encoding.Unicode.GetString(by);
            by = new byte[1024];

// 结束循环
            break;
        }

// 接收失败后,重试次数加1
        retryCount++;

// 重试次数超过5次
        if (retryCount > 5)
        {

// 放弃接收截图
            break;
        }
    }

// 前1024字节代表的字符串不为空
    if (!s.Equals(""))
    {

// 解析出截图内存流的总字节长度
        s = s.Split('!')[0];
        long len = long.Parse(s);

// 代表截图的内存流

snapMemoryStream = new MemoryStream();

// 依次接收字节流

while (len > 0)
        {
            int number;
            if ((number = clientSocket.Receive(by)) > 0)
            {

// 写入内存流
                snapMemoryStream.Write(by, 0, number);
                len -= number;
            }
            else
            {
                break;
            }
           by = new byte[1024];
        }

// 根据内存流创建图片
        snapMemoryStream.Position = 0;
        screenSnap = Image.FromStream(snapMemoryStream);
     }
}
catch (Exception ex)
{
    // 错误日志
}
finally
{
    // 关闭内存流
    if (snapMemoryStream != null)
    snapMemoryStream.Close();

// 关闭连接
    if (clientSocket != null)
    clientSocket.Close();

}

2. 服务端

1)创建并启动TCP侦听器,监听客户端截图请求

2)进行显示屏截图

3)向客户端发送截图

具体代码如下:

// 创建监听器并开始监听
mSnapRequestListener = new TcpListener(localhostIP, port);
mSnapRequestListener.Start();

// 只要服务端在运行,则一直监听
while (isRunning)
{
    TcpClient tcpClient = null;
    Bitmap screenSnap = null;
    try
    {
        // 获取客户端发送的TcpClient
        tcpClient = mSnapRequestListener.AcceptTcpClient();
        if (tcpClient == null || tcpClient.Client == null)
        {
            return;
        }

// 解析命令文本
        var inBuffer = new Byte["GetScreenSnap".Length];
        tcpClient.Client.Receive(inBuffer, "GetScreenSnap".Length, SocketFlags.None);
        var commandText = Encoding.ASCII.GetString(inBuffer);
        // 不是截图命令
        if (commandText != "GetScreenSnap")
            return;

// 创建一张空白图片
var primaryScreenSize = Screen.PrimaryScreen.Bounds.Size;
screenSnap = new Bitmap(primaryScreenSize.Width, primaryScreenSize.Height, PixelFormat.Format32bppArgb);
// 将屏幕复制到图片上
using (var grfx = Graphics.FromImage(screenSnap))
{
        grfx.CopyFromScreen(new Point(0, 0), new Point(0, 0), primaryScreenSize);
}

using (var snapStream = new MemoryStream())
{
    // 将屏幕截图保存入内存流,并将内存流位置设为0
    screenSnap.Save(snapStream, ImageFormat.Jpeg);
    snapStream.Position = 0;

// 截图内存流长度
    var streamLength = snapStream.Length.ToString();
    // 扩充为512位(Unicode编码中为1024字节),右边以'!'填充
    streamLength = streamLength.PadRight(512, '!');

// 初次发送截图内存流长度
    var snapBytes = Encoding.Unicode.GetBytes(streamLength);

do
    {
        tcpClient.Client.Send(snapBytes);
    }
    // 从流中读取1024个字节,直到读完为止
    while (snapStream.Read(snapBytes, 0, 1024) > 0);
}

}catch (Exception ex)
    {
        // 错误日志
    }
    finally
    {
        // 销毁图片
        if (screenSnap != null)
        {
            screenSnap.Dispose();
        }

// 关闭连接
        if (tcpClient != null && tcpClient.Client != null && tcpClient.Client.Connected)
        {
            tcpClient.Client.Close();
        }
    }
}

通过TCP实现显示屏截图请求及回传的更多相关文章

  1. TCP 连接与 HTTP 请求的相关问题

    1.现代浏览器在与服务器建立了一个 TCP 连接后是否会在一个 HTTP 请求完成后断开?什么情况下会断开? 默认情况下建立 TCP 连接不会断开,只有在请求报头中声明 Connection: clo ...

  2. TCP连接与HTTP请求

    一道经典面试题: 从 URL 在浏览器被被输入到页面展现的过程中发生了什么? 相信大多数准备过的同学都能回答出来,但是如果继续问:收到的 HTML 如果包含几十个图片标签,这些图片是以什么方式.什么顺 ...

  3. TCP连接建立系列 — 连接请求块

    连接请求块(request_sock)之于TCP三次握手,就如同网络数据包(sk_buff)之于网络协议栈,都是核心的数据结构. 内核版本:3.6 Author:zhangskd @ csdn blo ...

  4. TCP网络编程-----客户端请求连接服务器、向服务器发数据、从服务器接收数据、关闭连接

    SOCKET m_sockClient; unsigned short portNum; ------------------------------------------------------- ...

  5. jsonp 请求和回传实现

    JSONP最主要的是可以解决跨域问题,不然谁会没事用这种格式. 下面是我用JSONP的一些心得体会: JSONP是JSON with Padding的略称.它是一个非官方的协议,它允许在服务器端集成S ...

  6. IP封包协议头/TCP协议头/TCP3次握手/TCP4次挥手/UDP协议头/ICMP协议头/HTTP协议(请求报文和响应报文)/IP地址/子网掩码(划分子网)/路由概念/MAC封包格式

    IP协议头IP包头格式: 1.版本号:4个bit,用来标识IP版本号.这个4位字段的值设置为二进制的0100表示IPv4,设置为0110表示IPv6.目前使用的IP协议版本号是4. 2.首部长度:4个 ...

  7. 可能会搞砸你的面试:你知道一个TCP连接上能发起多少个HTTP请求吗?

    本文由原作者松若章原创发布,作者主页:zhihu.com/people/hrsonion/posts,感谢原作者的无私分享. 1.引言 一道经典的面试题是:从 URL 在浏览器被被输入到页面展现的过程 ...

  8. 夺命连环问:一个 TCP 连接可以发多少个 HTTP 请求?

    曾经有这么一道面试题:从 URL 在浏览器被被输入到页面展现的过程中发生了什么? 相信大多数准备过的同学都能回答出来,但是如果继续问:收到的 HTML 如果包含几十个图片标签,这些图片是以什么方式.什 ...

  9. 一个 TCP 连接可以发多少个 HTTP 请求

    第一个问题 第二个问题 第三个问题 第四个问题 第五个问题 曾经有这么一道面试题:从 URL 在浏览器被被输入到页面展现的过程中发生了什么? 相信大多数准备过的同学都能回答出来,但是如果继续问:收到的 ...

随机推荐

  1. rpm打包工具---FPM

    FPM的安装:fpm是由ruby gem仓库里面安装的所以要先装ruby.yum安装的ruby版本是1.8.7版本,使用gem命令会报错: >= 1.9.3,所以要安装一个比1.9.3版本高的 ...

  2. 关于JSP页面Property [ssid] not found on type [java.lang.String]错误的注意事项

    转发于:http://blog.csdn.net/w86440044/article/details/28277177

  3. [poj2585]Window Pains_拓扑排序

    Window Pains poj-2585 题目大意:给出一个4*4的方格表,由9种数字组成.其中,每一种数字只会出现在特定的位置,后出现的数字会覆盖之前在当前方格表内出现的.询问当前给出的方格表是否 ...

  4. dom4j 最常用最简单的用法(转)

    要使用dom4j读写XML文档,需要先下载dom4j包,dom4j官方网站在 http://www.dom4j.org/目前最新dom4j包下载地址:http://nchc.dl.sourceforg ...

  5. 使用Python中的mock模块进行单元测试

    在进行单元测试的时候,有时候会遇到这种情况: 出于某些原因,我们不想测试某一部分内容,但是我们想要测试的部分却依赖这部分内容. 这时候,可以使用mock模块来模拟调用这部分内容,并给出返回结果,举例如 ...

  6. js和jquery判断checkbox是否被选中

    js判断: if(document.getElementById("checkboxID").checked){ alert("checkbox is checked&q ...

  7. 跨平台原生AR/VR应用研发引擎-NVisionXR开放内测

      NVisionXR引擎正式开放内测.现在,对原生AR/VR应用开发有兴趣的企业和开发者均可通过NVisionXR官网(www.nvisionxr.com)申请试用. NVisionXR引擎介绍视频 ...

  8. 新事物学习---Chrome上使用PWA

    PWA是什么 PWA(Progressive Web Apps)是 Google 最近在提的一种 Web App 形态 (或者如 Wikipedia 所称的"软件开发方法").PW ...

  9. Beta冲刺 总结

    Beta冲刺 总结 1. 完成情况 经过了为其七天的beta冲刺,我们基本完成了之前在<beta开始前准备>博客中所列出的内容. 增加关于征信的功能,贴近选题主题.在学生的信用活动记录中添 ...

  10. B-day5

    1.昨天的困难,今天解决的进度,以及明天要做的事情 昨天的困难:昨天虽然完成了风险数据的图表统计,但是界面风格仍然不太满意,还在抓紧调试中:还有登录页的背景图,在想应该如何设计, 什么样的风格才好. ...