@charset "UTF-8";
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(255, 255, 255, 1); color: rgba(0, 0, 0, 1) }
.hljs-comment, .hljs-quote, .hljs-variable { color: rgba(0, 128, 0, 1) }
.hljs-built_in, .hljs-keyword, .hljs-name, .hljs-selector-tag, .hljs-tag { color: rgba(0, 0, 255, 1) }
.hljs-addition, .hljs-attribute, .hljs-literal, .hljs-section, .hljs-string, .hljs-template-tag, .hljs-template-variable, .hljs-title, .hljs-type { color: rgba(163, 21, 21, 1) }
.hljs-deletion, .hljs-meta, .hljs-selector-attr, .hljs-selector-pseudo { color: rgba(43, 145, 175, 1) }
.hljs-doctag { color: rgba(128, 128, 128, 1) }
.hljs-attr { color: rgba(255, 0, 0, 1) }
.hljs-bullet, .hljs-link, .hljs-symbol { color: rgba(0, 176, 232, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }

最近都不知道写点啥了,已经挺久没更新了,也是因为最近比较忙,现在才抽出空来,之前的文章大部分都是给大家普及一些知识点以及技术片段,估计大家平时也很少会用到。

因此我决定教大家一些经常会接触的东西,比如git;

作为一个程序员,大家肯定经常使用Git来进行版本控制。但是你有没有想过自己动手实现一个Git系统呢?今天掌门人就来教大家如何使用Node.js来实现一个简易的Git版本控制系统,我们将其命名为GitX

创建代码仓库

首先,我们需要创建一个新目录来作为我们的代码仓库,并初始化npm项目:

mkdir gitx
cd gitx
npm init -y

初始化项目(gitx init)

初始化仓库:我们需要在仓库根目录下创建一个.gitx目录来存放我们的版本控制信息。

// init.js
const fs = require('fs');
const path = require('path'); const initRepo = () => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (fs.existsSync(gitxPath)) {
console.log('仓库已存在');
return;
}
fs.mkdirSync(gitxPath);
console.log('初始化仓库成功');
} module.exports = initRepo;

实现跟踪文件变化(gitx add)

跟踪文件变化:跟踪文件变化的功能,我们需要监听文件的变化,并将变化记录下来。

// track.js
const fs = require('fs');
const path = require('path');
const crypto = require('crypto'); const trackFiles = (files) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} files.forEach(file => {
const filePath = path.join(process.cwd(), file);
if (!fs.existsSync(filePath)) {
console.log(`文件${file}不存在`);
return;
}
const fileContent = fs.readFileSync(filePath);
const fileHash = crypto.createHash('sha1').update(fileContent).digest('hex');
const trackPath = path.join(gitxPath, fileHash);
fs.writeFileSync(trackPath, fileContent);
console.log(`文件${file}变化已跟踪`);
});
} module.exports = trackFiles;

提交更新(gitx commit)

提交更新:是版本控制系统中非常重要的功能之一,我们需要将跟踪的文件变化提交到仓库中。

// commit.js
const fs = require('fs');
const path = require('path');
const crypto = require('crypto'); const commitChanges = (message) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const commitHash = crypto.createHash('sha1').update(Date.now().toString()).digest('hex');
const commitPath = path.join(gitxPath, commitHash);
fs.mkdirSync(commitPath); const trackedFiles = fs.readdirSync(gitxPath).filter(file => fs.statSync(path.join(gitxPath, file)).isFile());
trackedFiles.forEach(file => {
const trackedFilePath = path.join(gitxPath, file);
const newFilePath = path.join(commitPath, file);
fs.copyFileSync(trackedFilePath, newFilePath);
}); const commitInfo = {
message,
timestamp: new Date(),
files: trackedFiles
};
fs.writeFileSync(path.join(commitPath, 'info.json'), JSON.stringify(commitInfo, null, 2)); console.log(`提交成功,提交信息:${message}`);
} module.exports = commitChanges;

分支管理(gitx branch)

分支管理:我们需要实现创建、删除、查看和切换分支的功能。

