【iOS与EV3混合机器人编程系列之五】iOS_WiFi_EV3_Library 剖析之连接EV3

2024-01-21 19:58

本文主要是介绍【iOS与EV3混合机器人编程系列之五】iOS_WiFi_EV3_Library 剖析之连接EV3,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上一篇文章中,我们讲解了如何用开源代码库CocoaAsyncSocket来实现iOS上的UDP和TCP数据通信。那么在本文中,我们将介绍在CocoaAsyncSocket的基础如何使用UDP和TCP连接EV3的机制。

之所以我们能够通过无线连接EV3,根本原因在于EV3的源代码内建了一套无线连接通信的机制。
这套机制是这样的:
1)EV3在连接到无线网络后,就不断地从3015端口发送UDP数据,数据的格式如下:
Serial-Number: 0016533f0c1e
Port: 5555
Name: EV3
Protocol: EV3
从这个UDP数据中,我们可以获取其ip地址,设备序列号两个重要数据。
2)拥有了ip地址,我们就可以建立TCP连接连接到EV3,端口为5555
3)在连接上TCP后,我们就可以向EV3发送数据了,我们必须先发送一个解锁信息获取控制EV3的权限才能实现对EV3的有效控制,解锁信息格式为:
GET /target?sn=SERIAL_NUMBER VMTP1.0 Protocol: EV3
sn=对应设备的序列号,就是我们从UDP信息中获取的序列号
4)解锁信息发送成功后,EV3会返回一条信息:”Accept:EV340”。如果我们收到了这条信息,就意味着我们已经解锁成功,现在我们就可以发送特定的EV3命令来控制EV3了!!

连接EV3的奥秘就是上面这么几句话了。恭喜你,你已经知道Secret了!那么现在我们要通过具体的程序来实现它。

在代码库中我们建立了一个类EV3WifiManager来管理EV3的连接和数据传输功能,并建立了EV3WifiBrowserViewController的视图控制器来作为EV3连接管理的界面。

为了更好地存储管理EV3设备数据,我们还建立了EV3Device来存储EV3的信息。

关于界面的设计本文就不谈了,仅谈谈实现连接的核心代码。

== Step 1:连接UDP并接收数据 ==
连接方法我们在上一篇文章已经讲了,本文不再谈。
现在主要说明一下数据处理部分:
- (void)udpSocket:(GCDAsyncUdpSocket *)sock didReceiveData:(NSData *)data
      fromAddress:(NSData *)address
withFilterContext:(id)filterContext
{
        // 1
        NSString *msg = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
        if (msg)
        {
        // 2
        NSString *serialNumber = [msg substringWithRange:NSMakeRange(14, 12)];
        
        NSString *host = [GCDAsyncUdpSocket hostFromAddress:address];
        
        EV3Device *device = [self.devices objectForKey:host];

        if (!device && host.length < 20) {
            
// 3
            EV3Device *aDevice = [[EV3Device alloc] initWithSerialNumber:serialNumber address:host tag:self.devices.count isConnected:NO];
            
            
            
            // 4
            dispatch_queue_t tcpSocketQueue = dispatch_queue_create("com.manmanlai.tcpSocketQueue", DISPATCH_QUEUE_CONCURRENT);
            GCDAsyncSocket *tcpSocket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:tcpSocketQueue];
            
            aDevice.tcpSocket = tcpSocket;
            
            
            [self.devices setObject:aDevice forKey:aDevice.address];
            
            
            
            if ([self.delegate respondsToSelector:@selector(updateView)]) {
                dispatch_async(dispatch_get_main_queue(), ^{
                    [self.delegate updateView];
                });
            }
        }
        
        }
        else
        {
        NSLog(@"Error converting received data into UTF-8 String");
        }
        // 5
        [self.udpSocket sendData:data toAddress:address withTimeout:-1 tag:0];
}

1、将获取的转化为字符串
2、通过截取字符串来获取EV3设备的序列号,并通过hostFromAddress:获取特定连接的名称(每一个连接都有一个host名称)作为EV3设备的key来存储EV3的信息。
3、新建一个EV3Device实例来存储设备信息
4、新建一个TCP socket用于连接EV3。
5、将原UDP数据返回给EV3。(这一步省略也没有关系)

== 连接TCP并解锁 ==
TCP的连接这里不讲,不清楚的童鞋还请看上一篇文章。
在实现中,我们在界面上显示了ip地址,点击后开始连接。
一旦连接成功,我们就立即发送解锁数据,并立即接收数据。代码如下:

- (void)connectTCPSocketWithDevice:(EV3Device *)device
{
    
    GCDAsyncSocket *tcpSocket = device.tcpSocket;
    // connnect
    NSError *error = nil;
    if (![tcpSocket connectToHost:device.address
                                onPort:5555
                                 error:&error])
    {
        NSLog(@"Error connecting: %@", error);
        
    } else {
        NSLog(@"Connected");
        // write data
        NSLog(@"writing...");
        NSString *unlockMsg = [NSString stringWithFormat:@"GET /target?sn=%@ VMTP1.0 Protocol: EV3",device.serialNumber];
        NSData *unlockData = [unlockMsg dataUsingEncoding:NSUTF8StringEncoding];
        [tcpSocket writeData:unlockData withTimeout:-1 tag:MESSAGE_UNLOCK];
        
        [tcpSocket readDataWithTimeout:-1 tag:MESSAGE_UNLOCK];
        
    }
    
    
}

