前面一篇博客里面已经介绍过SSDP协议原理,本篇博客将实现实现Android上的SSDP协议。

关键技术分析:1、发送广播;须要发送送广播,所以须要使用MulticastSocket、SocketAddress、InetAddress,须要掌握。

2、SSDP数据报格式;标准的SSDP Server解析的时候对于分段的字段选用的特征码是"\r\n",须要特别注意。

3、訪问权限;须要互联网,要在Mainfest中加入�联网的相关权限。

下面是我的源代码:

1、SSDPConstants.java

public class SSDPConstants {

/* New line definition */

public static final String NEWLINE = "\r\n";

public static final String ADDRESS = "239.255.255.250";

public static final int PORT = 1900;

public static final String SL_MSEARCH = "M-SEARCH * HTTP/1.1";

public static final String SL_OK = "HTTP/1.1 200 OK";

public static final String ST_Product = "ST:urn:schemas-upnp-org:device:Server:1";

public static final String Found = "ST=urn:schemas-upnp-org:device:";

public static final String Root = "ST:urn:schemas-upnp-org:device:DZBA_HomeDP:1";

}

2、SSDPSearchMsg .java

public class SSDPSearchMsg {

static final String HOST = "Host:" + SSDP.ADDRESS + ":" + SSDP.PORT;

static final String MAN = "Man:\"ssdp:discover\"";

static final String NEWLINE = "\r\n";

int mMX = 5; /* seconds to delay response */

String mST; /* Search target */

public SSDPSearchMsg(String ST) {

mST = ST;

}

public int getmMX() {

return mMX;

}

public void setmMX(int mMX) {

this.mMX = mMX;

}

public String getmST() {

return mST;

}

public void setmST(String mST) {

this.mST = mST;

}

@Override

public String toString() {

StringBuilder content = new StringBuilder();

content.append(SSDP.SL_MSEARCH).append(NEWLINE);

content.append(HOST).append(NEWLINE);

content.append(MAN).append(NEWLINE);

content.append("MX:" + mMX).append(NEWLINE);

content.append(mST).append(NEWLINE);

content.append(NEWLINE);

return content.toString();

}

}

3、SSDPSocket .java

public class SSDPSocket {

SocketAddress mSSDPMulticastGroup;

MulticastSocket mSSDPSocket;

InetAddress broadcastAddress;

public SSDPSocket() throws IOException {

mSSDPSocket = new MulticastSocket(58000); // Bind some random port for receiving datagram

broadcastAddress = InetAddress.getByName(SSDPConstants.ADDRESS);

mSSDPSocket.joinGroup(broadcastAddress);

}

/* Used to send SSDP packet */

public void send(String data) throws IOException {

DatagramPacket dp = new DatagramPacket(data.getBytes(), data.length(), broadcastAddress, SSDPConstants.PORT);

mSSDPSocket.send(dp);

}

/* Used to receive SSDP packet */

public DatagramPacket receive() throws IOException {

byte[] buf = new byte[1024];

DatagramPacket dp = new DatagramPacket(buf, buf.length);

mSSDPSocket.receive(dp);

return dp;

}

public void close() {

if (mSSDPSocket != null) {

mSSDPSocket.close();

}

}

}

4、SSDP .java

public class SSDP {

/* New line definition */

public static final String NEWLINE = "\r\n";

public static final String ADDRESS = "239.255.255.250";

public static final int PORT = 1900;

public static final String ST = "ST";

public static final String LOCATION = "LOCATION";

public static final String NT = "NT";

public static final String NTS = "NTS";

/* Definitions of start line */

public static final String SL_NOTIFY = "NOTIFY * HTTP/1.1";

public static final String SL_MSEARCH = "M-SEARCH * HTTP/1.1";

public static final String SL_OK = "HTTP/1.1 200 OK";

@SuppressWarnings("resource")

public static String parseHeaderValue(String content, String headerName) {

Scanner s = new Scanner(content);

s.nextLine(); // Skip the start line

while (s.hasNextLine()) {

String line = s.nextLine();

int index = line.indexOf(':');

String header = line.substring(0, index);

if (headerName.equalsIgnoreCase(header.trim())) {

return line.substring(index + 1).trim();

}

}

return null;

}

public static String parseHeaderValue(DatagramPacket dp, String headerName) {

return parseHeaderValue(new String(dp.getData()), headerName);

}

@SuppressWarnings("resource")

public static String parseStartLine(String content) {

Scanner s = new Scanner(content);

return s.nextLine();

}

public static String parseStartLine(DatagramPacket dp) {

return parseStartLine(new String(dp.getData()));

}

}

