在网上流传的gh0st3.6源代码中,远程桌面总是存在CPU占用率高和画面更新不及时等问题。于是想到了著名的开源远程控制RealVNC  它采用了远程帧缓存的协议(Remote Frame buffer)

在网上找到的一段关于RFB的描述

RFB 是真正意义上的“瘦客机”协议。RFB协议设计的重点在于减少对客户端的硬件需求。这样客户端就可以运行在许多不同的硬件上,客户机的任务实现上就会尽量的简单。

RFB协议对于客户端是无状态的。也就是说:如果客户端从服务器端断开,那么如果它重新连接相同的服务器,客户端的状态会被保存。甚至,一个不同的客户端可以用来连接相同的RFB服务器。而在新的客户端已经能够获得与前一个客户端相同的用户状态。因此,用户的应用接口变的非常便捷。

只要合适的网络连接存在,那么用户就可以使用自己的应用程序,并且这些应用会一直保存,即使在不同的接入点也不会变化。这样无论在哪,系统都会给用户提供一个熟悉、独特的计算环境。

显示协议

显示协议是建立在“把像素数据放在一个由x,y 定位的方框内”这单一图形基础之上的。

乍一看上去,把这么多的用户接口组件绘制出来是非常低效的方法。但是,允许不同的像素数据编码方式,使得我们在处理不同的参数(如:网络带宽,客户端的绘制速度,服务器处理速度)有了很大程度的灵活性。

通过矩形的序列来完成帧缓存的更新。一次更新代表着从一个可用帧缓存状态转换到另一个可用,因此有点和视频的桢类似。尽管矩形的更新一般是分开的,但是并不是必须的。

显示协议的更新部分是由客户端通过命令驱动的。也就是说,更新只是在服务器端响应客户端的请求时发生的。这样就让协议更新质量是可变的。客户端/网络越慢,更新速度也就越慢。对于一些应用来说,相同区域的更新是连续不断的。如果用一个慢的客户端,那么帧缓存的缓存状态是可以被忽略的。这样也可以减少对客户端网络速度和绘制速度的要求。

输入协议

输入协议是基于标准工作站的键盘和鼠标等设备的连接协议。

输入事件就是通过把客户端的输入发送到服务器端。

这些输入事件也可以通过非标准的I /O 设备来综合。

例如,手写笔引擎可能产生一个键盘事件。

像素数据的表示

初始的交互涉及到RFB客户端和服务器之间传输像素数据格式和编码方式的协调。这种协调被设计的让客户端的工作尽量简单。而设计的底线是:服务器必须按照客户端的要求格式来提供像素数据。如果客户端可以同样的处理多种数据格式或编码格式,那么一般会选择服务器端易于生成的格式。

像素格式涉及如何通过像素值来实现不同颜色的重现。最常用的一般像素格式是24 位或16 位的“真彩色”,它通过位来直接实现像素值到红、绿、蓝亮度的转换。8 位“颜色映射”可以任意映射像素值到RGB亮度的转换。

编码指一个矩形的像素数据如何通过网线传输。每个像素数据的矩形都加上了一个头,给定矩形在屏幕上的X、Y坐标、矩形的宽和高,以及指定的编码类型。而后数据本身就是采用这种特定的编码方式。

数据本身遵循特定的编码。目前的编码方式主要有Raw、CopyRect、RRE、Hextile 和ZRLE.在实际应用中我们一般使用ZRLE、Hextile 和CopyRect,因为它们提供了典型桌面的最好压缩。其他可能的编码方式还包括,用于静态图片的JPEG和用于动态图像有效传输的MPEG。

协议可以通过增加新的编码方式来进行扩展。

协议扩展

协议可以通过以下方式进行扩展:

新的编码方式

一种新的协议可以通过与现存的客户端和服务端进行相关兼容的添加。因为现存的服务器将会忽略它们所不支持的新编码方式。所以客户端通过新的编码方式进行请求也就不会有结果返回。

