nim 语言使用 concept 实现 c# 的interface
nim简介
nim语言兼顾C#等高级语言语义表达的丰富性,又有 C 语言的灵活性,以及超强的性能。下面是中文站对他的总结,我抄下来:
Nim 是一种静态类型的、编译型、系统编程语言。
它结合了其他成熟语言的成功概念。
(如 Python、Ada 和 Modula)
效率
- Nim 生成原生且无依赖的可执行文件,不依赖于虚拟机,
所以它们小巧易分发。 - Nim 编译器和生成的可执行文件,对目前的任何主流平台都提供了支持,
包括 Windows、Linux、BSD 和 macOS。 - Nim 的内存管理是确定性的,可使用析构函数和移动语义进行自定义, 其灵感来自C++和Rust。 非常适合嵌入式硬实时系统。
- 零开销迭代器和用户自定义方法的编译期求值等现代概念,
结合优先使用分配在栈上的值类型数据,生成高性能代码。 - 支持各种后端:可以被编译为 C、C++ 或 JavaScript, 以便 Nim 可用于所有后端和前端需求。
表现力
- Nim 实现了自举:编译器和标准库都是用 Nim 本身来实现的。
- Nim 拥有强大的宏系统,允许直接操纵 AST,提供无限的可能性。
优雅
- 宏不会改变 Nim 的语法,因为并没有这个必要
—— Nim 语法本身已经足够灵活。 - 具有局部类型推断、元组、泛型和sum类型的现代类型系统。
- 语句按缩进分组,也可以跨行。
接口
我满怀信心的开始学习 nim 语言,之前有 c# 和 kotlin 的编程经验,所以自然会将面向对象的思路带到 nim 中,看官方文档,继承功能完全是内建的,所以没有遇到什么困难,结果在 接口 interface 问题上卡壳了。
查询很多的资料,有几种解法,一种是官方库中 stream 的解法,要定义一个 stream 和一个 streamObj ,然后在实现类中一一映射,我觉得太麻烦,直接放弃了。
还有一篇文章介绍了利用 nim 的宏功能,自己造了一个轮子,我觉得主要的问题是,不是官方支持的,我如果写了大量的代码,有问题不好解决,而且有更好的官方方案的话,就要大量重写。况且,他构建了 vTable ,有间接调用成本。
但是,功夫不负有心人,我终于找到官方的解决方案,使用 concept 来解决 interface 的问题, concept 中文翻译为 概念,有点拗口,我的理解是,定义一堆约束,如果某个类符合这个约束,那么就自然可以使用这个约束来访问,这不就是 interface 想要表达的意思吗?
但我觉得 nim 设计 concept 时,思想比 interface 更进一步,就是鸭子类型(duck typing)的概念,即如果它走起路来像鸭子,叫起来也是鸭子,那么它就是鸭子。
下面是定义 concept (接口) 的方法:
type
# 定义一个 IImmutableList<E> 这样的接口,在nim中叫 概念
IImmutableList[E] = concept self # 包含一个方法,允许用 var a = obj[32] 这样的方式索引访问。
self.`[]`(uint64) is E # 定义了一个 size 只读属性
self.size is uint64
然后你就可以写自己的类了,只要你的方法和签名与接口一致就可以了,不需要像 c# 或 kotlin ,java 那样要显式的写实现了此接口。
type
SeqImmutableList[E] = ref object of RootObj
items : seq[E] # 为 SeqImmutableList 定义了一个 size 属性
proc size*[E](this: SeqImmutableList[E]) : uint64 {.inline.} =
uint64(this.items.len) #用到类型转换 # 定义一个可以直接使用 [] 方式访问的方法。
proc `[]`*[E](this: SeqImmutableList[E], index : uint64) : E {.inline.} =
this.items[int(index)] # 相当于构造函数,只有构造函数才能访问私有变量。
proc newSeqImmutableList*[E](items : seq[E]) : SeqImmutableList[E] =
new(result)
result.items = items let target = newSeqImmutableList(@[1,2,3,4])
echo "size = ", target.size
echo "target[2] = ", target[2]
到目前,上面的代码仍然是类自己的方法去访问的,但是你想使用接口的方式访问,非常的方便,直接传递就可以了。
# 以接口(概念)的方式访问
proc printAll(this: IImmutableList) =
echo "size = ", this.size
for i in 0..<this.size:
echo "items[", i, "] = ", this[i] printAll(target)
这里我将 target (他是SeqImmutableList<int>类型)直接传递给 printAll 方法,编译器只要觉得符合 概念(concept),就可以传递。
接口的继承
如果我现在想定义一个新的接口,派生自原来的接口怎么办呢?
IMutableList[E] = concept self
# 相当于c# 的继承
self is IImmutableList[E] # 提供按索引更改的能力,类似 c# 的 this[i] = x 这样的赋值。
self.`[]=`(idx,E)
使用 is ... 表达了继承的概念,然后再编写你新的想定义的方法。
现在,我们新定义一个类,让他两个接口都支持,当然,我们顺便也学习一下对象继承的写法。
type
SeqMutableList[E] = ref object of SeqImmutableList[E]
isDirty : bool # 提供按索引的写
proc `[]=`[E](this: SeqMutableList, index:idx, value : E) =
this.elements[int(index)] = value proc newSeqMutableList[E](items : seq[E]) : SeqMutableList[E] =
return SeqMutableList[E](elements:items, isDirty:false)
现在,你可以按照类本身 SeqMutableList[E] 的方式访问他,也可以使用两个接口 IImmutableList[E] 和 IMutableList[E] 的方式,也可以使用基类 SeqImmutableList 的方式访问
# 创建可变的对象
let target2 = newSeqMutableList(@[1,2,3,4,5])
echo "size2 = ", target2.size #访问基类的方法 # 修改他的值,并使用基类提供的方法重新获取
target2[2] = 99
echo "target2[2] = ", target2[2] # 可以传递给基准的接口去访问。
printAll(target2) # 可变的接口访问,包含基类的方法。
let target3 : IMutableList[int] = target2
echo "size3 = ", target3.size
target3[2] = 9999
echo "target3[2] = ", target3[2]
echo "target2[2] = ", target2[2]
let target4 : SeqImmutableList[int] = target2
echo "target4[2] = ", target4[2]
概念除了可以作为接口使用,还可以对泛型约束,更多复杂的使用方法,参见:https://nim-lang.org/docs/manual_experimental.html#concepts
nim 语言使用 concept 实现 c# 的interface的更多相关文章
- Nim语言的模块化编程
前言 Nim支持把一大段程序分成若干个模块 一个模块就是一个源代码文件 每个模块都拥有它自己的名称空间 模块化可以起到封装(信息隐藏)和分步编译的作用 一个模块可以通过import语句获得另一个模块的 ...
- 学习Nim语言.rar(nim语言中文教程下载)
学习Nim语言 nim 语法上类似python ,是一门静态编译型语言,nim 使用空格缩进标示语句块的开始和结束, 喜欢python风格的程序员应该也会很容易适应和喜欢nim的风格. nim语言官方 ...
- Nim语言:Pascal的语法,Python的缩进
http://nim-lang.org/ 德国人Andreas Rumpf的作品,原因是他对过去使用的每种语言都不满意(Pascal也不满意?).以前叫Nimrod语言,从0.96版本开始改名为Nim ...
- GO学习-(38) Go语言结构体转map[string]interface{}的若干方法
结构体转map[string]interface{}的若干方法 本文介绍了Go语言中将结构体转成map[string]interface{}时你需要了解的"坑",也有你需要知道的若 ...
- 用Nim语言开发windows GUI图形界面程序
前言 本文得到了“樂師”的大力支持, 我们一起调试程序到深夜,要是没有他的帮忙, 我不知道要多久才能迈过这道坎, 另外“归心”还有其他人也提供了帮助, 他们都来自于QQ群:“Nim开发集中营”4693 ...
- nim的引用和指针
nim语言的引用和其他语言的指针有点相似 可以提供一种“多对一”的关系 这就意味着不同的引用可以指向同一个内存位置 nim区分可被追踪的引用和不可被追踪的引用 不可被追踪的引用又称为指针 可被追踪的引 ...
- Nim编码风格
介绍 Nim语言不限制开发人员使用哪种具体的编码风格, 但为了社区的发展,在编写一些标准库的时候还是应该遵从统一的编码风格 这篇文章会列出一系列的编码风格准则,供大家参考. 但值得注意的是,有很多例外 ...
- Nim教程【十五】【完结】
模版 模版是Nim语言中的抽象语法树,它是一种简单的替换机制,在编译期被处理 这个特性使Nim语言可以和C语言很好的运行在一起 像调用一个方法一样调用一个模版 请看如下代码: template `!= ...
- Nim教程【十四】
网友@沉没捕鱼,赞助了一台服务器 这个系列的教程写完之后,我们就要开始着手搭建Nim的社区了~ 异常 Nim中的异常类型是对象类型 根据惯例,Nim中的异常类型的命名都应该以Error后缀结尾 在sy ...
- Nim教程【九】
向关注这个系列的朋友们,道一声:久违了! 它并没有被我阉掉,他一定会得善终的,请各位不要灰心 Set集合类型 为了在特殊场景下提高程序的性能设置了Set类型,同时也是为了保证性能,所以Set只能容纳有 ...
随机推荐
- C#中使用正则将字符串中某字符不区分大小写并按全字匹配替换为空
具体代码如下所示: //将字符串中desc不区分大小写并按全字匹配替换为空 var strText = "CreatDeSce DeSc,UserName AsC"; string ...
- 解决Git拉取出现“bad config line 1 in file C:\Users\quber/.gitconfig”的错误
1.问题描述 我们在拉取Git项目的时候,突然出现如下图所示的错误提示: 2.解决办法 定位到.gitconfig文件,然后将其删除掉: 然后在项目文件夹中点击鼠标右键,选择Git Bash Here ...
- galaxy特色胡思乱想
有没有什么办法,让我不伤害任何人,什么也不破坏,被判死刑.我觉得这样比我紫砂要好的多. 我所可怜的是神不能紫砂.--芥川龙之介<某傻子的一生>
- [BZOJ4665] 小w的喜糖 题解
我们先假设同种糖间存在差异. 设 \(f_{i,j}\) 表示前 \(i\) 种糖至少有 \(j\) 人拿到的糖和原来一样,\(c_i\) 表示拿第 \(i\) 种糖的人的个数,则有: \[f_{i, ...
- [Jaav SE/程序生命周期] 优雅的Java应用程序的启停钩子框架
序 了解 spring 生态及框架的 java er 都知道,spring 应用的生命周期管理及配套接口较为优雅.可扩展. 但脱离 spring 的 java 应用程序,如何优雅地启停.管理程序的生命 ...
- Go红队开发—并发编程
目录 并发编程 go协程 chan通道 无缓冲通道 有缓冲通道 创建⽆缓冲和缓冲通道 等协程 sync.WaitGroup同步 Runtime包 Gosched() Goexit() 区别 同步变量 ...
- Selenium WebDriver上创建 WebDriver测试脚本
本文实现一个WebDriver测试脚本,介绍WebDrive的常用命令.UI元素定位的策略以及在脚本中的使用,还有Get命令. 你将学到: 脚本创建 代码走查 测试执行 定位Web元素 ...
- python二级 计算生态
生态地址: https://pypi.python.org/pypi 常用函数:
- jmespath 使用及案例
什么是jmespath jmespath 是python里面的一个库 主要在httprunner框架里使用 2.使用语法 列表: with_jmespath(jmes_path,var_name) m ...
- python excel 打开表格:表格名不知道应该怎么打开
取所有表格名的倒数第一个就是操作的表格 import pandas as pd xl = pd.ExcelFile(xlPath) names = xl.sheet_names df = xl.par ...