广告
返回顶部
首页 > 资讯 > 后端开发 > Python >Python实现自定义Jupyter魔法命令
  • 901
分享到

Python实现自定义Jupyter魔法命令

2024-04-02 19:04:59 901人浏览 八月长安

Python 官方文档:入门教程 => 点击学习

摘要

相信大家都用过 jupyter,也用过里面的魔法命令,这些魔法命令都以 % 或者 %% 开头,我们举个例子。 用法还是比较简单的,

相信大家都用过 jupyter,也用过里面的魔法命令,这些魔法命令都以 % 或者 %% 开头,我们举个例子。

用法还是比较简单的,但是我们能不能自定义魔法命令呢?毫无疑问是可以的,因为上面的 %%cython 就是 Cython 模块自定义的。

所以命令可以是 jupyter 内置的,比如 %time,直接拿来就能用;还可以是第三方模块里面的,在 jupyter 通过 %load_ext 加载之后,再嵌入进来。下面就来看看如何自定义魔法命令。

from Ipython.core.magic import (
    magics_class,
    Magics,
    line_magic,
    cell_magic
)


@magics_class
class MagicOrder(Magics):
    """
    自定义一个类,类名叫什么无所谓
    但要继承 Magics,并且要被 magics_class 装饰
    """

    @line_magic
    def hello(self, line):
        """
        在 jupyter 中就可以使用如下命令,比如:
        %hello <Your Code>,然后就会调用这个 hello 方法
        参数 line 就是 %hello 后面的代码
        """
        print(f"line: {line}")

    @cell_magic
    def world(self, line, cell):
        """
        在 jupyter 中就可以使用如下命令,比如:
        %%world
        <Your Code>
        <Your Code>
        ...

        然后就会调用这个 world 方法
        参数 cell 就是 %%world 下面整个单元格的代码

        然后还有一个参数 line,它表示 %%world 所在行后面的代码
        但对于 %% 开头的命令来说,我们一般都会新起一行,然后写代码
        所以 line 这个参数暂时用不到
        """
        print(f"line: \n{line}")
        print("-----------------")
        print(f"cell: \n{cell}")


# 必须定义 load_iPython_extension 函数
# %load_ext 本质上也是加载一个模块
# 但它会自动调用该函数
def load_ipython_extension(ip):
    # 在函数内部,我们将类 MagicOrder 注册进去
    # 然后就可以使用它内部的魔法命令了
    ip.reGISter_magics(MagicOrder)

# 如果不定义此函数,那么使用 %load_ext 加载时会报错
# The xxx module is not an IPython extension.

当前模块叫 main.py,我们来测试一下:

结果没有问题,但说实话对于 %% 开头的命令来说,我们很少会在它后面写代码,基本都是新起一行,就像下面这个样子。

自定义命令我们已经实现了,并且也知道怎么获取输入的代码了,下面要做的就是执行它。而将字符串当成代码执行,我们可以使用内置函数 exec。

@magics_class
class MagicOrder(Magics):

    @line_magic
    def hello(self, line):
        exec(line)

    @cell_magic
    def world(self, line, cell):
        exec(cell)

代码的其它部分不变,然后你觉得接下来调用魔法命令会执行成功吗?我们测试一下:

神奇的地方出现了,虽然命令执行成功了,但执行完之后,告诉我们变量未定义。其实原因很好想,我们调用 exec 的时候没有指定名字空间,那么默认会影响 exec 函数所在的名字空间,即 hello 和 world 函数的名字空间。

当打开一个 jupyter 的时候,内部相当于启动了一个 shell,所以在调用 exec 的时候,应该将整个 shell 的名字空间传进去。

from IPython.core.magic import (
    magics_class,
    Magics,
    line_magic,
    cell_magic,
    needs_local_scope
)


@magics_class
class MagicOrder(Magics):

    @line_magic
    def hello(self, line):
        # 通过 self.shell.user_ns
        # 可以拿到当前 shell 的名字空间
        # 注意:包含所有的单元格
        local_ns = self.shell.user_ns
        # 在 local_ns 当中执行代码
        exec(line, local_ns, local_ns)

    @needs_local_scope
    @cell_magic
    def world(self, line, cell, local_ns):
        # 或者通过 needs_local_scope 装饰器
        # 这样在调用函数的时候,会额外传递一个 local_ns 参数
        # 该参数和 self.shell.user_ns 等价
        exec(cell, local_ns, local_ns)

