Scalaz(5)- typeclass:my typeclass scalaz style-demo

2024-04-09 05:08
文章标签 style demo scalaz typeclass

本文主要是介绍Scalaz(5)- typeclass:my typeclass scalaz style-demo,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  我们在上一篇讨论中介绍了一些基本的由scalaz提供的typeclass。这些基本typeclass主要的作用是通过操作符来保证类型安全,也就是在前期编译时就由compiler来发现错误。在这篇讨论中我希望能按照scalaz的格式设计自己的typeclass并能使之融入scalaz库结构里去。

  我们来设计一个NoneZero typeclass。这个NoneZero typeclass能确定目标类型值是否为空,如:

0.nonZero = false

3.nonZero = true

"".nonZero = false

"value".nonZero = true

List().nonZero = false

List(1,2,3).nonZero = true

首先是trait: (定义typeclass行为)

trait NonZero[A] {def nonZero(a: A): Boolean
}

现在NonZero typeclass只有一项功能或行为,就是这个抽象函数NonZero:对任何类型A值a,返回Boolean结果。

为了方便使用NoneZero typeclass,我们在伴生对象里定义NonZero[A]的构建函数:

object NonZero {def create[A](f: A => Boolean): NonZero[A] = new NonZero[A] {def nonZero(a: A): Boolean = f(a)}
}

只要我们提供一个f: A => Boolean函数就能用create来构建一个NonZero[A]实例。实际上这个f函数定义了类型A在NonZero tyoeclass中的具体行为。

下一步是注入操作方法:

class NonZeroOps[A](a: A)(implicit ev: NonZero[A]) {def isNonZero: Boolean = ev.nonZero(a)
}

我们注入了一个操作方法isNonZero。注意:注入方法是针对所有类型A的,所以需要NonZero[A]作为参数。

跟着就是隐式作用域解析了(implicit resolution):

object ToNonZeroOps {implicit def toNonZeroOps[A](a: A)(implicit ev: NonZero[A]) = new NonZeroOps[A](a)
}

这是一个隐式视域(implicit view):从类型A转换到NonZeroOps[A]。这样类型A就具备isNonZero这个操作方法了。

我们按scalaz惯例在object NonZero放一个默认隐式转换:

object NonZero {def create[A](f: A => Boolean): NonZero[A] = new NonZero[A] {def nonZero(a: A): Boolean = f(a)}implicit val intNZInstance: NonZero[Int] = create {case 0 => falsecase _ => true}
}

注意我们在create参数里使用了partial function。

然后试试在Int类型上使用:

import ToNonZeroOps._10.isNonZero                                      //> res0: Boolean = true
0.isNonZero                                       //> res1: Boolean = false
2.isNonZero                                       //> res2: Boolean = true

不错,已经可以使用了。再试试其它即兴类型:

import ToNonZeroOps._
implicit val stringNZInstance: NonZero[String] = NonZero.create {case "" => falsecase _ => true
}                                                 //> stringNZInstance  : scalaz.ex5.NonZero[String] = scalaz.ex5$NonZero$$anon$1@//| 1c655221
implicit val booleanNZInstance: NonZero[Boolean] = NonZero.create { b => b }//> booleanNZInstance  : scalaz.ex5.NonZero[Boolean] = scalaz.ex5$NonZero$$anon$//| 1@6aaa5eb0implicit def listNZInstance[A]: NonZero[List[A]] = NonZero.create {case Nil => falsecase _ => true
}                                                 //> listNZInstance: [A]=> scalaz.ex5.NonZero[List[A]]"".isNonZero                                      //> res0: Boolean = false
"not empty".isNonZero                             //> res1: Boolean = true
true.isNonZero                                    //> res2: Boolean = true
false.isNonZero                                   //> res3: Boolean = false
List(1,2,3).isNonZero                             //> res4: Boolean = true
List("a","b").isNonZero                           //> res5: Boolean = true
List().isNonZero                                  //> res6: Boolean = false10.isNonZero                                      //> res7: Boolean = true
0.isNonZero                                       //> res8: Boolean = false
2.isNonZero                                       //> res9: Boolean = true

我把完整的代码贴在下面吧,供大家练习参考:

trait NonZero[A] {def nonZero(a: A): Boolean
}
object NonZero {def create[A](f: A => Boolean): NonZero[A] = new NonZero[A] {def nonZero(a: A): Boolean = f(a)}implicit val intNZInstance: NonZero[Int] = create {case 0 => falsecase _ => true}
}
class NonZeroOps[A](a: A)(implicit ev: NonZero[A]) {def isNonZero: Boolean = ev.nonZero(a)
}
object ToNonZeroOps {implicit def toNonZeroOps[A](a: A)(implicit ev: NonZero[A]) = new NonZeroOps[A](a)
}import ToNonZeroOps._
implicit val stringNZInstance: NonZero[String] = NonZero.create {case "" => falsecase _ => true
}                                                 //> stringNZInstance  : scalaz.ex5.NonZero[String] = scalaz.ex5$NonZero$$anon$1@//| 1c655221
implicit val booleanNZInstance: NonZero[Boolean] = NonZero.create { b => b }//> booleanNZInstance  : scalaz.ex5.NonZero[Boolean] = scalaz.ex5$NonZero$$anon$//| 1@6aaa5eb0implicit def listNZInstance[A]: NonZero[List[A]] = NonZero.create {case Nil => falsecase _ => true
}                                                 //> listNZInstance: [A]=> scalaz.ex5.NonZero[List[A]]"".isNonZero                                      //> res0: Boolean = false
"not empty".isNonZero                             //> res1: Boolean = true
true.isNonZero                                    //> res2: Boolean = true
false.isNonZero                                   //> res3: Boolean = false
List(1,2,3).isNonZero                             //> res4: Boolean = true
List("a","b").isNonZero                           //> res5: Boolean = true
List().isNonZero                                  //> res6: Boolean = false10.isNonZero                                      //> res7: Boolean = true
0.isNonZero                                       //> res8: Boolean = false
2.isNonZero                                       //> res9: Boolean = true