// branch.js
const fs = require('fs');
const path = require('path'); const createBranch = (name) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const branchPath = path.join(gitxPath, 'branches', name);
if (fs.existsSync(branchPath)) {
console.log(`分支${name}已存在`);
return;
} fs.mkdirSync(branchPath, { recursive: true });
console.log(`分支${name}创建成功`);
} const deleteBranch = (name) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if(!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const branchPath = path.join(gitxPath, 'branches', name);
if (!fs.existsSync(branchPath)) {
console.log(`分支${name}不存在`);
return;
} fs.rmdirSync(branchPath, { recursive: true });
console.log(`分支${name}删除成功`);
} const listBranches = () => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const branchesPath = path.join(gitxPath, 'branches');
const branches = fs.readdirSync(branchesPath); console.log('分支列表:');
branches.forEach(branch => console.log(branch));
} const switchBranch = (name) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const branchesPath = path.join(gitxPath, 'branches');
if (!fs.existsSync(path.join(branchesPath, name))) {
console.log(`分支${name}不存在`);
return;
}
console.log(`成功切换到分支${name}`);
} module.exports = {
createBranch,
deleteBranch,
listBranches,
switchBranch
};

合并分支(gitx merge)

合并分支:是Git中最为复杂的操作之一,我们将实现一个简单版本的合并功能

// merge.js
const fs = require('fs');
const path = require('path'); const mergeBranch = (sourceBranch, targetBranch) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const sourceBranchPath = path.join(gitxPath, 'branches', sourceBranch);
const targetBranchPath = path.join(gitxPath, 'branches', targetBranch);
if (!fs.existsSync(sourceBranchPath) || !fs.existsSync(targetBranchPath)) {
console.log(`分支不存在`);
return;
} const sourceFiles = fs.readdirSync(sourceBranchPath);
const targetFiles = fs.readdirSync(targetBranchPath); sourceFiles.forEach(file => {
if (!targetFiles.includes(file)) {
const filePath = path.join(sourceBranchPath, file);
fs.copyFileSync(filePath, path.join(targetBranchPath, file));
}
}); console.log(`合并分支${sourceBranch}到${targetBranch}成功`);
} module.exports = mergeBranch;

历史记录(gitx log)

历史记录:查看commit的历史记录以了解版本间的变化。

// log.js
const fs = require('fs');
const path = require('path'); const viewCommitHistory = () => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const commits = fs.readdirSync(gitxPath).filter(file => fs.statSync(path.join(gitxPath, file)).isDirectory());
commits.forEach(commitHash => {
const commitPath = path.join(gitxPath, commitHash, 'info.json');
if (fs.existsSync(commitPath)) {
const commitInfo = JSON.parse(fs.readFileSync(commitPath));
console.log(`提交哈希值:${commitHash}`);
console.log(`提交信息:${commitInfo.message}`);
console.log(`提交时间:${commitInfo.timestamp}`);
console.log('涉及文件:');
commitInfo.files.forEach(file => {
console.log(`- ${file}`);
});
console.log('-------------------------------------');
}
});
} module.exports = viewCommitHistory;

检出版本(gitx checkout version)

检出版本:检出版本,允许用户切换到不同的版本或分支

// checkout.js
const fs = require('fs');
const path = require('path'); const checkoutVersion = (version) => {
const gitxPath = path.join(process.cwd(), '.gitx');
if (!fs.existsSync(gitxPath)) {
console.log('仓库未初始化');
return;
} const versionPath = path.join(gitxPath, version);
if (!fs.existsSync(versionPath)) {
console.log(`版本${version}不存在`);
return;
} const files = fs.readdirSync(versionPath);
files.forEach(file => {
const filePath = path.join(versionPath, file);
fs.copyFileSync(filePath, path.join(process.cwd(), file)); });
console.log(`检出版本${version}成功`);
}
}
module.exports = checkoutVersion;

搭建入口

现在我们已经实现了所有的基础功能,我们可以创建一个入口文件来集中管理我们的命令。

