怎样使用Swift创建命令行脚本(2)
suiw9 2025-01-20 16:18 40 浏览 0 评论
本文由CocoaChina译者自来也大人(博客)翻译,校对BenBeng(博客),欢迎指正。
作者:Ben Snider
原文:Using Swift To Make Command Line Scripts - Part 2
欢迎回来!在上一节中,我们已经讲述了使用Swift语言搭配OptionKit来创建一个简单的命令行脚本。在本节中(最后一节)。我们将整理所有的思路,然后在之前已经创建的基于Apple Foundation的类中的方法来实现我们的BTC(比特币)命令行价格查询器。
回顾第一部分,我们学会了如何:
从命令行直接调用Swift脚本
使用Carthage管理带有Swift脚本的库
使用OptionKit框架来解析命令行参数
在这篇博客中,我们将学习到如何利用上述所有来创建一个Swift脚本查询BTC的价格。同时也将学到怎样使用一个URL从BitStamp API中请求数据,然后解析这些请求回来的JSON数据,最后为用户打印出他们需要的结果信息。当然我们也能够看到,怎么编译这些Swift脚本--为一个可执行的二进制以便节约执行时间。
完成后,我们就有了一个能够打印出最新BTC美元价格的脚本了,你还可以添加参数来指定计算价格的时间区间.这个区间参数,我们可以指定为最新(默认),每小时,以及“vwap”等类型。OptionKit生成的帮助信息也能够告诉我们关于如何调用这些脚本。不出意外,我们应该可以像下面这样运行:
$?./btc.swift 335.45 $?./btc.swift?--interval?hourly 338.27
命令行参数
首先,让我们先定义将如何接受用户输入。使用OptionKit,我们可以很容易的进行设置。基于之前演示的options.swift的修改版本,我将创建一个新的文件btc.swift,文件中包含如下代码:
正如我们所看到的,它与之前的option.swift脚本及其相似。最大的区别在于这我们是从一个方法的返回值中获取到—interval参数的。这样我们封装参数解析逻辑,所以我们就可以专注于实现其他功能。
于此同时,我们还需要花点时间来将从OptionKit得到的interval可选字符串,转换为Swift 枚举类型,便于后面方便使用。我们只需要写如下一些简单的代码就可以来实现:
URL载入
到目前为止,我们已经完成了参数解析,下一步工作就是根据用户选择的interval类型来加载URL。通过BitStamp,实际上是两个具有相同的JSON返回格式的URL。接下来,我们编写一个函数,确定正确的URL并使用Foundation框架中的NSURLSession类来发起一个网络请求。
在实现URL加载之前,我们需要实现调用函数,因此要考虑有哪些数据要传递。
在这里可以看到这些仍然建立在参数解析函数中,虽然开始调用实际的方法来检索和解析价格数据。我选择实现带有一个闭包作为参数的retrievePriceData函数。当网络请求完成之后(无论成功还是失败),闭包将会被调用。传递给闭包的参数是可选类型的,这可以让我们在网络成功或者失败情况下,分别对这两种情况进行处理。也有可能对实际响应返回的数据解析失败,所以parsePrice的返回值也是可选的。
这样带有if let block是一个很好的封装处理,如果一切顺利,将会打印出获取的价格。
接下来我们开始着手说说如何实现retrievePriceData方法。核心就是使用基于Foundation框架的NSURLSession类来完成实际的网络请求。如果网络请求成功,将可以通过所提供的闭包返回NSData类型的数据给调用者,否则就只需要返回nil值。
func?bitstampURL(interval:?IntervalType)?->?NSURL?{ ????if?(interval?==?.LastIntervalType?||?interval?==?.VWAPIntervalType)?{ ????????return?NSURL(string:?"https://www.bitstamp.net/api/ticker/")! ????}?else?{ ????????return?NSURL(string:?"https://www.bitstamp.net/api/ticker_hour/")! ????} } func?retrievePriceData(interval:?IntervalType,?completion:?NSData??->?Void)?->?Void?{ ????let ????????url?=?bitstampURL(interval), ????????request?=?NSURLRequest(URL:?url), ????????session?=?NSURLSession.sharedSession, ????????semaphore?=?dispatch_semaphore_create(0) ???????? ????let?task?=?session.dataTaskWithRequest(request)?{ ?????????(data,?response,?error)?->?Void?in ????????if?error?==?nil?{ ????????????completion(data) ????????}?else?{ ????????????completion(nil) ????????} ????????dispatch_semaphore_signal(semaphore) ????} ????task.resume ???? ????dispatch_semaphore_wait(semaphore,?DISPATCH_TIME_FOREVER) }
这里比较复杂的地方就是,命令行脚本的执行,会在最后一个函数返回后结束。然而,NSURLSession提供给我们的是一个异步的接口,在网络调用完成之前,这个方法就已经返回。所以为了解决这一点,我们需要使用GCD的信号灯(semaphore)来阻塞主线程,直到网络请求完成。
JSON解析
我们剩下的最后一个任务就是解析retriecePriceData返回的数据,我们将使用parsePrice函数对之进行解析。我们通过BitStamp API获取的数据是如下JSON格式:
{ ????high:?"502.00", ????last:?"446.02", ????timestamp:?"1446693126", ????bid:?"446.03", ????vwap:?"445.3", ????volume:?"108095.47008023", ????low:?"368.11", ????ask:?"446.60", ????open:?407.99 }
我们主要关注的是last和vwap属性,但是需要解析整个对象为字典类型数据,然后返回我们所关注的值即可。所以,我们接下来使用Foundation框架中带有一些有条件的费覆盖的NSJSONSerialization类,来进行处理:
func?priceKey(interval:?IntervalType)?->?String?{ ????if?(interval?==?.LastIntervalType?||?interval?==?.HourlyIntervalType)?{ ????????return?"last" ????}?else?{ ????????return?"vwap" ????} } func?parsePrice(interval:?IntervalType,?data:?NSData)?->?Double??{ ????do?{ ????????let?json?=?try?NSJSONSerialization.JSONObjectWithData(data,?options:?[]) ????????if?let ????????????priceData?=?json?as??Dictionary, ????????????priceString?=?priceData[priceKey(interval)]?as??String?{ ????????????//?Using?the?failable?initializer?to?convert?to?a?Double? ????????????return?Double(priceString) ????????} ????}?catch?{ ????????//?died?parsing?the?JSON ????} ????return?nil }
运行和编译 我们已经完成准备工作了!全部的entire script源码都在我们的github上,可以给你提供一些参考。它实现了我们最初设想的所有需求,为了更清楚的看到效果,我们来运行看看:
$?./btc.swift?-i?vwap 333.95 $?./btc.swift?--interval?hourly 334.05
使用上面这些命令,我们已经可以把程序作为一个脚本来运行:Swift能够动态编译和执行它。如果我们想要直接编译btc.swift成可执行文件,然后运行它。
在我做过多次尝试之后发现,它不能使用swiftc动态链接到第三方库。虽然可以正常编译btc.swift脚本,但是在运行的时候,还是不能引用到OptionKit框架。我怀疑移除OptionKit依赖或许能够正常编译这个脚本。如果有人知道怎么能让这个正常工作,那就太感谢了。
你可以用下面的命令来编译执行,这样就能看到,在运行的时候报错了:
#?Compile?using?swiftc $?xcrun?-sdk?macosx?swiftc?-F?Carthage/Build/Mac/?btc.swift?-o?btc #?Attempt?to?run $?./btc dyld:?Library?not?loaded:?@rpath/OptionKit.framework/Versions/0/OptionKit
总结
除了把脚本编译成可执行文件外,我们已经完成了全部工作。我们已经解决了加载第三方库、解析命令行参数、加载远程的JSON数据以及将这些JSON数据解析成我们能够熟练操纵的Swift数据类型。到此,已经编写好了一个可运行的脚本来通过BitStamp监测比特币的价格。通过完成这个功能,我们也了解到了一些Swift语言的新特性。
所以,通过这样一个专题的讲解,我希望你们都认同专题中预期的结论,同时也能够学到更多的新知识。我个人希望这个专题作为Swift脚本语言开发的一个入门实例,当我们使用命令行脚本时,帮助解决我们所面临的一些比较常见的问题。感谢苹果公司为我们提供这样一种非常棒的编程语言,它不仅能够非常灵活简洁地让我们开发一个复杂的iOS应用,也支持像我们上面所写的例子中使用更加简单的脚本进行编程。我很期待Swift脚本能更多应用在其他的方面,希望Swift的开源发布,能够使Swift脚本进入到Linux系统一个新的起点。
更多译者翻译文章,请查看:http://www.cocoachina.com/special/translation/
本文仅用于学习和交流目的,转载请注明文章译者、出处和本文链接。
感谢博文视点对本期活动的支持
- 上一篇:Swift 语言指南-Issue 45
- 下一篇:Swift—高仿单糖App
相关推荐
- 10款超实用JavaScript音频库(js播放音频代码)
-
HTML5提供了一种新的音频标签实现和规范用一个简单的HTML对象而无需音频插件来控制音频。这只是一个简单的整合这些新的HTML5音频特征及使用JavaScript来创建各种播放控制。下面将介绍10款...
- PROFINET转Modbus网关——工业协议融合的智能枢纽
-
三格电子SG-PNh750-MOD-221,无缝连接Profinet与Modbus,赋能工业物联产品概述...
- 简单实用的Modbus类库,支持从站和DTU
-
一、简介...
- [西门子PLC] S7-200 SMART PROFINET :通过GSD组态PLC设备
-
从S7-200SMARTV2.5版本开始,S7-200SMART开始支持做PROFINETIO通信的智能设备。从而,两个S7-200SMART之间可以进行PROFINETI...
- Modbus(RTU / TCP)有什么异同(modbus tcp和tcp)
-
Modbus是一种广泛使用的工业自动化通信协议,它支持设备之间的数据交换。Modbus协议有两个主要的变体:ModbusRTU(二进制模式)和ModbusTCP(基于TCP/IP网络的模式)。尽管...
- Modbus通信调试步骤详解(modbus调试工具怎么用)
-
Modbus通信调试步骤详解 Modbus通信分为串口和以太网,无论是串口还是以太网,只要是标准Modbus,就可以用Modbus模拟器进行调试。按以下几步进行调试。...
- 理解Intel手册汇编指令(intel 汇编指令手册)
-
指令格式...
- 「西门子PLC」S7-200 SMART的Modbus RTU通讯
-
S7-200SMART集成的RS485端口(端口0)以及SBCM01RS485/232信号板(端口1)两个通信端口可以同时做MODBUSRTU主站,或者一个做MODBUSRTU主站一个做MO...
- InfiniBand网络运维全指南:从驱动安装到故障排查
-
一、InfiniBand网络概述InfiniBand(直译为“无限带宽”技术,缩写为IB)是一种用于高性能计算的计算机网络通信标准,具有极高的吞吐量和极低的延迟,用于计算机与计算机之间的数据互连。它...
- 一加回归 OPPO,背后的秘密不可告人
-
有这样一个手机品牌,它诞生于互联网品牌。在大众群体看来,它的身世似乎模糊不清,许多人以为它是国外品牌。它的产品定位是极客群体,深受国内发烧友,甚至国外极客玩家喜爱。...
- [西门子PLC] S7-200SMART快速高效的完成Modbus通信程序的设计
-
一、导读Modbus通信是一种被广泛应用的通信协议,在变频器、智能仪表还有其他一些智能设备上都能见到它的身影。本文呢,就把S7-200SMART系列PLC当作Modbus主站,把...
- 狂肝10个月手搓GPU,他们在我的世界中玩起我的世界,梦想成真
-
梦晨衡宇萧箫发自凹非寺量子位|公众号QbitAI自从有人在《我的世界》里用红石电路造出CPU,就流传着一个梗:...
- [西门子PLC] 博途TIA portal SCL编程基础入门:1-点动与自锁
-
一、S7-SCL编程语言简介...
- 工作原理系列之:Modbus(modbus工作过程)
-
MODBUS是一种在自动化工业中广泛应用的高速串行通信协议。该协议是由Modion公司(现在由施耐德电气公司获得)于1979年为自己的可编程逻辑控制器开发的。该协议充当了PLCS和智能自动化设备之间的...
你 发表评论:
欢迎- 一周热门
-
-
Linux:Ubuntu22.04上安装python3.11,简单易上手
-
宝马阿布达比分公司推出独特M4升级套件,整套升级约在20万
-
MATLAB中图片保存的五种方法(一)(matlab中保存图片命令)
-
别再傻傻搞不清楚Workstation Player和Workstation Pro的区别了
-
如何提取、修改、强刷A卡bios a卡刷bios工具
-
Linux上使用tinyproxy快速搭建HTTP/HTTPS代理器
-
Element Plus 的 Dialog 组件实现点击遮罩层不关闭对话框
-
日本组合“岚”将于2020年12月31日停止团体活动
-
MacOS + AList + 访达,让各种云盘挂载到本地(建议收藏)
-
SpringCloud OpenFeign 使用 okhttp 发送 HTTP 请求与 HTTP/2 探索
-
- 最近发表
-
- 10款超实用JavaScript音频库(js播放音频代码)
- Howler.js,一款神奇的 JavaScript 开源网络音频工具库
- PROFINET转Modbus网关——工业协议融合的智能枢纽
- 简单实用的Modbus类库,支持从站和DTU
- [西门子PLC] S7-200 SMART PROFINET :通过GSD组态PLC设备
- Modbus(RTU / TCP)有什么异同(modbus tcp和tcp)
- Modbus通信调试步骤详解(modbus调试工具怎么用)
- 理解Intel手册汇编指令(intel 汇编指令手册)
- 「西门子PLC」S7-200 SMART的Modbus RTU通讯
- InfiniBand网络运维全指南:从驱动安装到故障排查
- 标签列表
-
- dialog.js (57)
- importnew (44)
- windows93网页版 (44)
- yii2框架的优缺点 (45)
- tinyeditor (45)
- qt5.5 (60)
- windowsserver2016镜像下载 (52)
- okhttputils (51)
- android-gif-drawable (53)
- 时间轴插件 (56)
- docker systemd (65)
- slider.js (47)
- android webview缓存 (46)
- pagination.js (59)
- loadjs (62)
- openssl1.0.2 (48)
- velocity模板引擎 (48)
- pcre library (47)
- zabbix微信报警脚本 (63)
- jnetpcap (49)
- pdfrenderer (43)
- fastutil (48)
- uinavigationcontroller (53)
- bitbucket.org (44)
- python websocket-client (47)