Doris FE启动流程源码解读

news2024/11/17 2:51:54

FE启动流程分析

Doris中FE主要负责接收和返回客户端请求、元数据以及集群管理、查询计划生成等工作。

本文主要看一下Doris的fe在启动时做了什么。

启动流程分析

启动流程图:

在这里插入图片描述

代码路径:

doris/fe/fe-core/src/main/java/org/apache/doris/DorisFE.java

FE server的启动代码主要在这个JAVA文件中。在启动FE的时候,主要做了以下几件事:

  1. 环境检查

    主要是检查一些启动时必要的环境变量以及初始化配置文件。包含:

    **DORIS_HOME_DIR **

    如果没有人为配置 DORIS_HOME_DIR,则该变量的值就是doris的解压安装目录。

    PID_DIR

    PID_DIR是为了判断FE进程是第一次启动还是之前启动过。

     if (Strings.isNullOrEmpty(dorisHomeDir)) {
                System.err.println("env DORIS_HOME is not set.");
                return;
            }
    
            if (Strings.isNullOrEmpty(pidDir)) {
                System.err.println("env PID_DIR is not set.");
                return;
            }
    

    初始化fe.conf、fe_custom.conf、ldap.conf

     // init config
                Config config = new Config();
                config.init(dorisHomeDir + "/conf/fe.conf");
                // Must init custom config after init config, separately.
                // Because the path of custom config file is defined in fe.conf
                config.initCustom(Config.custom_config_dir + "/fe_custom.conf");
    
                LdapConfig ldapConfig = new LdapConfig();
                if (new File(dorisHomeDir + "/conf/ldap.conf").exists()) {
                    ldapConfig.init(dorisHomeDir + "/conf/ldap.conf");
                }
    

    检测JDK版本是否匹配,主要是检测compile的JDK和runtime的jdk版本,需要要求runtimeVersion > compileVersion

        public static boolean checkJavaVersion() {
            if (!Config.check_java_version) {
                return true;
            }
    
            String javaCompileVersionStr = getJavaVersionFromFullVersion(Version.DORIS_JAVA_COMPILE_VERSION);
            String javaRuntimeVersionStr = System.getProperty("java.version");
    
            int compileVersion = JdkUtils.getJavaVersionAsInteger(javaCompileVersionStr);
            int runtimeVersion = JdkUtils.getJavaVersionAsInteger(javaRuntimeVersionStr);
    
            if (runtimeVersion < compileVersion) {
                System.out.println("The runtime java version " + javaRuntimeVersionStr + " is less than "
                        + "compile version " + javaCompileVersionStr);
                return false;
            }
    
            return true;
        }
    

    检查 解析启动FE时输入的命令行参数,以便进行不同的操作,主要会包含这几类:

    **–version:**或者执行 -v ,主要是打印FE的版本

    –helper: -h ,主要是指定 helper node 然后加入FE的 bdb je的副本组

    –image: -i,主要是检查image文件

    –bdb: -b,主要是用以运行bdbje的命令行工具

    具体解析逻辑如下(bdbje tool的代码逻辑过长,有兴趣的可以自己去看一下 parseArgs的实现):

     // version
            if (cmd.hasOption('v') || cmd.hasOption("version")) {
                return new CommandLineOptions(true, "", null, "");
            }
            // helper
            if (cmd.hasOption('h') || cmd.hasOption("helper")) {
                String helperNode = cmd.getOptionValue("helper");
                if (Strings.isNullOrEmpty(helperNode)) {
                    System.err.println("Missing helper node");
                    System.exit(-1);
                }
                return new CommandLineOptions(false, helperNode, null, "");
            }
            // image
            if (cmd.hasOption('i') || cmd.hasOption("image")) {
                // get image path
                String imagePath = cmd.getOptionValue("image");
                if (Strings.isNullOrEmpty(imagePath)) {
                    System.err.println("imagePath is not set");
                    System.exit(-1);
                }
                return new CommandLineOptions(false, "", null, imagePath);
            }
         //bdb tool
    

    根据输入的参数,如果不是运行image tool、bdbje tool或者打印FE的version信息,就继续往下执行

    这个时候就要准备开始启动FE了。

    同样,启动FE时,需要初始化一些操作。

    初始化的时候,主要是检查了FE的启动IP,是不是一个合法的IP。

    这里需要注意的就是,我们在配置文件中配置的CIDR活着FQDN的配置,在初始化的时候会检测。很多小伙伴在启动FE的时候,没有正确配置IP的时候,最后用了localhost或者本地回环IP启动,导致没有使用我们想要的IP启动,具体的判断逻辑就是在这:

       static void initAddrUseIp(List<InetAddress> hosts) {
            useFqdn = false;
            analyzePriorityCidrs();
            // if not set frontend_address, get a non-loopback ip
            InetAddress loopBack = null;
            boolean hasMatchedIp = false;
            for (InetAddress addr : hosts) {
                LOG.debug("check ip address: {}", addr);
                if (addr instanceof Inet4Address) {
                    if (addr.isLoopbackAddress()) {
                        loopBack = addr;
                    } else if (!priorityCidrs.isEmpty()) {
                        if (isInPriorNetwork(addr.getHostAddress())) {
                            localAddr = addr;
                            hasMatchedIp = true;
                            break;
                        }
                    } else {
                        localAddr = addr;
                        break;
                    }
                }
            }
            //if all ips not match the priority_networks then print the warning log
            if (!priorityCidrs.isEmpty() && !hasMatchedIp) {
                LOG.warn("ip address range configured for priority_networks does not include the current IP address");
            }
            // nothing found, use loopback addr
            if (localAddr == null) {
                localAddr = loopBack;
            }
            LOG.info("local address: {}.", localAddr);
        }
    

    上面的逻辑看,初始化的时候会遍历网卡信息,拿遍历的IP地址和填写的PRIORITY_CIDR_SEPARATOR的值做匹配,匹配上了,就会用处于填写的CIDR范围中的ip启动,匹配不上的时候会从网卡IP中拿出一个合法的IP作为FE的启动IP,这个就不一定是我们想要的那个启动IP。特别是当前机器上有很多虚拟网卡的IP信息,就会很大概率用排在前面的虚拟IP启动。

    当然,这里还会根据配置文件中的信息,去检查是不是FQDN,是不是IPV6,有兴趣的的同学都可以看一下具体的代码逻辑。

    Init操作其实就是获取了当前FE的启动IP,获取完IP后,就需要检测端口,看FE的启动的需要的这些端口是否是正常的。

     private static void checkAllPorts() throws IOException {
            if (!NetUtils.isPortAvailable(FrontendOptions.getLocalHostAddress(), Config.edit_log_port,
                    "Edit log port", NetUtils.EDIT_LOG_PORT_SUGGESTION)) {
                throw new IOException("port " + Config.edit_log_port + " already in use");
            }
            if (!NetUtils.isPortAvailable(FrontendOptions.getLocalHostAddress(), Config.http_port,
                    "Http port", NetUtils.HTTP_PORT_SUGGESTION)) {
                throw new IOException("port " + Config.http_port + " already in use");
            }
            if (Config.enable_https && !NetUtils.isPortAvailable(FrontendOptions.getLocalHostAddress(),
                    Config.https_port, "Https port", NetUtils.HTTPS_PORT_SUGGESTION)) {
                throw new IOException("port " + Config.https_port + " already in use");
            }
            if (!NetUtils.isPortAvailable(FrontendOptions.getLocalHostAddress(), Config.query_port,
                    "Query port", NetUtils.QUERY_PORT_SUGGESTION)) {
                throw new IOException("port " + Config.query_port + " already in use");
            }
            if (!NetUtils.isPortAvailable(FrontendOptions.getLocalHostAddress(), Config.rpc_port,
                    "Rpc port", NetUtils.RPC_PORT_SUGGESTION)) {
                throw new IOException("port " + Config.rpc_port + " already in use");
            }
        }
    

    这里的代码就有很多大家比较熟悉的 " already in use",所以我们在启动的时候,尽可能先检测一下。

    端口如果被占用启动是不会成功的。


    现在是不是觉得所有的准备工作都做完了,可以正式启动FE了?

    还有一个比较重要的检测,就是需要根据fe.conf中的 enable_bdbje_debug_mode参数的值,来决定怎么启动。

    这个值主要是某些时候,我们的FE的leader选举出现一定问题,做元数据运维的时候,会走运维模式逻辑。如果是正常情况下,这个值默认是FALSE,就会走后续的正常启动FE的流程。

    1. 元数据环境初始化

      • 元数据目录,如果不存在,需要手动创建,

         File meta = new File(metaDir);
                if (!meta.exists()) {
                    LOG.warn("Doris' meta dir {} does not exist." + " You need to create it before starting FE",
                            meta.getAbsolutePath());
                    throw new Exception(meta.getAbsolutePath() + " does not exist, will exit");
                }
        
                if (Config.edit_log_type.equalsIgnoreCase("bdb")) {
                    File bdbDir = new File(this.bdbDir);
                    if (!bdbDir.exists()) {
                        bdbDir.mkdirs();
                    }
                }
                File imageDir = new File(this.imageDir);
                if (!imageDir.exists()) {
                    imageDir.mkdirs();
                }
        

        这里主要是需要手动创建最外层的metaDir,内层的bdb的目录和image的目录会自己创建。

      • 初始化 插件管理器,启动审计日志进程

      • 根据当前的元数据信息获取集群ID和节点角色信息(ROLE和VERSION文件的判断)

        代码较长,只节选了关键代码。具体逻辑可以看一下getClusterIdAndRole 的具体实现,这里主要就是根据指定的helper的节点的元数据信息或者本地存在的元数据信息,获取到集群的ROLE信息和VERSION信息。

        如果集群是非helper节点且第一次启动,ROLE文件实没有,这个时候需要创建这个文件。同时赋予相关值(ROLE=FOLLOWER),将节点信息写入到元数据文件中。

        role = FrontendNodeType.FOLLOWER;
                        nodeName = genFeNodeName(selfNode.getIdent(),
                                selfNode.getPort(), false /* new style */);
                        storage.writeFrontendRoleAndNodeName(role, nodeName);
        

        如果当前阶段存在这些元数据文件,则会去元数据文件中获取当前节点的角色信息:

              role = storage.getRole();
                        if (role == FrontendNodeType.REPLICA) {
                            // for compatibility
                            role = FrontendNodeType.FOLLOWER;
                        }
        
                        nodeName = storage.getNodeName();
                        if (Strings.isNullOrEmpty(nodeName)) {
                            // In normal case, if ROLE file exist, role and nodeName should both exist.
                            // But we will get a empty nodeName after upgrading.
                            // So for forward compatibility, we use the "old-style" way of naming: "ip_port",
                            // and update the ROLE file.
                            nodeName = genFeNodeName(selfNode.getHost(), selfNode.getPort(), true/* old style */);
                            storage.writeFrontendRoleAndNodeName(role, nodeName);
                            LOG.info("forward compatibility. role: {}, node name: {}", role.name(), nodeName);
                        }
        

        如果VERSION文件不存在,则会生成一个包含新的ClusterID和token信息的文件。同时实例化一个FE实例(第一次启动)。

         if (!versionFile.exists()) {
                        clusterId = Config.cluster_id == -1 ? Storage.newClusterID() : Config.cluster_id;
                        token = Strings.isNullOrEmpty(Config.auth_token) ? Storage.newToken() : Config.auth_token;
                        storage = new Storage(clusterId, token, this.imageDir);
                        storage.writeClusterIdAndToken();
        
                        isFirstTimeStartUp = true;
                        Frontend self = new Frontend(role, nodeName, selfNode.getHost(),
                                selfNode.getPort());
                        // Set self alive to true, the BDBEnvironment.getReplicationGroupAdmin() will rely on this to get
                        // helper node, before the heartbeat thread is started.
                        self.setIsAlive(true);
                        // We don't need to check if frontends already contains self.
                        // frontends must be empty cause no image is loaded and no journal is replayed yet.
                        // And this frontend will be persisted later after opening bdbje environment.
                        frontends.put(nodeName, self);
                        LOG.info("add self frontend: {}", self);
        

        存在的话,就会直接去对应的文件中获取相关的ClusterIdAndToken。

        如果我们启动了一个FE,无法从给出的helper节点信息中,同helper节点建立连接,就会出现:

        current node is not added to the group. please add it first. "
                                    + "sleep 5 seconds and retry, current helper nodes: {}", helperNodes
        

        的日志信息,这个异常原因就是由于当前节点无法和指定的helper节点建立正常的连接信息导致的。

        当和helper节点构建正常连接后,就会从helper节点同步 VERSION信息。

        如果本身节点存在VERSIN文件的信息,说明不是第一次启动,这个时候就会用本地的这个文件的元数据信息同HELPER节点的VERSION信息进行比对。主要是比较clusterID。如果不一致,说明两个节点不是同一个集群的节点,启动进程就直接退出了。

        经过这一步 VERSION和ROLE的元数据信息比对后,确定是同一个集群内的节点,也确定了这个FE的ROLE信息了,就需要从image中同步editlog。同时创建一系列的cleaner 线程和监听线程:

         // 3. Load image first and replay edits
                this.editLog = new EditLog(nodeName);
                loadImage(this.imageDir); // load image file
                editLog.open(); // open bdb env
                this.globalTransactionMgr.setEditLog(editLog);
                this.idGenerator.setEditLog(editLog);
        
                // 4. create load and export job label cleaner thread
                createLabelCleaner();
        
                // 5. create txn cleaner thread
                createTxnCleaner();
        
                // 6. start state listener thread
                createStateListener();
                listener.start();
        

        此时启动前初始化工作就做完了。等待catalog信息的同步完成即可进行下一步。

    2. 启动FE的SERVER

        FeServer feServer = new FeServer(Config.rpc_port);
                  feServer.start();
      
                  if (options.enableHttpServer) {
                      HttpServer httpServer = new HttpServer();
                      httpServer.setPort(Config.http_port);
                      httpServer.setHttpsPort(Config.https_port);
                      httpServer.setMaxHttpPostSize(Config.jetty_server_max_http_post_size);
                      httpServer.setAcceptors(Config.jetty_server_acceptors);
                      httpServer.setSelectors(Config.jetty_server_selectors);
                      httpServer.setWorkers(Config.jetty_server_workers);
                      httpServer.setKeyStorePath(Config.key_store_path);
                      httpServer.setKeyStorePassword(Config.key_store_password);
                      httpServer.setKeyStoreType(Config.key_store_type);
                      httpServer.setKeyStoreAlias(Config.key_store_alias);
                      httpServer.setEnableHttps(Config.enable_https);
                      httpServer.setMaxThreads(Config.jetty_threadPool_maxThreads);
                      httpServer.setMinThreads(Config.jetty_threadPool_minThreads);
                      httpServer.setMaxHttpHeaderSize(Config.jetty_server_max_http_header_size);
                      httpServer.start();
                      Env.getCurrentEnv().setHttpReady(true);
                  }
      
                  if (options.enableQeService) {
                      QeService qeService = new QeService(Config.query_port, ExecuteEnv.getInstance().getScheduler());
                      qeService.start();
                  }
      
                  ThreadPoolManager.registerAllThreadPoolMetric();
      
                  while (true) {
                      Thread.sleep(2000);
                  }
              } catch (Throwable e) {
                  LOG.warn("", e);
              }
      

      这里正常启动相关的SERVER即可。

      还有一个比较重要的点就是,如果enable_bdbje_debug_mode值为TRUE,也就是需要走元数据运维,他又是如何启动的呢?

      我们回到前面判断这个值的代码:

                  if (Config.enable_bdbje_debug_mode) {
                      // Start in BDB Debug mode
                      BDBDebugger.get().startDebugMode(dorisHomeDir);
                      return;
                  }
      

      继续往下看:

        public void startDebugMode(String dorisHomeDir) {
              try {
                  initDebugEnv();
                  startService(dorisHomeDir);
                  while (true) {
                      Thread.sleep(2000);
                  }
              } catch (Throwable t) {
                  LOG.warn("BDB debug mode exception", t);
                  System.exit(-1);
              }
          }
      

      主要是初始化 Debug环境的代码:

         public void init() throws BDBDebugException {
                  EnvironmentConfig envConfig = new EnvironmentConfig();
                  envConfig.setAllowCreate(false);
                  envConfig.setReadOnly(true);
                  envConfig.setCachePercent(20);
      
                  try {
                      env = new Environment(new File(metaPath), envConfig);
                  } catch (DatabaseException e) {
                      throw new BDBDebugException("failed to init bdb env", e);
                  }
                  Preconditions.checkNotNull(env);
              }
      

      这里可以看到,是利用了一个元数据,创建了一个新的环境。然后从这个环境中做了一系列的事情:

      • List all database in bdbje.
      • get the number of journal in specified database.
      • get list of journal id (key) in specified database.
      • get the journal entity of the specified journal id.

    此时,我们就可以通过MySQL client来查看这些信息。同时,后续的启动流程还会继续。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/661628.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