这篇关于Scalaz(5)- typeclass:my typeclass scalaz style-demo的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

linux 内核提权总结(demo+exp分析) -- 任意读写(四)

hijack_modprobe_path篇 本文转自网络文章,内容均为非盈利,版权归原作者所有。 转载此文章仅为个人收藏,分享知识,如有侵权,马上删除。 原文作者:jmpcall 专栏地址:https://zhuanlan.kanxue.com/user-815036.htm     原理同hijack_prctl, 当用户执行错误格式的elf文件时内核调用call_usermod

linux 内核提权总结(demo+exp分析) -- 任意读写(三)

hijack_prctl篇 本文转自网络文章,内容均为非盈利,版权归原作者所有。 转载此文章仅为个人收藏,分享知识,如有侵权,马上删除。 原文作者:jmpcall 专栏地址:https://zhuanlan.kanxue.com/user-815036.htm   prctl函数: 用户态函数,可用于定制进程参数,非常适合和内核进行交互 用户态执行prctl函数后触发prctl系统

linux 内核提权总结(demo+exp分析) -- 任意读写(二)

hijack_vdso篇 本文转自网络文章,内容均为非盈利,版权归原作者所有。 转载此文章仅为个人收藏,分享知识,如有侵权,马上删除。 原文作者:jmpcall 专栏地址:https://zhuanlan.kanxue.com/user-815036.htm     vdso: 内核实现的一个动态库,存在于内核,然后映射到用户态空间,可由用户态直接调用 内核中的vdso如果被修改

linux 内核提权总结(demo+exp分析) -- 任意读写(一)

cred篇 本文转自网络文章,内容均为非盈利,版权归原作者所有。 转载此文章仅为个人收藏,分享知识,如有侵权,马上删除。 原文作者:jmpcall 专栏地址:https://zhuanlan.kanxue.com/user-815036.htm   每个线程在内核中都对应一个线程结构块thread_infothread_info中存在task_struct类型结构体 struct t

linux 内核提权总结(demo+exp分析) -- ROP(二)

ret2usr CR4篇 本文转自网络文章,内容均为非盈利,版权归原作者所有。 转载此文章仅为个人收藏,分享知识,如有侵权,马上删除。 原文作者:jmpcall 专栏地址:https://zhuanlan.kanxue.com/user-815036.htm   smep: smep是内核的一种保护措施, 使得内核不可执行用户态代码 内核通过CR4寄存器的第20位来控制smep,

linux 内核提权总结(demo+exp分析) -- ROP(一)

基础ROP篇(linux 5.0.21) 本文转自网络文章,内容均为非盈利,版权归原作者所有。 转载此文章仅为个人收藏,分享知识,如有侵权,马上删除。 原文作者:jmpcall 专栏地址:https://zhuanlan.kanxue.com/user-815036.htm   内核提权与用户态攻击的区别 攻击流程 用户态攻击: 执行 system("/bin/sh") 获得shel

实例demo理解面向接口思想

浅显的理解面向接口编程 Android开发的语言是java,至少目前是,所以理解面向接口的思想是有必要的。下面通过一个简单的例子来理解。具体的概括我也不知道怎么说。 例子: 现在我们要开发一个应用,模拟移动存储设备的读写,即计算机与U盘、MP3、移动硬盘等设备进行数据交换。已知要实现U盘、MP3播放器、移动硬盘三种移动存储设备,要求计算机能同这三种设备进行数据交换,并且以后可能会有新的第三方的

类codepen的实现可拖拽窗口demo

首先说下思想 flex或者其他布局方式,实现左右分割布局,主盒子宽度100%,左右布局中包含一个分割条(可在布局容器中,也可以单独定义)为分隔条绑定鼠标点击事件,为document绑定鼠标移动事件和鼠标放开事件,通过监听鼠标移动事件和上一个状态保存下来的鼠标位置作对比,判断当前鼠标移动方向(往左还是往右)然后计算当前鼠标位置和鼠标点击位置的距离,来计算左右容器的变化,然后通过dom的方式设置宽度

llama.cpp demo

git clone https://github.com/ggerganov/llama.cppcd llama.cpp 修改Makefile使能mfma参数     MK_CFLAGS   += -mfma -mf16c -mavx     MK_CXXFLAGS += -mfma -mf16c -mavx 安装python3依赖 cat ./requirements/requirem

WebAPI (一)DOM树、DOM对象,操作元素样式(style className,classList)。表单元素属性。自定义属性。间歇函数定时器

文章目录 Web API基本认知一、 变量声明二、 DOM1. DOM 树2. DOM对象3. 获取DOM对象(1)、选择匹配的第一个元素(2)、选择匹配多个元素 三、 操作元素1. 操作元素内容2. 操作元素属性(1)、常用属性(href之类的)(2)、通过style属性操作CSS(3)、通过类名(className)操作CSS(4)、通过classList操作控制CSS(5)、操作表单