伪编码方式

除了真正的编码方式,客户端也可以请求“伪编码”通告服务器,它支持某一协议的扩展。服务器如果不支持这种扩展,那么它将忽略。值得注意的是:客户端必须先假设服务器端不支持这种扩展,直到它获得服务器端支持的确认。

新的安全方式

添加一个新型的安全方式会带来无限的灵活性,它通过修改协议的一些行为,但是并没有牺牲现存客户端和服务器端的兼容性。客户端和服务器端可以通过协议好的安全方式进行交流,当然并不一定与RFB协议类似。

无论如何你都不应使用不同的版本号。

RFB协议的版本是由RealVNC公司来制定的。如果你使用一个不同的协议版本可能与RFB/VNC不兼容,要保证协议的兼容性,请联系RealVNC公司。这样会减少在编码方式和安全类型上的冲突。

协议消息

RFB协议可以进行可靠的传输,如字节流或基于消息的。和大多数协议一样,它也是通过TCP /IP协议簇连接。协议由三步完成连接。首先是握手报文,目的是对协议版本和加密方式进行协商。第二步是初始化报文,主要用于客户和服务器的初始化消息。最后就是正常协议的交互,客户端可以按需发送消息,然后可以获得服务器的回复。

所有的消息以消息类型开始,接下来是特定的消息数据。

协议消息描述的基本类型有:U8、U16、U32、S8、S16、S32。

U表示无符号整数,S表示有符号整数。

所有字节整数(除了像素值本身)遵从big endian顺序。

big endian或者little endian跟cpu有关,从而影响整数在内存中的排列顺序。big endian是高字节在前,little endian是低字节在前,网络字节序一般是big-endian。

PIXEL代表一个像素值bytesPerPixel字节,8XbytesPerPixel = bits-per-pixel

RealVNC 分为客户端和服务端

客户端名为 VNCViewer

服务端名为 VNC

客户端连接到服务器端

这时候 服务端  会给 客户端发送当前提供最大支持的版本号。

  1. //初始化RFB协议
  2. void SConnection::initialiseProtocol()
  3. {
  4. cp.writeVersion(os);
  5. state_ = RFBSTATE_PROTOCOL_VERSION;
  6. }
  7. //写入版本
  8. void ConnParams::writeVersion(rdr::OutStream* os)
  9. {
  10. char str[13];
  11. sprintf(str, "RFB %03d.%03d\n", majorVersion, minorVersion);
  12. os->writeBytes(str, 12);
  13. os->flush();
  14. }

这里对应的发送的内容如下

00000000    52 46 42 20 30 30 33 2E 30 30 38 0A                 RFB 003.008.

发送数据长度  12

发送数据内容  RFB 003.008

此时VNC 的数据处理状态为 RFBSTATE_PROTOCOL_VERSION

2.viewer收到来自vnc的版本号信息