5、MainActivity .java

public class MainActivity extends Activity implements OnClickListener {

protected void onCreate(Bundle savedInstanceState) {

super.onCreate(savedInstanceState);

WifiManager wm = (WifiManager) getSystemService(Context.WIFI_SERVICE);

WifiManager.MulticastLock multicastLock = wm.createMulticastLock("multicastLock");

multicastLock.setReferenceCounted(true);

multicastLock.acquire();

setContentView(R.layout.activity_main);

((Button) this.findViewById(R.id.btnSendSSDPSearch)).setOnClickListener(this);

}

@Override

public void onClick(View v) {

switch (v.getId()) {

case R.id.btnSendSSDPSearch:

new Thread(new Runnable() {

@Override

public void run() {

SendMSearchMessage();

}

}).start();

default:

break;

}

}

private void SendMSearchMessage() {

// SSDPSearchMsg searchContentDirectory = new SSDPSearchMsg(SSDPConstants.ST_ContentDirectory);

// SSDPSearchMsg searchAVTransport = new SSDPSearchMsg(SSDPConstants.ST_AVTransport);

SSDPSearchMsg searchProduct = new SSDPSearchMsg(SSDPConstants.Root);

SSDPSocket sock = null;

try {

sock = new SSDPSocket();

for (int i = 0; i < 2; i++) {

// sock.send(searchContentDirectory.toString());

// sock.send(searchAVTransport.toString());

sock.send(searchProduct.toString());

//String s = "M-SEARCH * HTTP/1.1 \n HOST= 239.255.255.250:1900 \n MAN= \"ssdp:discover\" \n MX: 3 \n ST= upnp:rootdevice";

//sock.send(s);

Log.i("-------------", "发送的数据为:\n" + searchProduct.toString());

}

while (true) {

DatagramPacket dp = sock.receive(); // Here, I only receive the same packets I initially sent above

String c = new String(dp.getData()).trim();

String ip = new String(dp.getAddress().toString()).trim();

Log.i("------------", "接收到的数据为:\n" + c + "數據來源IP:" + ip);

}

} catch (IOException e) {

Log.e("M-SEARCH", e.getMessage());

}

}

}

界面xml非常easy,仅仅有一个button

Mainfest.xml:

package="com.example.ssdp"

android:versionCode="1"

android:versionName="1.0" >

android:minSdkVersion="8"

android:targetSdkVersion="18" />

android:allowBackup="true"

android:icon="@drawable/ic_launcher"

android:label="@string/app_name"

android:theme="@style/AppTheme" >

android:name="com.example.ssdp.MainActivity"

android:label="@string/app_name" >

使用须知:须要有Server端执行,http://download.csdn.net/detail/zhu530548851/7451201下载源代码,该源代码是js的。

将Server放于Linux系统文件夹下,进入test文件夹,运行node server.js就可以。

须要Linux安装有nodejs:sudo apt-get install nodejs

这样在执行Androidclient就能够从Log中看到来自于Server的信息了。

Android源代码在此:http://download.csdn.net/detail/zhu530548851/7451179

个人辛勤劳动成果,如有转载,请注明出处,谢谢!