华为云CodeArts Build快速上手编译构建-进阶玩家体验

华为云CodeArts Build编译构建为开发者提供配置简单的混合语言构建平台&#xff0c;实现编译构建云端化&#xff0c;支撑企业实现持续交付&#xff0c;缩短交付周期&#xff0c;提升交付效率。支持编译构建任务一键创建、配置和执行&#xff0c;实现获取代码、构建、打包等活动…

react+antd实现表格封装,可动态控制列显示隐藏。

实现效果 import { Table, Pagination, Button, Dropdown, Checkbox, message } from antd; import { useState, useEffect } from react; import { PicRightOutlined } from ant-design/icons;import ./index.less;const TableComponent (props) > {const powerList JSON…

【方法】Excel表格的“打开密码”不想要了,如何取消?

对于重要的Excel表格&#xff0c;很多小伙伴都会设置“打开密码”&#xff0c;这样就无法随意打开表格&#xff0c;只有输入正确的密码才可以打开。 如果后续表格不再需要保护&#xff0c;每次打开都要输一次密码&#xff0c;这样操作也是很麻烦。 那不想要“打开密码”&…

(5)(5.8) 保存微调和自动微调

文章目录 前言 1 保存微调 2 自动微调 3 保存微调和自动微调的视频演示 4 桌面方法 前言 当然&#xff0c;风对你的旋翼飞机有很大的影响&#xff0c;会把它推来推去。然而&#xff0c;你可能也会发现&#xff0c;在自稳模式下飞行时&#xff0c;即使在无风的环境中&#xff0…

chatgpt赋能python:Python重新编辑引擎优化(SEO)文章

Python重新编辑引擎优化(SEO)文章 介绍 Python是一种多用途的高级编程语言&#xff0c;用于开发网络应用程序&#xff0c;算法&#xff0c;科学计算和数据分析等。 随着越来越多的网站和应用程序采用Python编写&#xff0c;优化Python代码以提高搜索引擎优化(SEO)变得越来越重…

基于html+css的图展示133

准备项目 项目开发工具 Visual Studio Code 1.44.2 版本: 1.44.2 提交: ff915844119ce9485abfe8aa9076ec76b5300ddd 日期: 2020-04-16T16:36:23.138Z Electron: 7.1.11 Chrome: 78.0.3904.130 Node.js: 12.8.1 V8: 7.8.279.23-electron.0 OS: Windows_NT x64 10.0.19044 项目…

科技孵化制造蝶变:国际智造节上,群硕获评2023杰出数字化创新企业

论起2023年的热点有哪些&#xff1f;由ChatGPT掀起的智能科技浪潮&#xff0c;绝对是逃不开的话题。 6月16日&#xff0c;以“科技驱动&#xff0c;智造未来”为主题的2023国际智造节&#xff0c;在北京隆重举行。通过此次活动&#xff0c;在数字化领域深耕二十年的群硕软件&a…

K8s Kubectl 技巧集锦

kubectl 是 Kubernetes 的一个命令行管理工具&#xff0c;可用于 Kubernetes 上的应用部署和日常管理。本文列举了 9 个常见的 kubectl 命令&#xff0c;并总结了一些使用技巧&#xff0c;希望可以帮助系统管理员简化管理工作。 一、使用 Kubectl 查询、创建、编辑和删除资源 …

【C数据结构】循环队列_CyclicQueue

目录 循环队列_CyclicQueue 【1】循环队列 【1.1】循环队列的各个接口 【1.2】循环队列初始化 【1.3】循环队列初销毁 【1.4】循环队列插入 【1.5】循环队列删除 【1.6】循环队列获取头位置数据 【1.7】循环队列获取尾位置数据 【1.8】循环队列判满 【1.9】循环队列…

MMOE(Multi-gate Mixture-of-Experts)

1.前提和动机 随着推荐技术的发展&#xff0c;目前越来越多的推荐场景&#xff0c;往往并不是单独的优化一个指标&#xff0c;比如&#xff1a; 视频推荐领域&#xff1a;推荐排序任务不仅需要考虑到用户点击率、完播率&#xff0c;也需要考虑到一些满意度指标&#xff0c;例如…

优秀java实习报告范文5篇

优秀java实习报告范文(一) 一:实习介绍 1)实习题目 学生信息管理系统的设计与开发 2)实习目的 《Java程序设计》课程是电子商务专业的一门专业必修课程&#xff0c;特针对此课程进行为期三周的实践教学&#xff0c;旨在提升本专业学生对此课程中的理论知识的综合应用能力、提高…