// index.js
const program = require('commander'); const initRepo = require('./init');
const trackFiles = require('./track');
const commitChanges = require('./commit');
const { createBranch, deleteBranch, listBranches, switchBranch } = require('./branch');
const mergeBranch = require('./merge');
const viewCommitHistory = require('./log');
const checkoutVersion = require('./checkout'); program
.command('init')
.description('初始化一个新的gitx仓库')
.action(initRepo); program
.command('track <files...>')
.description('跟踪指定的文件变化')
.action(trackFiles); program
.command('commit <message>')
.description('提交变化到仓库')
.action(commitChanges); program
.command('branch <command> [name]')
.description('创建、删除、查看或切换分支')
.action((cmd, name) => {
switch (cmd) {
case 'create':
createBranch(name);
break;
case 'delete':
deleteBranch(name);
break;
case 'list':
listBranches();
break;
case 'switch':
switchBranch(name);
break;
default:
console.log('未知的分支命令');
}
}); program
.command('merge <source> <target>')
.description('合并分支')
.action(mergeBranch); program
.command('log')
.description('查看提交的历史记录')
.action(viewCommitHistory); program
.command('checkout <version>')
.description('检出指定版本或分支')
.action(checkoutVersion); program.parse(process.argv);

使用测试

现在我们可以通过Node.js来运行我们的命令。这样我们就实现了一个简易的Git版本控制系统GitX。以使用如下命令来测试各功能:

node index.js init
node index.js track file1.txt file2.txt
node index.js commit 'init commit'
node index.js branch create feature1
node index.js branch switch feature1
node index.js merge master feature1
node index.js log
node index.js checkout feature1

要在命令行直接使用gitx init这种命令格式,我们需要将你的Node.js应用程序发布为全局npm包,并设置bin字段在package.json中。下面是步骤详解:

  • 在你的package.json文件中,添加一个bin字段,该字段是一个对象,键是你希望用户输入的命令名称,值是该命令对应的文件路径。
{
"name": "gitx",
"version": "0.0.1",
"bin": {
"gitx": "./index.js"
},
//...
}
  • index.js的开头,添加一个shebang行来指定脚本的解释程序。这行代码告诉系统这个脚本应当使用Node.js执行。
#!/usr/bin/env node

// 其余的代码
  • 给你的index.js文件加上可执行权限,通过运行下面的命令:
chmod +x index.js
  • 确保你的项目中有正确的package.json文件,并且所有必要的依赖项都已包含在内。
    在项目根目录下运行下面的命令,将你的·npm包链接到全局模块,这样就可以通过命令行在任意位置使用了:
npm link
  • 这样就可以在命令行中直接使用gitx init来运行我们自己写的gitx了。

发布到npm

当然了,如果你想发布到npm中,让别人也用起来,那就需要发布到npm中,这需要你有npm的账号,需要使用npm发布包的命令:

npm publish

发布完成后使用,就可以通过npm install -g gitx来全局安装你的gitx命令了

总结

至此,你就完成了一个简易的git版本控制系统,当然了,现在我们写的还很粗糙,大家要是有兴趣的话,可以参考一下git的功能,然后在完善一下这个demo,开发出属于自己的版本控制系统。