ssdp java_SSDP协议的Android实现以及使用相关推荐

  1. 基于AOA协议的android USB通信

    摘 要:AOA协议是Google公司推出的用于实现Android设备与外围设备之间USB通信的协议.该协议拓展了Android设备USB接口的功能,为基于Android系统的智能设备应用于数据采集和设 ...

  2. linux虚拟化桌面协议,桌面虚拟化传输协议之android spice

    背景 云计算是目前计算机领域的一个热门领域,桌面虚拟化是其中的一个重要应用,即把桌面系统在服务器端虚拟化,然后通过传输协议传输数据到客户端来实现桌面虚拟化,这样的好处就在于不管用什么设备,只要通过客户 ...

  3. 基于Http协议的Android网络编程

    之前写过一篇Android网络编程<浅谈android网络编程>,随着了解了更多Android知识,我意识到HttpClient已经不推荐使用了,更是在Android 6.0中被废弃了,原 ...

  4. Android与服务器端数据交互(基于SOAP协议整合android+webservice)

    转自:http://www.cnblogs.com/zhangdongzi/archive/2011/04/19/2020688.html 上一节中我们通过http协议,采用HttpClient向服务 ...

  5. 基于XMPP协议的Android即时通信系

    2019独角兽企业重金招聘Python工程师标准>>> 以前做过一个基于XMPP协议的聊天社交软件,总结了一下.发出来. 设计基于开源的XMPP即时通信协议,采用C/S体系结构,通过 ...

  6. 一文弄懂关于证书的一切,ssl协议,android包签名机制。

    所有的概念都基于一个非常重要的基础: rsa 非对称加密算法 : 1. 在加解密上,两个秘钥是对等的 任何一个可以加密,另一个可以用来解密. 2. 用openssl创建一个秘钥,然后可用该秘钥可以生成 ...

  7. Android与服务器端数据交互(基于SAOP协议整合android+webservice)

    上一节中我们通过http协议,采用HttpClient向服务器端action请求数据.当然调用服务器端方法获取数据并不止这一种.WebService也可以为我们提供所需数据, 那么什么是webSer ...

  8. android智能家居使用的协议,基于Android的智能家居APP的设计与实现论文.docx

    基于Android的智能家居APP的设计与实现中文摘要I I 基于 Android 的智能家居 APP 的设计与实现 中文摘要 本课题来源于某 IT 企业"智能家居系统"项目.移动 ...

  9. android 编程klinon,android - 您尚未接受以下SDK组件的许可协议

    android - 您尚未接受以下SDK组件的许可协议 这个问题在这里已有答案: 自动接受所有SDK许可证                                     34个答案 我下载了 ...

  10. android页面跳转时获取地址栏,Android 利用scheme页面内跳转协议进行跳转

    什么是 URL Scheme? android中的scheme是一种页面内跳转协议. 通过定义自己的scheme协议,可以非常方便跳转app中的各个页面: 通过scheme协议,服务器可以定制化告诉A ...

最新文章

  1. R语言KMeans聚类分析确定最优聚类簇数实战:间隙统计Gap Statistic(确定最优聚类簇数)
  2. c语言虚函数是什么,C语言中什么函数不能声明为虚函数?
  3. python文件打开模式rb表示只读模式打开文件_一篇搞懂python文件读写操作(r/r+/rb/w/w+/wb/a/a+/ab)...
  4. 获取spring里的bean
  5. Android 系统(48)---WindowManager.LayoutParams 详解
  6. qt_2_QT5.6+VS2015配置
  7. CitrixVDI新版动手实验手册
  8. mysql改根用户_MySQL数据库中复位根用户密码的方法 | 很文博客
  9. 人工智能python营_AI人工智能训练营
  10. 深度好文| Redis面试全攻略
  11. 频繁模式挖掘 (Frequent pattern mining):01 概念篇 (附例题)
  12. Moment工具库学习
  13. “瓮中捉鳖”——涨停板一般出现在什么时候?
  14. 博士毕业选择回老家县城大专任教!事业编、副教授待遇、外加几十万安家费......
  15. 代码不sao,如何成大佬迎娶白富美!
  16. 公司担心 DDoS 攻击对客户的挫败感和品牌损害
  17. python3.6安装scrapy出错_python3.6安装scrapy
  18. OpenTracing-Java实现的灵魂十问
  19. crc32算法简单理解
  20. excel函数:汉字转全拼_30天中30个Excel函数:01 –精确

热门文章

  1. 【数学建模】数据包络分析法
  2. 传输线理论 1/4波长阻抗变换器的分析匹配
  3. matlab绘制折线图
  4. apache VSF 操作类
  5. Pycharm 远程连接服务器(ssh)运行深度学习代码 | 详细步骤
  6. 二维图像(数组)的fftshift
  7. 二进制像素绘制程序 scratch编程三级
  8. java gd库_[转]gd库的安装
  9. 卫星对地观察模型系统操作说明
  10. keep 虚拟路线修改器_从初音未来到歌王子殿下,尚未VTuber化的虚拟偶像