微信研发体系下的分布式配置系统设计实践

腾小云导读 对很多的开发者而言&#xff0c;处理运营素材反复变更等需求场景不是一件轻松的事。开发者通常需要定制化地进行数据清理、格式转换和工具开发等等。在这个时候&#xff0c;建设分布式配置系统就显得尤为重要。本文旨在分析分布式配置系统的必要性、可行性及其关键…

SpringBoot的日志

SpringBoot的日志 &#x1f50e;日志是什么&#x1f50e;日志的作用&#x1f50e;日志级别日志级别的作用日志级别的分类日志级别的设置 &#x1f50e;打印日志打印日志具体内容划分 &#x1f50e;常用的日志框架为什么这样设计对比System.out.ptintln()与日志框架 &#x1f50…

Kubernetes那点事儿——暴露服务之Ingress

Kubernetes那点事儿——暴露服务之Ingress 前言一、ingress负载均衡器Ingress Controller路由规则Ingress 二、Ingress Controller三、案例 前言 在 k8s 集群中&#xff0c;如果我们将服务暴露出来&#xff0c;提供访问&#xff0c;可以使用Nodeport方式&#xff0c;但是Nodepo…

SIG Mesh协议学习

1. 简介 Bluetooth SIG组织在2017年7月17日发布了蓝牙Mesh标准. 蓝牙Mesh不同于传统Bluetooth Low Energy(BLE)协议的1对1, 1对多的通信方式, 它实现了多对多的通信. 这使得mesh网络中的各个节点之间可以相互通信. 蓝牙Mesh协议建立在BLE的物理层和链路层之上, 也就是说它可以…

