使用Node.js打造自己的Git版本控制系统
.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版本控制系统的更多相关文章
- 你要是还学不会,请提刀来见 Typora+PicGo+Gitee + node.js 打造个人高效稳定优雅图床
你要是还学不会,请提刀来见 Typora+PicGo+Gitee + node.js 打造个人高效稳定优雅图床 经过前面两弹的介绍,相信大家对图床都不陌生了吧, 但是小魔童觉得这样做法还是不方便,使用 ...
- git版本控制系统小白教程(上)
前言:本文主要介绍git版本控制系统的一些基础使用,适合小白入门,因为内容较多,会分为两部分进行分享. Git介绍 Git是目前世界上最先进的分布式版本控制系统.并且它是一个开源的分布式版本控制系 ...
- Google Code项目代码托管网站上Git版本控制系统使用简明教程
作为一个著名的在线项目代码托管网站,Google Code目前主要支持三种版本控制系统,分别为Git, Mercurial和 Subversion.Subversion即SVN相信大家都已经熟知了,这 ...
- Git版本控制系统VCS
Git版本控制系统VCS 一.版本控制系统基本情况说明 版本控制是一种记录一个或者若干个文件内容的变化,以便将来查阅特定版本修订情况的系统 1.作用 记录文件的所有历史变化 随时可回复到任何一个历史状 ...
- git版本控制系统重新认识
git 版本控制系统 目标:完全搞懂git分布式版本控制系统 搭建git版本控制系统 cvs集中化版本控制系统--集中式管理的服务器 git分布式版本控制系统--会将原始代码仓库镜像下来 新项目使用g ...
- git版本控制系统小白教程(下)
前言:本文主要介绍git版本控制系统的一些基础使用,适合小白入门,因为内容较多,会分为两部分进行分享,查看上部请点传送门. 删除文件 git删除文件一般有三种情况,第一种是在工作区修改了文件,但是 ...
- Node.js 打造实时多人游戏框架
在 Node.js 如火如荼发展的今天,我们已经可以用它来做各种各样的事情.前段时间UP主参加了极客松活动,在这次活动中我们意在做出一款让“低头族”能够更多交流的游戏,核心功能便是 Lan Party ...
- Mac升级Node.js和npm到最新版本指令
一.查看本机当前Node.js和npm版本 node -v npm -v 二.清除node.js的cache sudo npm cache clean -f 三.安装"n"版本管理 ...
- git版本控制系统更新
版本控制系统: 一.概念: 版本控制系统(Version Control System):是一种记录一个或若干文件内容变化,以便将来查阅特定版本修订情况的系统. 二.版本控制系统分类 1.本地版本控制 ...
- Git 版本控制系统的基本使用、常用操作
以Ubuntu16.04操作系统为例(其他系统类似),主要记录常用的.基本操作: 0. 安装Git 分散型版本控制系统(CVS): sudo apt-get install git 1. 初始化本地配 ...
随机推荐
- flutter3-trip仿携程酒店预订|Flutter3.27+Getx预约旅游酒店App程序
基于Flutter3.x+Dart3+GetX跨平台仿携程/飞猪旅行酒店客房预订查询app系统. flutter3_trip原创2025新版flutter3.27.1+dart3.6+getx+flu ...
- D pid(16916) tid(19140) 14:05:45 EdgeSnapFeature::PostExitSizeMove: WM_TWINVIEW_SHOW_GRIDLINES -> off
D pid(16916) tid(19140) 14:05:45 EdgeSnapFeature::PostExitSizeMove: WM_TWINVIEW_SHOW_GRIDLINES -> ...
- RedHat8密码复杂度策略配置
1.密码复杂度策略概念 在Linux系统中,确保用户密码的复杂度是提高系统安全性的重要措施之一.通过配置密码策略,可以强制用户使用强密码,从而降低被破解的风险.本文将详细介绍如何在 Linux 系统中 ...
- linux服务器压力/性能测试命令
linux命令下使用ab -c500 -n2000 http://www.test.cn 进行压力测试 在Linux命令行中,ab 是 Apache HTTP server benchmarki ...
- Selenium IDE 使用过程
这篇文章主要介绍如何成功搭建firefox与Selenium IDE环境及IDE使用过程,以登录功能介绍. Selenium IDE是Firefox浏览器的一款插件,实现网页的录制及回放,可以脚本导出 ...
- Win系统重装备忘
蒙德,致态的盘坏块激增,似乎损坏到了系统文件:屏幕截屏会卡,关机后直接该块硬盘内的文件内容回滚,出现驱动报错要求重启... 然后尝试了DiskGenuis迁移系统,PE模式不能用,热迁移后似乎正常分区 ...
- 解决nvm ls-remote 列表只出现iojs版本
前言 在 nvm 安装 node 时发现显示不存在此版本,使用 nvm ls-remote 查看可安装列表时发现,列表中只有 iojs $ nvm ls-remote iojs-v1.0.0 iojs ...
- Kubernetes的工作机制
云计算时代的操作系统 Kubernetes 是一个生产级别的容器编排平台和集群管理系统,能够创建.调度容器,监控.管理服务器. Kubernetes 的基本架构 操作系统的一个重要功能就是抽象,从繁琐 ...
- 『Plotly实战指南』--柱状图绘制基础篇
柱状图作为最基础的数据可视化形式之一,能直观展示不同类别数据的对比关系,适用于一下的场景: 比较不同类别之间的数据大小,如不同产品的销售额对比. 展示数据的分布情况,如各年龄段的人口数量分布. 分析时 ...
- BUUCTF---达芬奇的密码
题目 达芬奇隐藏在蒙娜丽莎中的数字列:1 233 3 2584 1346269 144 5 196418 21 1597 610 377 10946 89 514229 987 8 55 6765 2 ...