def load_ipython_extension(ip):
    ip.register_magics(MagicOrder)

然后再来测试一下:

此时就没有任何问题了。

下面我们模仿 jupyter 的 %time 命令,实现一个 %my_time,来加深一遍印象。

@magics_class
class MagicOrder(Magics):

    @needs_local_scope
    @line_magic
    def my_time(self, line, local_ns):
        start = time.perf_counter()
        exec(line, local_ns, local_ns)
        end = time.perf_counter()
        print(f"总耗时: {round(end - start, 3)}")

测试一下:

结果没有问题,是我们想要的结果。

最后再来看看如何设置可选参数,举一个 Cython 的例子:

我们说对于以 %% 开头的命令,应该新起一行,在它的下面写代码。而之所以新起一行,是因为命令所在的行,要用于设置可选参数。那么问题来了,如何设置指定的可选参数呢?

from IPython.core.magic import (
    magics_class,
    Magics,
    cell_magic,
    needs_local_scope
)
from IPython.core import magic_arguments


@magics_class
class MagicOrder(Magics):
    @magic_arguments.magic_arguments()
    # 在 jupyter 中可以通过 -n=xxx 或者 --name=xxx
    # 然后是 dest="name",用于指定参数的名字
    # 后续便可以通过 name 字段来获取该参数的值
    @magic_arguments.argument(
        "-n", "--name", dest="name", default="satori"
    )
    # "-" 和 "--" 可以只出现一个,并且默认解析得到的是字符串
    # 而 age 我们希望是整数,所以指定 type 为 int
    # 解析完参数之后,会自动调用 int 进行转化
    # 如果不指定该参数,则使用 default
    # 而这里没有 default,那么结果就是 None
    @magic_arguments.argument(
        "--age", dest="age", type=int
    )
    @magic_arguments.argument(
        "-h", "--hobby", dest="hobby", default=[],
        action="append"
    )
    @needs_local_scope
    @cell_magic
    def order(self, line, cell, local_ns):
        # 显然 line 就是可选参数,cell 就是代码块
        exec(cell, local_ns, local_ns)
        # 解析参数
        args = magic_arguments.parse_argstring(
            self.order, line)
        # 打印
        print(args)


def load_ipython_extension(ip):
    ip.register_magics(MagicOrder)

我们测试一下:

还是很简单的,而且这里的参数解析和 argparse 模块非常类似,可以自己看一下。

到此这篇关于Python实现自定义Jupyter魔法命令的文章就介绍到这了,更多相关Python Jupyter魔法命令内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: Python实现自定义Jupyter魔法命令