因为viewer在init()函数中已经设置数据处理状态为 RFBSTATE_PROTOCOL_VERSION

  1. void CConnection::initialiseProtocol()
  2. {
  3. state_ = RFBSTATE_PROTOCOL_VERSION;
  4. }
  1. void CConnection::processMsg()
  2. {
  3. switch (state_)
  4. {
  5. case RFBSTATE_PROTOCOL_VERSION:
  6. processVersionMsg();
  7. break;
  8. case RFBSTATE_SECURITY_TYPES:
  9. processSecurityTypesMsg();
  10. break;
  11. case RFBSTATE_SECURITY:
  12. processSecurityMsg();
  13. break;
  14. case RFBSTATE_SECURITY_RESULT:
  15. processSecurityResultMsg();
  16. break;
  17. case RFBSTATE_INITIALISATION:
  18. processInitMsg();
  19. break;
  20. case RFBSTATE_NORMAL:
  21. reader_->readMsg();
  22. break;
  23. case RFBSTATE_UNINITIALISED:
  24. throw Exception("CConnection::processMsg: not initialised yet?");
  25. default:
  26. throw Exception("CConnection::processMsg: invalid state");
  27. }
  28. }
  1. 进入 processVersionMsg()函数
  2. //处理 RFBSTATE_PROTOCOL_VERSION
  3. void CConnection::processVersionMsg()
  4. {
  5. vlog.debug("reading protocol version");
  6. bool done;
  7.   //这里读取服务器版本,如果读取失败 数据处理状态切换为  RFBSTATE_INVALID
  8. if (!cp.readVersion(is, &done))
  9. {
  10. state_ = RFBSTATE_INVALID;
  11. throw Exception("reading version failed: not an RFB server?");
  12. }
  13.   //读取失败  返回
  14. if (!done)
  15. return;
  16. vlog.info("Server supports RFB protocol version %d.%d",
  17. cp.majorVersion, cp.minorVersion);
  18. // The only official RFB protocol versions are currently 3.3, 3.7 and 3.8
  19.   //如果RFB版本号低于 3.3 不支持。数据处理状态切换为 RFB_INVALID  抛出异常
  20. if (cp.beforeVersion(3, 3))
  21. {
  22. char msg[256];
  23. sprintf(msg, "Server gave unsupported RFB protocol version %d.%d",
  24. cp.majorVersion, cp.minorVersion);
  25. vlog.error(msg);
  26. state_ = RFBSTATE_INVALID;
  27. throw Exception(msg);
  28. }
  29. else if (useProtocol3_3 || cp.beforeVersion(3, 7))
  30. {
  31. cp.setVersion(3, 3);
  32. }
  33. else if (cp.afterVersion(3, 8))
  34. {
  35. cp.setVersion(3, 8);
  36. }
  37.   //这里写入版本号信息
  38. cp.writeVersion(os);
  39.   //Viewer切换数据处理状态 RFBSTATE_SECURITY_TYPES
  40. state_ = RFBSTATE_SECURITY_TYPES;
  41. vlog.info("Using RFB protocol version %d.%d",
  42. cp.majorVersion, cp.minorVersion);
  43. }
  44. //向服务器发送版本号
  45. void ConnParams::writeVersion(rdr::OutStream* os)
  46. {
  47. char str[13];
  48. sprintf(str, "RFB %03d.%03d\n", majorVersion, minorVersion);
  49. os->writeBytes(str, 12);
  50. os->flush();
  51. }

这个函数发送的内容

00000000    52 46 42 20 30 30 33 2E 30 30 38 0A                 RFB 003.008.

长度为  12 个字节

  这时候 VNCViewer的数据处理状态为 RFBSTATE_SECURITY_TYPES

3.vnc收到来自viewer的版本信息

因为VNC当前的数据处理状态是  RFBSTATE_PROTOCOL_VERSION