MachXO2系列 FPGA LCMXO2-7000HC-4TG144C-可编程逻辑器件介绍

莱迪思深力科MachXO2系列LCMXO2-7000HC-4TG144C是高度可配置的串行逻辑器件&#xff08;PLD&#xff09;&#xff0c;具有低误差&#xff0c;具有和高系统集成等特性。MachXO2系列逻辑密度增加了3倍&#xff0c;嵌入存储增加了10倍&#xff0c;静态降低100倍&#xff0c;而成本…

动态获取项目根目录的两种方式

说明&#xff1a;如果项目上传的文件是存储在本地的&#xff0c;为了项目可以在其他电脑上也可能正常运行&#xff0c;可以将文件上传至项目的static文件夹里&#xff0c;并将文件路径保存至数据库中&#xff0c;前端显示时使用也可以通过该路径访问。要想获取到本地项目static…

如何评估大型语言模型(LLM)?

编者按&#xff1a;近期几乎每隔一段时间&#xff0c;就有新的大语言模型发布&#xff0c;但是当下仍然没有一个通用的标准来评估这些大型语言模型的质量&#xff0c;我们急需一个可靠的、综合的LLM评估框架。 本文说明了为什么我们需要一个全面的大模型评估框架&#xff0c;并…

DOCker安装(一)

DOCker的安装 1、简介 Docker使用客户端-服务器(C/S)架构模式&#xff0c;使用远程API来管理和创建Docker容器。 Docker容器通过Docker镜像来创建。 容器之间互不干扰 容器与镜像的关系类似于面向对象编程中的对象与类。 对象->容器 镜像->类 通过镜像来创建容器 …

如何用MarkDown轻松排版和格式化文档?

前言 大家好哦&#xff01;我写作用的语言是MarkDown&#xff0c;真的非常好用&#xff0c;在本文中我将一字字的教你如何使用这个轻量化&#xff0c;容易上手的轻标记语言。 1. 什么是MarkDown MarkDown是一种轻量级的标记语言&#xff0c;它可以在文本中通过简单的标记实现…