本文链接: https://www.lsjlt.com/news/120076.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Python实现自定义Jupyter魔法命令
    相信大家都用过 jupyter,也用过里面的魔法命令,这些魔法命令都以 % 或者 %% 开头,我们举个例子。 用法还是比较简单的,...
    99+
    2022-11-11
  • python魔法方法-自定义序列详解
    自定义序列的相关魔法方法允许我们自己创建的类拥有序列的特性,让其使用起来就像 python 的内置序列(dict,tuple,list,string等)。 如果要实现这个功能,就要遵循 python 的相关...
    99+
    2022-06-04
    自定义 序列 详解
  • uboot添加自定义命令的实现步骤
    目录uboot简介实现步骤:头文件:函数:添加命令update:uboot简介 uboot 属于bootloader的一种,是用来引导启动内核的,它的最终目的就是:从flash中读出...
    99+
    2022-11-21
    uboot添加自定义命令 uboot命令
  • vue3的自定义指令directives实现
    目录一、什么是自定义指令二、指令的分类三、指令的作用四、指令的钩子五、钩子参数六、指令的使用指令的参数和修饰符一、什么是自定义指令 我们已经熟悉Vue内置的一系列指令 ,比如 v-m...
    99+
    2022-11-13
  • Angular自定义指令Tooltip的方法实例
    目录目录结构编写 tooltip 组件编写 tooltip 指令页面上调用总结Yeah,关注我的读者应该知道,上一篇文章了解 Angular 开发的内容,我们已经概览了 Angula...
    99+
    2022-11-13
  • python实现自定义日志的具体方法
    1、导入logging模块: import logging 2、创建日志收集器: logger = logging.getLogger(“日志收集器的name”) 3、设置日志收集器的日志级别: lo...
    99+
    2022-06-02
    python 自定义日志
  • Python实现自定义异常实例
    目录前言1、使用 raise 语句来抛出异常2、自定义异常类总结前言 在Python中,抛出自定义异常的语法为 raise 异常类对象。也就是说可以使用 raise 语句来...
    99+
    2022-11-13
  • Android自定义Style实现方法
    styles.xml如下: [html] 代码如下:<resources xmlns:android="http://schemas.android.com/apk/r...
    99+
    2022-06-06
    方法 Android
  • Vue自定义指令深入探讨实现
    目录指令自定义指令关于注册自定义指令全局注册局部注册自定义指令的钩子函数封装自定义指令指令 想要了解自定义指令,那肯定得先明白什么是指令。 指令的本质:语法糖,标志位。在编译阶段 r...
    99+
    2023-05-19
    Vue自定义指令 Vue自定义指令实现
  • 怎么在Vue3中实现自定义指令
    这篇文章主要介绍“怎么在Vue3中实现自定义指令”,在日常操作中,相信很多人在怎么在Vue3中实现自定义指令问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么在Vue3中实现自定义指令”的疑惑有所帮助!接下来...
    99+
    2023-07-02
  • Vue3directive自定义指令内部实现示例
    目录directive-自定义指令(属于破坏性更新)Vue3指令的钩子函数在setup内定义局部指令生命周期钩子参数详解函数简写directive-自定义指令(属于破坏性更新) Vu...
    99+
    2022-12-15
    Vue3 directive自定义指令 Vue3 directive
  • Vue中的自定义指令怎么实现
    今天小编给大家分享一下Vue中的自定义指令怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。试炼:实现v-mymodel...
    99+
    2023-07-04
  • python自定义分页器的实现
    目录自定义分页器封装代码自定义分页器使用后端前端自定义分页器封装代码 封装分页相关数据: :param current_page: 当前页:param all_count: 数据库中...
    99+
    2022-11-10
  • Python+LyScript实现自定义反汇编
    LyScript 插件默认提供了一个get_disasm_code()方法可以直接获取到指定行数的反汇编代码,但如果需要自定义获取或者是需要自己封装一个反汇编方法,则你可以用如下两种...
    99+
    2022-11-11
  • python自定义日志怎么实现
    这篇文章主要介绍了python自定义日志怎么实现,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1、导入logging模块:import logging2、创建日志收...
    99+
    2023-06-15
  • android自定义组件实现方法
    本文实例讲述了android自定义组件实现方法。分享给大家供大家参考。具体如下: atts.xml: <?xml version="1.0" encoding...
    99+
    2022-06-06
    方法 Android
  • Android自定义ViewGroup的实现方法
         在android中提供了常见的几种ViewGroup的实现,包括LinearLayout、Relativeayout、Fram...
    99+
    2022-06-06
    方法 Android
  • Python实现自定义包的实例详解
    目录一.实例:自定义包二.详解1.新建my_utils包2.新建str_util.py和file_util.py两个python file3.str_util.py中的代码4.fil...
    99+
    2022-12-29
    Python自定义包 Python
  • 利用Python实现自定义连点器
    目录前言整体思路所有功能简单演示点击功能延时功能连点功能存储功能读取功能存储导入功能【升级版】拖拽功能右击、中击、双击功能的实现按键功能滚动滚轮功能查看功能执行功能逻辑判断功能板块判...
    99+
    2022-11-11
  • python怎么实现自定义异常类
    本篇内容介绍了“python怎么实现自定义异常类”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!说明通常可以继承Exception或子类。命名...
    99+
    2023-06-30
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作