所以会进入processVersionMsg()函数处理viewer发送过来的版本信息

  1. //处理 RFBSTATE_PROTOCOL_VERSION
  2. void SConnection::processVersionMsg()
  3. {
  4. vlog.debug("reading protocol version");
  5. bool done;
  6.   //读取来自viewer的版本号
  7. if (!cp.readVersion(is, &done))
  8. {
  9. state_ = RFBSTATE_INVALID;
  10. throw Exception("reading version failed: not an RFB client?");
  11. }
  12. if (!done)
  13. return;
  14. vlog.info("Client needs protocol version %d.%d",cp.majorVersion, cp.minorVersion);
  15.   
  16. if (cp.majorVersion != 3)
  17. {
  18. // unknown protocol version
  19. char msg[256];
  20. sprintf(msg, "Error: client needs protocol version %d.%d, server has %d.%d",cp.majorVersion, cp.minorVersion,defaultMajorVersion, defaultMinorVersion);
  21. throwConnFailedException(msg);
  22. }
  23. if (cp.minorVersion != 3 && cp.minorVersion != 7 && cp.minorVersion != 8)
  24. {
  25. vlog.error("Client uses unofficial protocol version %d.%d",cp.majorVersion, cp.minorVersion);
  26. if (cp.minorVersion >= 8)
  27. cp.minorVersion = 8;
  28. else if (cp.minorVersion == 7)
  29. cp.minorVersion = 7;
  30. else
  31. cp.minorVersion = 3;
  32. vlog.error("Assuming compatibility with version %d.%d",cp.majorVersion, cp.minorVersion);
  33. }
  34. versionReceived();
  35.   
  36.   //这里是获取到加密类型
  37. std::list<rdr::U8> secTypes;
  38. std::list<rdr::U8>::iterator i;
  39. securityFactory->getSecTypes(&secTypes, reverseConnection);
  40. if (cp.isVersion(3, 3))
  41. {
  42. // cope with legacy 3.3 client only if "no authentication" or "vnc
  43. // authentication" is supported.
  44. for (i = secTypes.begin(); i != secTypes.end(); i++)
  45. {
  46. if (*i == secTypeNone || *i == secTypeVncAuth)
  47. break;
  48. }
  49. if (i == secTypes.end())
  50. {
  51. char msg[256];
  52. sprintf(msg, "No supported security type for %d.%d client",
  53. cp.majorVersion, cp.minorVersion);
  54. throwConnFailedException(msg);
  55. }
  56. os->writeU32(*i);
  57. if (*i == secTypeNone)
  58. os->flush();
  59. state_ = RFBSTATE_SECURITY;
  60. security = securityFactory->getSSecurity(*i, reverseConnection);
  61. processSecurityMsg();
  62. return;
  63. }
  64. // list supported security types for >=3.7 clients
  65. if (secTypes.empty())
  66. throwConnFailedException("No supported security types");
  67. os->writeU8(secTypes.size());
  68. for (i = secTypes.begin(); i != secTypes.end(); i++)
  69. os->writeU8(*i);
  70. os->flush();
  71. state_ = RFBSTATE_SECURITY_TYPE;
  72. }
  73. 这个函数主要是
  74. a.验证RFB版本号
  75. b.获取到加密类型
  76. c.发送加密类型到VncViewer
  77. d.设置数据处理标志为  RFBSTATE_SECURITY_TYPE

4.vncViewer状态为RFBSTATE_SECURITY_TYPES

收到vnc发送过来的加密类型后执行

