给markdown文档标题添加有序编号的python脚本

1. 背景

我们通过 typora 软件编写的 md 文档,所有的标题是不会进行自动编号的,就像下图所示:

在这里插入图片描述

如果我们想看到编号效果,有一种常见的解决方法,就是在软件的主题目录下新增 base.user.css 文件,然后写入特定内容,具体可参考这儿

操作成功之后,我们就可以在软件的左边大纲页面看到标题都添加了具体的编号。

在这里插入图片描述

但是这种方式有一个缺点:这只是在大纲页面显示出了编号,但是在具体的标题上是没有显示编号的,具体看下图

在这里插入图片描述

所以这只是修改了展示效果,但是实际上并没有修改具体的内容。

如果我们想将自己写的 md 文档内容迁移到博客上,最后显示到博客上的内容,标题也是没有编号的,如果我们想要真正给标题内容添加上编号,这种方式就不可行了。

2. python 脚本解决

先给自己的机器上安装好 python 环境,具体可参考这儿

这儿提供了一个我写好的 python 脚本,用来给没有编号的 md 文档添加有序编号。

# 给 markdown 文档的所有标题添加编号,编号形式为:1.2.3.4.5.6.

import re
import sys

pattern = r'^(\d+\.)+$'
'''正则表达式,判断字符串是否为一个数字后面跟着一个小数点的形式'''

in_format = False
'''当前行是否在 ```格式化中,如果为格式化内容,则直接忽略'''


def is_title(line: str):
    """
    判断该行内容是否为标题

    :param line: 文件中的某一行内容
    :return: 该行内容是否为标题
    """
    global in_format
    if line.__contains__('```'):
        if in_format:
            in_format = False
        else:
            in_format = True
    if line.startswith('#'):
        if in_format:
            return False
        else:
            return True


def remove_old_num(line: str):
    """
    移除标题中之前生成的编号,依据:
    1. 排除该行内容第一个空格前面的所有 # 号,
    2. 删除剩下的内容以空格分隔之后,从前往后符合编号正则的所有的内容,之后再把前面所有的 # 号和最后剩下的内容拼接起来

    :param line: 文件中的某个标题行
    :return: 去掉标题内容中之前添加的编号
    """
    split1 = line.split(' ', 1)
    if len(split1) < 2:
        raise Exception(F'该行标题内容不符合规范,具体内容为:\n${line}\n标题中的多个 # 后面应该使用空格和标题内容分开')
    # 结果,开头是标题符号
    result = split1[0]
    # 可能会有多个冗余编号,需要循环去除
    title_num_content = split1[1]
    split2 = title_num_content.split(' ')

    # 将去除了标题之后的内容按照空格分隔,如果分隔结果有多个,则循环去除符合编号正则的内容,并将非编号内容拼接到结果中
    if len(split2) > 1:
        for split in split2:
            # 不符合标题编号正则,可以拼接到结果中
            if not re.search(pattern, split):
                result += (' ' + split)
    else:
        result += ' ' + split2[0]

    return result


def title_symbol_num(line):
    """
    判断标题为几级标题
    """
    line = str(line)
    return len(line.split(' ', 1)[0])


auto_num = []
'''存储自动编号,根据标题来不断变化'''


def add_auto_num(line: str):
    """
    给标题行添加新的自动标号\n
    如果标题级别和标题存储变量长度一样,则直接最后一个数字自增即可\n
    如果不一样:\n
    标题级别长,则变量后面新增一个数字1\n
    标题级别短,则变量对应长度位置的数字加1,并且去掉后面所有的数字

    :param line: 无编号的标题行内容
    :return: 添加了编号的标题行内容
    """
    global auto_num
    title_num = title_symbol_num(line)
    title_symbol_content = line.split(' ', 1)
    title_symbol = title_symbol_content[0]
    title_content = title_symbol_content[1]
    if title_num == len(auto_num):
        auto_num[len(auto_num) - 1] += 1
    elif title_num > len(auto_num):
        auto_num.append(1)
    else:
        auto_num = auto_num[0: title_num]
        auto_num[len(auto_num) - 1] += 1
    title_num = ''
    for num in auto_num:
        title_num += str(num) + '.'
    return title_symbol + ' ' + title_num + ' ' + title_content


def file_content_handle(md_file_path: str):
    """
    处理文件内容

    :param md_file_path: 待处理文件路径
    """
    origin_file = open(md_file_path, 'r', encoding='utf-8')
    new_file = open(md_file_path.rsplit('.', 1)[0] + '_num' + '.md', 'w', encoding='utf-8')
    lines = origin_file.readlines()
    for line in lines:
        if is_title(line):
            line = remove_old_num(line)
            line = add_auto_num(line)
        new_file.writelines(line)
    origin_file.close()
    new_file.close()


if __name__ == '__main__':
    file_path = sys.argv[1]
    file_content_handle(sys.argv[1])

你可以直接将上面的 python 脚本代码复制到一个 python 文件中,比如md_title_auto_num.py,然后执行该脚本即可。具体执行命令为:

python md_title_auto_num.py Spring5笔记.md

命令解释:

  • python:这是执行 python 脚本的命令。

  • md_title_auto_num.py:这个是保存了上述 python 代码的文件。

  • Spring5笔记.md:这是原始的 md 文件,这个文件可以是相对于执行的 python 文件的相对路径,也可以是绝对路径。这里我将 python 脚本文件和要处理的 md 文件都放到了同一个目录下。下面演示几个其他类型的命令

    • 相对路径

      python .\md_title_auto_num.py D:\PYWorkspace\study\教学项目\markdown\Spring5笔记.md
      
    • 绝对路径

      python .\md_title_auto_num.py .\spring全家桶\Spring5笔记.md
      

运行无出错之后,在原文件的同级目录下应该会有一个在原文件名称基础上添加了_num_后缀的文件,打开之后,就会发现该文件中的所有标题都已经添加上了有序的编号,如下图。

在这里插入图片描述
标题前面确确实实添加了真正的编号内容,之后,就可以将文件导入到博客中了。

在这里插入图片描述

可以看到,博客中的标题,也确实是有了编号。

  • 7
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

第一片心意

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值