使用Node.js打造自己的Git版本控制系统的更多相关文章

  1. 你要是还学不会,请提刀来见 Typora+PicGo+Gitee + node.js 打造个人高效稳定优雅图床

    你要是还学不会,请提刀来见 Typora+PicGo+Gitee + node.js 打造个人高效稳定优雅图床 经过前面两弹的介绍,相信大家对图床都不陌生了吧, 但是小魔童觉得这样做法还是不方便,使用 ...

  2. git版本控制系统小白教程(上)

    前言:本文主要介绍git版本控制系统的一些基础使用,适合小白入门,因为内容较多,会分为两部分进行分享. Git介绍 ​ Git是目前世界上最先进的分布式版本控制系统.并且它是一个开源的分布式版本控制系 ...

  3. Google Code项目代码托管网站上Git版本控制系统使用简明教程

    作为一个著名的在线项目代码托管网站,Google Code目前主要支持三种版本控制系统,分别为Git, Mercurial和 Subversion.Subversion即SVN相信大家都已经熟知了,这 ...

  4. Git版本控制系统VCS

    Git版本控制系统VCS 一.版本控制系统基本情况说明 版本控制是一种记录一个或者若干个文件内容的变化,以便将来查阅特定版本修订情况的系统 1.作用 记录文件的所有历史变化 随时可回复到任何一个历史状 ...

  5. git版本控制系统重新认识

    git 版本控制系统 目标:完全搞懂git分布式版本控制系统 搭建git版本控制系统 cvs集中化版本控制系统--集中式管理的服务器 git分布式版本控制系统--会将原始代码仓库镜像下来 新项目使用g ...

  6. git版本控制系统小白教程(下)

    前言:本文主要介绍git版本控制系统的一些基础使用,适合小白入门,因为内容较多,会分为两部分进行分享,查看上部请点传送门. 删除文件 ​ git删除文件一般有三种情况,第一种是在工作区修改了文件,但是 ...

  7. Node.js 打造实时多人游戏框架

    在 Node.js 如火如荼发展的今天,我们已经可以用它来做各种各样的事情.前段时间UP主参加了极客松活动,在这次活动中我们意在做出一款让“低头族”能够更多交流的游戏,核心功能便是 Lan Party ...

  8. Mac升级Node.js和npm到最新版本指令

    一.查看本机当前Node.js和npm版本 node -v npm -v 二.清除node.js的cache sudo npm cache clean -f 三.安装"n"版本管理 ...

  9. git版本控制系统更新

    版本控制系统: 一.概念: 版本控制系统(Version Control System):是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统. 二.版本控制系统分类 1.本地版本控制 ...

  10. Git 版本控制系统的基本使用、常用操作

    以Ubuntu16.04操作系统为例(其他系统类似),主要记录常用的.基本操作: 0. 安装Git 分散型版本控制系统(CVS): sudo apt-get install git 1. 初始化本地配 ...

随机推荐

  1. Thymeleaf 在页面中直接显示内容

    Thymeleaf 在页面中直接显示内容   一般情况下 Thymeleaf 模板要输出变量需要在某个标签中(如<div>.<span>)写th:text等属性来实现.但有时我 ...

  2. Android ADB 使用笔记

    ADB 工作原理 当启动某个adb客户端时,该客户端会先检查是否有adb服务器正在运行,如果没有则启动服务器进程.服务器会在启动后与本地TCP端口 5037 绑定,并监听adb客户端 发出的命令. 服 ...

  3. Scala重写构造函数

    package com.wyh.scala.classData object Demo4Class { def main(args: Array[String]): Unit = { val stu ...

  4. SQLSugar 支持 TDengine 超级表的使用指南

    TDengine 是一款高性能.分布式的时序数据库,广泛应用于物联网.工业互联网等领域.其核心概念之一是超级表(Super Table),它类似于传统数据库中的表结构模板,允许用户通过标签(Tag)动 ...

  5. DeepSeek 全套资料pdf合集免费下载(持续更新)

    有很多朋友都关注DeepSeek相关使用的教程资料,本站也一直持续分享DeepSeek 学习相关的pdf资料,由于比较零散,这篇文章主要就是做一个汇总,并且持续更新,让大家可以及时获取下载最新的相关D ...

  6. 如何学好.net core?

    https://www.zhihu.com/question/348740859/answer/842656513

  7. Python字符串前缀u、r、b、f含义(转)

    1.字符串前加 u 例子: u"字符串中有中文" 含义: 前缀u表示该字符串是unicode编码,Python2中用,用在含有中文字符的字符串前,防止因为编码问题,导致中文出现乱码 ...

  8. Windows下Cmake+VS2017+OpenSceneGraph环境搭建及采坑

    一.资料准备 1.OSG源码官网下载地址:http://www.openscenegraph.org/index.php/download-section/stable-releases或者OSGCh ...

  9. TaskPyro:一个轻量级的 Python 任务调度和爬虫管理平台

    前言 推荐一款本人在使用的Python爬虫管理平台,亲测不错!!! TaskPyro 是什么? TaskPyro 是一个轻量级的 Python 任务调度平台,专注于提供简单易用的任务管理和爬虫调度解决 ...

  10. BUUCTF---达芬奇的密码

    题目 达芬奇隐藏在蒙娜丽莎中的数字列:1 233 3 2584 1346269 144 5 196418 21 1597 610 377 10946 89 514229 987 8 55 6765 2 ...