+一般在进行Socket编程或者网络访问的时候,首先需要确认对方网络服务已经开启,且需要知道对方的域名或地址以及端口,然后才可以进行进一步操作。在互联网上好点,网络服务方一般常年开启,且一般IP地址是固定的,另由于DNS服务的存在,只要记住对方的域名便可以。但是在局域网,设备不一定连在上面,即使连上了,服务也不一定开了,每当设备连接到局域网的时候,IP地址一般都是动态分配的,所以情况变的复杂。Bonjour的存在便是苹果为了解决局域网设备间连接麻烦的问题。
|
| |
5 |
+直白的说Bonjour就是是一种协议,使得局域网中的计算机可以方便的发布服务,发现服务和连接服务,达到零配置([Zeroconf][link-zeroconf])的目的。
|
| |
6 |
+[link-zeroconf]:http://zeroconf.org
|
| |
7 |
+
|
| |
8 |
+Zeroconf Working Group指出要实现零配置网络服务的3个要求:
|
| |
9 |
+
|
| |
10 |
+>+ IP地址
|
| |
11 |
+>+ **名字** 到 **IP地址** 的转换(即使没有DNS服务器的情况下)
|
| |
12 |
+>+ 发现网络中的服务
|
| |
13 |
+
|
| |
14 |
+对于第一个要求相关系统和设备可以直接支持的,如动态IP地址分配。
|
| |
15 |
+第二个要求则可以通过多播(UDP协议向局域网内一组机器发送数据)的方式发送 类似DNS查询的请求,开启着的网络服务收到之后便作出回应,告知自己的名字。
|
| |
16 |
+第三个要求则通过DNS-SD来实现
|
| |
17 |
+
|
| |
18 |
+Bonjour一般的工作模式便是:在同一个局域网中,一方开启服务,通过Bonjour接口将这个服务发布,服务搜索方在服务列表中便可以看到对应的设备的名字,选择设备便可以进行连接了。整个过程无需事先知道服务发布方的IP地址和端口号。
|
| |
19 |
+我们常用的软件如iTunes的共享,keynote的remote控制或者支持Bonjour协议的打印机都可以看到Bonjour的影子。
|
| |
20 |
+
|
| |
21 |
+
|
| |
22 |
+###二.Bonjour的实现及使用
|
| |
23 |
+从上面的描述可以看出,Bonjour的用途便是在局域网内发布服务和搜索服务。
|
| |
24 |
+下面从实现层面讲解Bonjour。
|
| |
25 |
+
|
| |
26 |
+|层次|名称|
|
| |
27 |
+|:----|:----|
|
| |
28 |
+|Foundation|NSNetService/NSNetServiceBroswer|
|
| |
29 |
+|CoreFoundation|CFNetService/CFNetServiceBroswer|
|
| |
30 |
+|Low-Level Socket Based API|dns_sd.h(The DNS Service Discovery API)|
|
| |
31 |
+|Multicast DNS Responder|mDNSResponder (开源项目)|
|
| |
32 |
+
|
| |
33 |
+一般情况下我们使用Foundation这一层接口就可以了,也是最方便的。
|
| |
34 |
+当然服务方在发布服务之前你得先启好网络服务,比如listening socket创建好,且开始侦听某个端口,关于socket编程的知识可以查看[Socket编程][link-socket]
|
| |
35 |
+
|
| |
36 |
+[link-socket]:Socket编程.md
|
| |
37 |
+
|
| |
38 |
+**1.发布服务**
|
| |
39 |
+
|
| |
40 |
+ netService = [[[NSNetService alloc] initWithDomain:@""
|
| |
41 |
+ type:@"_test._tcp"
|
| |
42 |
+ name:@""
|
| |
43 |
+ port:port] autorelease];
|
| |
44 |
+ if(netService != nil) {
|
| |
45 |
+ [netService scheduleInRunLoop:[NSRunLoop currentRunLoop]
|
| |
46 |
+ forMode:NSRunLoopCommonModes];
|
| |
47 |
+ netService.delegate = self;
|
| |
48 |
+ [netService publish];
|
| |
49 |
+ }
|
| |
50 |
+
|
| |
51 |
+**2.浏览服务**
|
| |
52 |
+
|
| |
53 |
++ 创建Service Broswer, 需要指定service type和domain,得和发布服务时候的type对应。还得设置delegate,然后实现其delegate方法,以便发现了服务之后进行处理以及对发现的服务进行获取IP地址和端口的结果进行处理。
|
| |
54 |
+
|
| |
55 |
+
|
| |
56 |
+ testServiceBrowser = [[NSNetServiceBrowser alloc] init];
|
| |
57 |
+ testServiceBrowser.delegate = self;
|
| |
58 |
+ [testServiceBrowser searchForServicesOfType:@"_test._tcp" inDomain:@""];
|
| |
59 |
+
|
| |
60 |
+
|
| |
61 |
++ 实现Service Broswer 的delegate方法,处理服务增加或减少的事件
|
| |
62 |
+
|
| |
63 |
+
|
| |
64 |
+ //pragma mark NetServiceBroswer Delegate
|
| |
65 |
+ - (void)netServiceBrowser:(NSNetServiceBrowser*)netServiceBrowser
|
| |
66 |
+ didFindService:(NSNetService*)service
|
| |
67 |
+ moreComing:(BOOL)moreComing {
|
| |
68 |
+ [netServiceArray addObject:service];
|
| |
69 |
+ if (!moreComing) {
|
| |
70 |
+ [serviceTableView reloadData];
|
| |
71 |
+ }
|
| |
72 |
+ }
|
| |
73 |
+
|
| |
74 |
+ - (void)netServiceBrowser:(NSNetServiceBrowser*)netServiceBrowser
|
| |
75 |
+ didRemoveService:(NSNetService*)service
|
| |
76 |
+ moreComing:(BOOL)moreComing {
|
| |
77 |
+ [netServiceArray removeObject:service];
|
| |
78 |
+ if (!moreComing) {
|
| |
79 |
+ [serviceTableView reloadData];
|
| |
80 |
+ }
|
| |
81 |
+ }
|
| |
82 |
+
|
| |
83 |
+
|
| |
84 |
++ 连接服务
|
| |
85 |
+
|
| |
86 |
+上面发现的Net Service是不带IP地址和端口信息的。
|
| |
87 |
+从服务列表中选择一个已经发现的服务,进行Resolve,便可以获取服务的详细信息了。
|
| |
88 |
+
|
| |
89 |
+ - (IBAction)connect:(id)sender{
|
| |
90 |
+ NSUInteger selectedRow = [serviceTableView selectedRow];
|
| |
91 |
+ NSNetService *selectedServiece = [netServiceArray objectAtIndex:selectedRow];
|
| |
92 |
+ selectedServiece.delegate = self;
|
| |
93 |
+ [selectedServiece resolveWithTimeout:5.0];
|
| |
94 |
+ }
|
| |
95 |
+
|
| |
96 |
+Resolve成功
|
| |
97 |
+
|
| |
98 |
+ //NSNetService Delegate
|
| |
99 |
+ - (void)netServiceDidResolveAddress:(NSNetService *)sender{
|
| |
100 |
+ NSLog(@"service ip:%@ port:%d",sender.address,sender.port);
|
| |
101 |
+ if ([sender getInputStream:&inputStream outputStream:&outputStream]) {
|
| |
102 |
+ [outputStream scheduleInRunLoop:[NSRunLoop currentRunLoop]
|
| |
103 |
+ forMode:NSDefaultRunLoopMode];
|
| |
104 |
+ [outputStream open];
|
| |
105 |
+ //发送数据
|
| |
106 |
+ NSData *helloData = [@"Hello" dataUsingEncoding:NSUTF8StringEncoding];
|
| |
107 |
+ [outputStream write:[helloData bytes] maxLength:[helloData length]];
|
| |
108 |
+ }
|
| |
109 |
+ }
|
| |
110 |
+
|
| |
111 |
+上面的代码充分利用了输入输出流进行通信。如果你自己的使用socket的连接也是可以的,因为这个时候已经可以获取了对方的IP地址和端口了。
|