processSecurityTypesMsg()

  1. void CConnection::processSecurityTypesMsg()
  2. {
  3. vlog.debug("processing security types message");
  4. int secType = secTypeInvalid;
  5. if (cp.isVersion(3, 3))
  6. {
  7. // legacy 3.3 server may only offer "vnc authentication" or "none"
  8. secType = is->readU32();
  9. if (secType == secTypeInvalid)
  10. {
  11. throwConnFailedException();
  12. }
  13. else if (secType == secTypeNone || secType == secTypeVncAuth)
  14. {
  15. int j;
  16. for (j = 0; j < nSecTypes; j++)
  17. if (secTypes[j] == secType)
  18. break;
  19. if (j == nSecTypes)
  20. secType = secTypeInvalid;
  21. }
  22. else
  23. {
  24. vlog.error("Unknown 3.3 security type %d", secType);
  25. throw Exception("Unknown 3.3 security type");
  26. }
  27. }
  28. else
  29. {
  30. // >=3.7 server will offer us a list
  31. int nServerSecTypes = is->readU8();
  32. if (nServerSecTypes == 0)
  33. throwConnFailedException();
  34. int secTypePos = nSecTypes;
  35. for (int i = 0; i < nServerSecTypes; i++)
  36. {
  37. rdr::U8 serverSecType = is->readU8();
  38. vlog.debug("Server offers security type %s(%d)",
  39. secTypeName(serverSecType), serverSecType);
  40. // If we haven't already chosen a secType, try this one
  41. // If we are using the client's preference for types,
  42. // we keep trying types, to find the one that matches and
  43. // which appears first in the client's list of supported types.
  44. if (secType == secTypeInvalid || clientSecTypeOrder)
  45. {
  46. for (int j = 0; j < nSecTypes; j++)
  47. {
  48. if (secTypes[j] == serverSecType && j < secTypePos)
  49. {
  50. secType = secTypes[j];
  51. secTypePos = j;
  52. break;
  53. }
  54. }
  55. // NB: Continue reading the remaining server secTypes, but ignore them
  56. }
  57. }
  58. // Inform the server of our decision
  59. if (secType != secTypeInvalid)
  60. {
  61. os->writeU8(secType);
  62. os->flush();
  63. vlog.debug("Choosing security type %s(%d)", secTypeName(secType), secType);
  64. }
  65. }
  66. if (secType == secTypeInvalid)
  67. {
  68. state_ = RFBSTATE_INVALID;
  69. vlog.error("No matching security types");
  70. throw Exception("No matching security types");
  71. }
  72. state_ = RFBSTATE_SECURITY;
  73.   //因为这里没有开启认证,所以security返回的是一个CSecurityNone类
  74. security = getCSecurity(secType);
  75.   //调用getCSecurity函数进行下一步的认证工作
  76. processSecurityMsg();
  77. }
  78. CSecurity* CConn::getCSecurity(int secType)
  79. {
  80. switch (secType)
  81. {
  82. case secTypeNone:
  83. return new CSecurityNone();
  84. case secTypeVncAuth:
  85. return new CSecurityVncAuth(this);
  86. default:
  87. throw Exception("Unsupported secType?");
  88. }
  89. }

这个函数

a.获取加密类型

b.切换数据处理状态为 RFBSTATE_SECURITY

C.生成用例处理加密的类

d.调用processSecurityMsg()函数继续处理加密

  1. void CConnection::processSecurityMsg()
  2. {
  3. vlog.debug("processing security message");
  4. if (security->processMsg(this))
  5. {
  6. state_ = RFBSTATE_SECURITY_RESULT;
  7. processSecurityResultMsg();
  8. }
  9. }

因为我这里没有开启认证,所以processMsg()函数直接返回TRUE就通过验证。

如果开启了VNC验证,会执行这个函数

  1. bool CSecurityVncAuth::processMsg(CConnection* cc)
  2. {
  3. rdr::InStream* is  = cc->getInStream();
  4. rdr::OutStream* os = cc->getOutStream();
  5. // Read the challenge & obtain the user's password
  6. rdr::U8 challenge[vncAuthChallengeSize];
  7. is->readBytes(challenge, vncAuthChallengeSize);
  8. PlainPasswd passwd;
  9. upg->getUserPasswd(0, &passwd.buf);
  10. // Calculate the correct response
  11. rdr::U8 key[8];
  12. int pwdLen = strlen(passwd.buf);
  13. for (int i = 0; i < 8; i++)
  14. key[i] = i < pwdLen ? passwd.buf[i] : 0;
  15. deskey(key, EN0);
  16. for (int j = 0; j < vncAuthChallengeSize; j += 8)
  17. des(challenge + j, challenge + j);
  18. // Return the response to the server
  19. os->writeBytes(challenge, vncAuthChallengeSize);
  20. os->flush();
  21. return true;
  22. }

认证通过后数据处理状态切换为

RFBSTATE_SECURITY_RESULT

