摘要:上周又推出了新一轮服务器端增强函数库。对,你没看错,在服务器端其实可以轻松从其他语种的函数库中直接拿来调用,不需要修改任何内容。执行函数方法用于带参数执行某个函数。
上周Perfect又推出了新一轮服务器端Swift增强函数库:Perfect-Python。对,你没看错,在服务器端Swift 其实可以轻松从其他语种的函数库中直接拿来调用,不需要修改任何内容。如果没有类似经验的童鞋可以参考拙作:
(1)CSwift:https://github.com/RockfordWe... ,
——如何在Swift程序中直接嵌入C语言源代码
(2)csweet:https://github.com/RockfordWe...,
——如何在Swift程序中直接嵌入C++语言源代码
(周末刚做的,文档还没来得及写呢,凑合看源代码吧哈)
而Perfect-Python不是简单嵌入源代码的问题,是允许用户直接调用Python函数库本身!!!!
为什么花大力气做这些东西?很简单,程序员都是很懒的家伙,与其重新把几万行代码重写一遍,不如直接抓壮丁来的快——现成成熟的代码在快速原型法中起的作用之大难以想象。
不错,是由一些很矫情的家伙宣称什么“语言的纯洁性”,比如Vapor,去年一直号称要“纯洁的Swift”,结果两周前终于顶不住压力,参考上一期Ryan的报告可以看到:
Vapor is no longer pure Swift as of Vapor 2, and includes at least chttp and LibreSSL / OpenSSL.
翻译“Vapor已经不像Vapor 2那样使用纯Swift语言了,因为至少包括了一 个C语言组件chttp和LibreSSL / OpenSSL加密函数库”
看见了吗?Vapor作为典型的反面教材,在陷入服务器性能问题的泥潭后终于招架不住投降了——用了两年时间证明一个光鲜的口号是错误的,这是什么样的代价!“实践是检验真理的唯一标准”——闭门思过去吧。
实践证明,全栈开发需要扬长避短,然后用统一的一种编译型语言做主调是在性能和效率上达到最合理平衡点的,目的是:
快速开发项目
维持代码稳定性
项目人工最小化
最短学习曲线
保持组件最新
高性能服务器
迅速增加更多新功能——这是最关键最关键的
好了,言归正传,我们看一下Perfect-Python的具体用法:
本项目提供了在Swift服务器应用上直接引用Python 2.7函数库的简便方法。
本项目采用Swift Package Manager 软件包管理器编译,是Perfect 项目的一部分,但是也可以独立运行
在使用之前请准备好最新的Swift 3.1 / 4.0 工具链
Linux 编译事项首先请确保 libpython2.7-dev 已经在 Ubuntu 16.04 上正确安装:
$ sudo apt-get install libpython2.7-devMacOS 编译事项
请确定 Xcode 8.3.3 / 9.0 以上版本已经正确安装
快速上手首先在Package.swift中增加依存关系:
.Package(url: "https://github.com/PerfectlySoft/Perfect-Python.git", majorVersion: 1)
然后将下列头文件导入Swift源代码:
import PythonAPI import PerfectPython
请注意在任何程序调用之前,必须调用Py_Initialize()函数初始化python嵌入环境:
Py_Initialize()导入Python函数库模块
使用 PyObj 类对象用于导入python模块。下列参考范例中,一个名为/tmp/clstest.py的脚本被动态导入到当前Swift运行环境:
let pymod = try PyObj(path: "/tmp", import: "clstest")访问Python变量
导入模块后,您可以使用PyObj.load()函数加载任何一个变量;也可以反过来用 PyObj.save()命令保存当前变量为一个新的值。
比如,以下python脚本中有个叫做 stringVar 的字符串变量:
stringVar = "Hello, world"
那么要取得这个字符串的值只需要这样做:
if let str = pymod.load("stringVar")?.value as? String { print(str) // 会打印变量的字符串值 "Hello, world!" }
此时您还可以为该变量直接写入新的字符串值:
try pymod.save("stringVar", newValue: "Hola!")
注意 目前,Perfect-Python仅支持如下Swift / Python数据类型自动转换:
Python 类型 | Swift 类型 | 备注 |
---|---|---|
int | Int | |
float | Double | |
str | String | |
list | [Any] | 递归转换 |
dict | [String:Any] | 递归转换 |
比如,您可以把一个字符串 String 转换为 PyObj,通过 let pystr = "Hello".python() 或者 let pystr = try PyObj(value:"Hello") 完成转换。
反过来,如果要把 PyObj 类转换为Swift数据类型,比如字符串,则仍然有两种方法:let str = pystr.value as? String 和 let str = String(python: pystr)。
执行Python函数方法 PyObj.call() 用于带参数执行某个python函数。以如下python脚本为例:
def mymul(num1, num2): return num1 * num2
Perfect-Python 可以用下列方法封装并调用以上函数,您所需要注意的仅仅是其函数名称以及参数。其中函数名称用字符串代替,而参数用一个数组表达:
if let res = pymod.call("mymul", args: [2,3])?.value as? Int { print(res) // 结果为 6 }Python类对象
请同样使用 PyObj.load() 函数用于家在Python类对象,但是注意后面一定要紧跟一个PyObj.construct() 用于初始化类对象实例。该方法同样支持用一个任意类型的数组作为参数进行对象构造。
假设如下脚本的典型python类对象 Person,该类有两个属性姓名name 和年龄age,还有一个名为“自我介绍”的类对象方法intro():
class Person: def __init__(self, name, age): self.name = name self.age = age def intro(self): return "Name: " + self.name + ", Age: " + str(self.age)
在Swift中初始化上述类对象的方法需要进行以下两步走:
if let personClass = pymod.load("Person"), let person = personClass.construct(["rocky", 24]) { // person is now the object instance }
之后就可以访问类实例的属性变量和方法了,如同上文所提到的普通变量和函数调用的方法一样:
if let name = person.load("name")?.value as? String, let age = person.load("age")?.value as? Int, let intro = person.call("intro", args: [])?.value as? String { print(name, age, intro) }回调函数
参考以下python代码,此时如果执行 x = caller("Hello", callback) 则可以将函数作为参数进行回调:
def callback(msg): return "callback: " + msg def caller(info, func): return func(info)
在Swift中等效的代码平淡无奇,只不过将待调函数作为参数而已::
if let fun = pymod.load("callback"), let result = pymod.call("caller", args: ["Hello", fun]), let v = result.value as? String { print(v) // 结果是 "callback: Hello" }更多信息
关于本项目更多内容,请参考perfect.org.
扫一扫 Perfect 官网微信号文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/40801.html
摘要:的官方网址为,其使用手册网址为本次分享将实现的功能为利用爬取某个搜索词语暂仅限英文的百度百科的介绍部分,具体的功能介绍可以参考博客爬虫自制简单的搜索引擎。 Jsoup 是一款Java 的HTML解析器,可直接解析某个URL地址、HTML文本内容。它提供了一套非常省力的API,可通过DOM,CSS以及类似于jQuery的操作方法来取出和操作数据。Jsoup的官方网址为: https:...
摘要:写在前面最近每日一更,我这菜鸡都有点儿不好意思了简单介绍简单用法是让我们用程序来测试这个,如果相当于,那么一个出来。 写在前面 最近每日一更,我这菜鸡都有点儿不好意思了 简单介绍 简单用法是: assert expression 让我们用程序来测试这个expression,如果expression相当于False,那么raise一个AssertionError出来。即逻辑上等同于: i...
摘要:说到,大家可能觉得很简单呀,不就是用来调用父类方法的嘛。单继承在单继承中就像大家所想的那样,主要是用来调用父类的方法的。你觉得执行下面代码后,的值是多少呢执行结果如下这个结果说明了两个问题确实调用了父类的方法。 说到 super, 大家可能觉得很简单呀,不就是用来调用父类方法的嘛。如果真的这么简单的话也就不会有这篇文章了,且听我细细道来。? 约定 在开始之前我们来约定一下本文所使用的 ...
摘要:有着一堆神秘的语法和过时的功能。我试图列出一些鲜为人知的特性。虽然它们很酷,但毕竟是鲜为人知的特性,你的同事可能会看不懂。类似这样使用的话会始终保持返回正确的。 By Viral Shah | Nov 26, 2018 原文 js一门很容易入门但是很难精通的语言。我很认同这句话。这是因为js是一门古老的语言同时也是一门很灵活的语言。有着一堆神秘的语法和过时的功能。我已经使用js很多年了...
摘要:熟悉的人都知道,提供两种包含预定义增删改查操作的接口对比这两个接口,操作都差不多,名字有一点点改变,比如里面叫的方法,在里面叫。另外还有一种方式就是通过把暴露出来这个也是的实例。这样的话,使用的时候就只需要引用一个对象了 熟悉 mybatis-plus 的人都知道,mybatis-plus 提供两种包含预定义增删改查操作的接口: com.baomidou.mybatisplus.co...
阅读 1325·2021-11-15 18:11
阅读 2469·2021-08-19 10:56
阅读 642·2021-08-09 13:42
阅读 744·2019-08-30 15:53
阅读 2038·2019-08-30 10:55
阅读 3090·2019-08-29 17:18
阅读 1358·2019-08-29 13:45
阅读 518·2019-08-29 13:15