读取到数据后,我们就着手进行数据处理,由于这个代码库要支持多机连接,因此我们在每个EV3Device中处理数据。这里大家要注意就是我们从EV3那边接收的任何数据包含端口数据这是在这里进行处理。

- (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
{
    NSString *host = sock.connectedHost;
    EV3Device *device = [self.devices objectForKey:host];
    
    [device handleReceivedData:data withTag:tag];        
}

这里我们只看处理解锁数据的过程:
case MESSAGE_UNLOCK:
        {
            NSString *httpResponse = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
            NSString *response =[httpResponse substringToIndex:12];
            if ([response isEqualToString:@"Accept:EV340"]) {
                self.isConnected = YES;
                NSLog(@"ev3 connected");
                
                dispatch_async(dispatch_get_main_queue(), ^{
                    [self scanPorts];
                    [[NSNotificationCenter defaultCenter] postNotificationName:EV3DeviceConnectedNotification object:self];
                });
            }
                break;
        }

需要注意的是我们发送的每一条数据都有特定的标签以便于区分。对于解锁数据的信息,其对应的标签是MESSAGE_UNLOCK。代码方面需要说明的不是很多,核心就是判断接收到的信息是不是Accept:EV340,其余的代码就是直接发送命令让EV3返回端口数据并且发送notification让界面更新。只要接收到了信息,那么我们就已经成功连接并且可以控制EV3了。

需要说明的是有一些情况局域网中无法发送TCP数据,导致虽然连接上了TCP,但无法发送解锁信息。一般我都是用Android手机做热点。iPhone热点连接问题确实值得大家一起好好研究!

OK,那么大家可以看到,连接EV3并不是一件非常困难的事。只要通过上面几步,我们就能做到。

在下一篇文章中,我们讲介绍如何给EV3发送命令,如何创建命令。敬请期待!

【本文为原创文章,版权所有,转载请注明出处,谢谢!songrotek@qq.com】


这篇关于【iOS与EV3混合机器人编程系列之五】iOS_WiFi_EV3_Library 剖析之连接EV3的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/630663

相关文章

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

90%的人第一步就错了! 顺利登录wifi路由器后台的技巧

《90%的人第一步就错了!顺利登录wifi路由器后台的技巧》登录Wi-Fi路由器,其实就是进入它的后台管理页面,很多朋友不知道该怎么进入路由器后台设置,感兴趣的朋友可以花3分钟了解一下... 你是不是也遇到过这种情况:家里网速突然变慢、想改WiFi密码却不知道从哪进路由器、新装宽带后完全不知道怎么设置?别慌

MySQL的JDBC编程详解

《MySQL的JDBC编程详解》:本文主要介绍MySQL的JDBC编程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、前置知识1. 引入依赖2. 认识 url二、JDBC 操作流程1. JDBC 的写操作2. JDBC 的读操作总结前言本文介绍了mysq

java.sql.SQLTransientConnectionException连接超时异常原因及解决方案

《java.sql.SQLTransientConnectionException连接超时异常原因及解决方案》:本文主要介绍java.sql.SQLTransientConnectionExcep... 目录一、引言二、异常信息分析三、可能的原因3.1 连接池配置不合理3.2 数据库负载过高3.3 连接泄漏

深度剖析SpringBoot日志性能提升的原因与解决

《深度剖析SpringBoot日志性能提升的原因与解决》日志记录本该是辅助工具,却为何成了性能瓶颈,SpringBoot如何用代码彻底破解日志导致的高延迟问题,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言第一章:日志性能陷阱的底层原理1.1 日志级别的“双刃剑”效应1.2 同步日志的“吞吐量杀手”

Mac电脑如何通过 IntelliJ IDEA 远程连接 MySQL

《Mac电脑如何通过IntelliJIDEA远程连接MySQL》本文详解Mac通过IntelliJIDEA远程连接MySQL的步骤,本文通过图文并茂的形式给大家介绍的非常详细,感兴趣的朋友跟... 目录MAC电脑通过 IntelliJ IDEA 远程连接 mysql 的详细教程一、前缀条件确认二、打开 ID

Python异步编程之await与asyncio基本用法详解

《Python异步编程之await与asyncio基本用法详解》在Python中,await和asyncio是异步编程的核心工具,用于高效处理I/O密集型任务(如网络请求、文件读写、数据库操作等),接... 目录一、核心概念二、使用场景三、基本用法1. 定义协程2. 运行协程3. 并发执行多个任务四、关键

AOP编程的基本概念与idea编辑器的配合体验过程

《AOP编程的基本概念与idea编辑器的配合体验过程》文章简要介绍了AOP基础概念,包括Before/Around通知、PointCut切入点、Advice通知体、JoinPoint连接点等,说明它们... 目录BeforeAroundAdvise — 通知PointCut — 切入点Acpect — 切面

Go语言连接MySQL数据库执行基本的增删改查

《Go语言连接MySQL数据库执行基本的增删改查》在后端开发中,MySQL是最常用的关系型数据库之一,本文主要为大家详细介绍了如何使用Go连接MySQL数据库并执行基本的增删改查吧... 目录Go语言连接mysql数据库准备工作安装 MySQL 驱动代码实现运行结果注意事项Go语言执行基本的增删改查准备工作

python连接sqlite3简单用法完整例子

《python连接sqlite3简单用法完整例子》SQLite3是一个内置的Python模块,可以通过Python的标准库轻松地使用,无需进行额外安装和配置,:本文主要介绍python连接sqli... 目录1. 连接到数据库2. 创建游标对象3. 创建表4. 插入数据5. 查询数据6. 更新数据7. 删除