继续执行processSecurityResultMsg()函数

  1. void CConnection::processSecurityResultMsg()
  2. {
  3. vlog.debug("processing security result message");
  4. int result;
  5. if (cp.beforeVersion(3, 8) && security->getType() == secTypeNone)
  6. {
  7. result = secResultOK;
  8. }
  9. else
  10. {
  11. if (!is->checkNoWait(1))
  12. return;
  13. result = is->readU32();
  14. }
  15. switch (result)
  16. {
  17. case secResultOK:
  18. securityCompleted();
  19. return;
  20. case secResultFailed:
  21. vlog.debug("auth failed");
  22. break;
  23. case secResultTooMany:
  24. vlog.debug("auth failed - too many tries");
  25. break;
  26. default:
  27. throw Exception("Unknown security result from server");
  28. }
  29. CharArray reason;
  30. if (cp.beforeVersion(3, 8))
  31. reason.buf = strDup("Authentication failure");
  32. else
  33. reason.buf = is->readString();
  34. state_ = RFBSTATE_INVALID;
  35. throw AuthFailureException(reason.buf);
  36. }

验证通过

执行securityCompleted()函数

  1. void CConnection::securityCompleted()
  2. {
  3. state_ = RFBSTATE_INITIALISATION;
  4. reader_ = new CMsgReaderV3(this, is);
  5. writer_ = new CMsgWriterV3(&cp, os);
  6. vlog.debug("Authentication success!");
  7. authSuccess();
  8. writer_->writeClientInit(shared);
  9. }

设置 vncViewer的数据处理状态为 RFBSTATE_INITIALISATION

创建CMsgReaderV3类用于读取

创建CMsgWriterV3类用于写入

调用 writeClientInit 函数

  1. void CMsgWriterV3::writeClientInit(bool shared)
  2. {
  3.   //是否分享屏幕 默认为false
  4. os->writeU8(shared);
  5. endMsg();
  6. }
  7. void CMsgWriterV3::endMsg()
  8. {
  9. os->flush();
  10. }

这里,viewer数据处理状态为 RFBSTATE_INITIALISATION

所以这个函数发送了两次数据

第一次是发送加密类型   os->writeU8(secType)

第二次是发送的屏幕是否共享   os->writeU8(shared);

发送数据内容

5.  vnc当前状态为 RFBSTATE_SECURITY_TYPE

调用processSecurityTypeMsg函数来处理Viewer的数据

  1. //RFBSTATE_SECURITY_TYPE
  2. void SConnection::processSecurityTypeMsg()
  3. {
  4. vlog.debug("processing security type message");
  5.   //第一次是读取 加密类型
  6. int secType = is->readU8();
  7. // Verify that the requested security type should be offered
  8. std::list<rdr::U8> secTypes;
  9. std::list<rdr::U8>::iterator i;
  10. securityFactory->getSecTypes(&secTypes, reverseConnection);
  11. for (i = secTypes.begin(); i != secTypes.end(); i++)
  12. if (*i == secType)
  13. break;
  14. if (i == secTypes.end())
  15. throw Exception("Requested security type not available");
  16. vlog.info("Client requests security type %s(%d)",secTypeName(secType), secType);
  17. try
  18. {
  19. state_ = RFBSTATE_SECURITY;
  20.   //同viewer一样,这里根据加密类型生成对应的加密处理类  如果有加密就是       CSecurityVncAuth 类 这里因为没有启动加密,所以生成的是SSecurityNone类
  21. security = securityFactory->getSSecurity(secType, reverseConnection);
  22. }
  23. catch (rdr::Exception& e)
  24. {
  25. throwConnFailedException(e.str());
  26. }
  27.   
  28. processSecurityMsg();
  29. }

processSecurityMsg()继续处理数据

  1. void SConnection::processSecurityMsg()
  2. {
  3. vlog.debug("processing security message");
  4. try
  5. {
  6.   //对于未加密的认证类处理processMsg函数,总是返回true
  7. bool done = security->processMsg(this);
  8. if (done)
  9. {
  10. state_ = RFBSTATE_QUERYING;
  11. queryConnection(security->getUserName());
  12. }
  13. }
  14. catch (AuthFailureException& e)
  15. {
  16. vlog.error("AuthFailureException: %s", e.str());
  17. os->writeU32(secResultFailed);
  18. if (!cp.beforeVersion(3, 8)) // 3.8 onwards have failure message
  19. os->writeString(e.str());
  20. os->flush();
  21. throw;
  22. }
  23. }

这个函数切换vnc数据处理状态为 RFBSTATE_QUERYING

执行queryConnection函数

  1. void SConnection::queryConnection(const char* userName)
  2. {
  3. approveConnection(true);
  4. }

这个函数会进入 approveConnection()函数 ,传入参数为true 即是接受连接

  1. void SConnection::approveConnection(bool accept, const char* reason)
  2. {
  3. if (state_ != RFBSTATE_QUERYING)
  4. throw Exception("SConnection::approveConnection: invalid state");
  5. if (!reason)
  6. reason = "Authentication failure";
  7. if (!cp.beforeVersion(3, 8) || security->getType() != secTypeNone)
  8. {
  9. if (accept)
  10. {
  11. os->writeU32(secResultOK);
  12. }
  13. else
  14. {
  15. os->writeU32(secResultFailed);
  16. if (!cp.beforeVersion(3, 8)) // 3.8 onwards have failure message
  17. os->writeString(reason);
  18. }
  19. os->flush();
  20. }
  21. if (accept)
  22. {
  23. state_ = RFBSTATE_INITIALISATION;
  24. reader_ = new SMsgReaderV3(this, is);
  25. writer_ = new SMsgWriterV3(&cp, os);
  26. authSuccess();
  27. }
  28. else
  29. {
  30. state_ = RFBSTATE_INVALID;
  31. throw AuthFailureException(reason);
  32. }
  33. }

如果RFB版本不低于3.8这个函数写入 secResultOK 也就是 0

否则写入错误信息

切换数据处理状态为RFBSTATE_INITIALISATION

然后调用一个空函数

  1. void SConnection::authSuccess()
  2. {
  3.   
  4. }

所以在这一次处理中 先读取了加密类型。但是对于发送过来的屏幕是否共享这个值是否有读取,我还没有找到相关的代码。

接着发送 secResultOK 的值 0

Vnc状态 RFBSTATE_INITIALISATION

http://blog.csdn.net/witch_soya/article/details/27214405

开源远程控制RealVNC源代码中的通讯协议RFB(远程帧缓冲)(转)的更多相关文章

  1. MQTT是IBM开发的一个即时通讯协议,构建于TCP/IP协议上,是物联网IoT的订阅协议,借助消息推送功能,可以更好地实现远程控制

    最近一直做物联网方面的开发,以下内容关于使用MQTT过程中遇到问题的记录以及需要掌握的机制原理,主要讲解理论. 背景 MQTT是IBM开发的一个即时通讯协议.MQTT构建于TCP/IP协议上,面向M2 ...

  2. ISO7816通讯协议在工控主板EM9160中的实现方案

    在新的国家电网智能终端相关标准中,规定了通过专门的加密芯片来保证设备数据安全性的方法,而设备主控单元与加密芯片采用了广泛应用的ISO7816通讯协议.工控主板EM9160为了适应这一新的技术需求,对其 ...

  3. CPU卡中T=0通讯协议的分析与实现

    IC卡的应用越来越广泛,从存储卡到逻辑加密卡,目前CPU卡已经逐渐在应用中占据主导地位.CPU卡根据通讯协议可分为两种:接触式和非接触式.接触式CPU卡主要采用两种通讯协议:T=0和T=1通讯协议.T ...

  4. html文件在head标签中引入js地址和直接写js代码,所用时间是不同的,因为引入js地址,文件加载的时候需要通过通讯协议去解析地址,读取外部文件

    html文件在head标签中引入js地址和直接写js代码,所用时间是不同的,因为引入js地址,文件加载的时候需要通过通讯协议去解析地址,读取外部文件

  5. TwinCAT 3中基于UDP协议通讯的C++实现

    因为项目需要,学习了TwinCAT3中使用UDP协议进行通讯的基本知识.这个做个简单的笔记,方便以后查询. 1 概述 倍福为了实现从实时环境中直接访问网卡(network cards)专门提供了一个函 ...

  6. HslCommunication库的二次协议扩展,适配第三方通讯协议开发,基础框架支持长短连接模式

    本文将使用一个gitHub开源的项目来扩展实现二次协议的开发,该项目已经搭建好了基础层架构,并实现了三菱,西门子,欧姆龙,MODBUS-TCP的通讯示例,也可以参照这些示例开发其他的通讯协议,并Pul ...

  7. 几种通讯协议的比较RMI > Httpinvoker >= Hessian >> Burlap >> web service

    一.综述本文比较了RMI,Hessian,Burlap,Httpinvoker,web service等5种通讯协议的在不同的数据结构和不同数据量时的传输性能.RMI是java语言本身提供的远程通讯协 ...

  8. Mavlink - 无人机通讯协议

    http://qgroundcontrol.org/mavlink/start mavlink协议介绍https://pixhawk.ethz.ch/mavlink/ 消息简介 MAVLink简介 M ...

  9. 几种通讯协议的比较RMI > Httpinvoker >= Hessian >> Burlap >> web service (转)

    一.综述 本文比较了RMI,Hessian,Burlap,Httpinvoker,web service等5种通讯协议的在不同的数据结构和不同数据量时的传输性能.RMI是java语言本身提供的通讯协议 ...

随机推荐

  1. 关于Relay Log无法自己主动删除的问题(Neither --relay-log nor --relay-log-index were used)

    今天查看mysql err日志.发现mysql重新启动时总会有例如以下日志出现: [Warning] Neither --relay-log nor --relay-log-index were us ...

  2. windows phone (21) Grid元素的Background和Clip

    原文:windows phone (21) Grid元素的Background和Clip Grid是唯一可以在内部定制单元格的panel类,我们可以在grid中定制单元格,然后通过grid.row和g ...

  3. javascript - 浏览TOM大叔博客的学习笔记

    part1 ---------------------------------------------------------------------------------------------- ...

  4. C该程序生成一个唯一的序列号

    在实际的软件开发项目,通常,它包括产生一唯一的序列号.在本文中,一个切实可行的方案,例如,它引入了一个唯一的序列号生成过程. 本文生成的序列号的样式为:MMDDHHMINSS_XXXXXX. 程序例如 ...

  5. iOS 删除黑色边框线导航栏&amp;删除搜索框的阴影边界线和中黑色文本输入框 - 解

    删除黑色边框线导航栏 in viewDidload: [self.navigationController.navigationBar setBackgroundImage:[[UIImage all ...

  6. 【安卓笔记】高速的发展设置界面-----PreferenceActivity

    通常app都会有一个设置界面,例如以下: 通常做法是自定义布局,然后在代码里面加入响应函数,并将结果保存到Sharedpreferences中. android给我们提供了PreferenceActi ...

  7. Linux安装jdk 8和环境变量配置

    1.下载jdk 地址:http://www.oracle.com/technetwork/java/javase/downloads/jdk8-downloads-2133151.html 2.将刚刚 ...

  8. 于Unity3D动态创建对象和创建Prefab三种方式的原型对象

    于Unity3D动态创建对象和创建Prefab三种方式的原型对象 u3d在动态创建的对象,需要使用prefab 和创建时 MonoBehaviour.Instantiate( GameObject o ...

  9. 国外代理server

    这里有几个国外的代理server 另外在网上能够找到很多这种 不能用的时候就在网上搜搜 稳定代理server 有非常多的 IP port 显示地址 24.245.58.130:32167 美国 新泽西 ...

  10. 【ThinkingInC++】53、构造函数,析构函数,全局变量

    /** * 图书:[ThinkingInC++] * 特征:构造函数,析构函数,全局变量 * 时刻:2014年9一个月17日本18:07:43 * 笔者:cutter_point */ #includ ...