深入浅出:AST 技术的应用与实践
.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 } }.hljs-comment, .hljs-quote { color: rgba(212, 208, 171, 1) }
.hljs-deletion, .hljs-name, .hljs-regexp, .hljs-selector-class, .hljs-selector-id, .hljs-tag, .hljs-template-variable, .hljs-variable { color: rgba(255, 160, 122, 1) }
.hljs-built_in, .hljs-builtin-name, .hljs-link, .hljs-literal, .hljs-meta, .hljs-number, .hljs-params, .hljs-type { color: rgba(245, 171, 53, 1) }
.hljs-attribute { color: rgba(255, 215, 0, 1) }
.hljs-addition, .hljs-bullet, .hljs-string, .hljs-symbol { color: rgba(171, 227, 56, 1) }
.hljs-section, .hljs-title { color: rgba(0, 224, 224, 1) }
.hljs-keyword, .hljs-selector-tag { color: rgba(220, 198, 224, 1) }
.markdown-body pre, .markdown-body pre>code.hljs { background: rgba(43, 43, 43, 1); color: rgba(248, 248, 242, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }
@media screen and (-ms-high-contrast: active) { .hljs-addition, .hljs-attribute, .hljs-built_in, .hljs-builtin-name, .hljs-bullet, .hljs-comment, .hljs-link, .hljs-literal, .hljs-meta, .hljs-number, .hljs-params, .hljs-quote, .hljs-string, .hljs-symbol, .hljs-type { color: rgba(181, 213, 255, 1) } .hljs-keyword, .hljs-selector-tag { font-weight: 700 } }
随着前端技术的不断发展,AST 技术逐渐成为了越来越多开发者关注的一个热点话题。那么,什么是 AST 技术?为什么越来越多的开发者开始使用 AST 技术呢?本文将深入浅出地介绍 AST 技术的应用和实践,提供代码示例帮助读者更好地理解和应用 AST 技术。
什么是 AST 技术?
AST,即抽象语法树(Abstract Syntax Tree),是源代码中的一个抽象语法结构树状表示,它表示了代码的结构和逻辑。将源代码转换为 AST 可以让开发者更加方便地处理代码,比如:
- 代码分析和检查:开发者可以使用 AST 技术对代码进行语法分析、语义分析和错误检查等操作,从而更好地保证代码的质量和规范。
- 代码优化:开发者可以使用 AST 技术对代码进行优化,比如进行代码压缩、混淆等操作,以提高代码的执行效率和安全性。
- 编译器:开发者可以使用 AST 技术来构建自己的编译器,并用于将代码转换为其他形式的代码。
AST 技术的优点在于它可以在不损失代码功能的前提下,对代码进行修改和优化。开发者可以通过 AST 技术来改进代码的性能、可读性和可维护性。
AST 技术的应用
一、前端代码优化
代码优化是前端开发中的重要工作之一。使用 AST 技术可以对代码进行优化,从而提高代码的执行效率和安全性。
1.代码压缩
代码压缩是指对源代码进行压缩,以减少代码文件的大小,提高网页的加载速度。在储存和传输代码文件时,压缩技术可以减少文件大小、提高访问速度。这是因为压缩后的代码文件体积变小,可以更快地下载或加载,同时也可以消耗较少的网络流量。
下面是一个使用 AST 技术实现代码压缩的代码示例:
const parser = require('@babel/parser');
const traverse = require('@babel/traverse').default;
const { transformFromAstSync } = require('@babel/core');
const code = `
const add = (a, b) => a + b;
const result = add(1, 2);
`;
// 将源代码转换为 AST
const ast = parser.parse(code);
// 使用插件来对 AST 进行修改
traverse(ast, {
enter(path) {
if (path.isArrowFunctionExpression()) {
path.arrowFunctionToExpression();
}
},
});
// 将 AST 转换为压缩后的代码
const result = transformFromAstSync(ast).code;
console.log(result);
在上面的代码示例中,我们使用了 @babel/parser
和 @babel/traverse
这两个库来将源代码转换成 AST 并进行修改。在这个例子中,我们使用 arrowFunctionToExpression
插件将箭头函数表达式转换为函数表达式。
2.代码混淆
代码混淆是指对源代码进行混淆,以提高代码的安全性,防止代码被破解。在代码混淆的过程中,开发者会对代码进行压缩和重组,从而改变代码的结构和逻辑,使得破解者很难还原和理解代码。
下面是一个使用 AST 技术实现代码混淆的代码示例:
const obfuscator = require('javascript-obfuscator');
const { parse } = require('@babel/parser');
const code = `
function hello(name) {
console.log("Hello, " + name + "!");
}
`;
// 将源代码转换为 AST
const ast = parse(code);
// 将 AST 转换为混淆后的代码
const result = obfuscator.obfuscateFromAst(ast, {
compact: true,
controlFlowFlattening: true,
controlFlowFlatteningThreshold: 1,
numbersToExpressions: true,
simplify: true,
shuffleStringArray: true,
splitStrings: true,
stringArray: true,
stringArrayEncoding: 'base64',
});
console.log(result.getObfuscatedCode());
在上面的代码示例中,我们使用 javascript-obfuscator
这个库对源代码进行混淆。该库提供了多种混淆选项,比如镜像变量、使用随机命名等,这些选项可以在混淆时随意组合使用。
3.代码分离
代码分离是指将源代码拆分成若干个模块,以提高代码可维护性和复用性。使用 AST 技术可以对代码进行分离,从而可以更好地管理和维护代码。
下面是一个使用 AST 技术实现代码分离的代码示例:
const parser = require('@babel/parser');
const traverse = require('@babel/traverse').default;
const { transformFromAstSync } = require('@babel/core');
const code = `
import add from './add.js';
const result = add(1, 2);
`;
// 将源代码转换为 AST
const ast = parser.parse(code, { sourceType: 'module' });
// 使用插件来对 AST 进行修改
traverse(ast, {
ImportDeclaration(path) {
const source = path.get('source');
const moduleName = source.node.value;
const newModuleName = `./${moduleName}`;
source.replaceWith(
parser.parseExpression(`require('${newModuleName}')`)
);
},
});
// 将 AST 转换为分离后的代码
const result = transformFromAstSync(ast).code;
console.log(result);
在上面的代码示例中,我们使用了 @babel/parser
和 @babel/traverse
这两个库来将源代码转换成 AST 并进行修改。在这个例子中,我们使用 ImportDeclaration
插件将 ES6 的模块导入语句转换为 CommonJS 的 require
语句,从而实现了代码分离。
二、前端代码检查
代码检查是指对源代码进行静态分析,从而找出代码中的问题和错误。使用 AST 技术可以对代码进行检查,从而可以更好地保证代码的质量和规范。
1.代码风格检查
代码风格检查是指对源代码进行风格检查,并根据代码风格规范对代码进行自动修复,以提高代码的可读性和可维护性。使用 AST 技术可以对代码进行风格检查和自动修复。
下面是一个使用 AST 技术实现代码风格检查的代码示例:
const eslint = require('eslint');
const code = `
const a = 1
if(a === 1){
console.log("Hello World!");
}
`;
// 创建一个 ESLint 实例
const linter = new eslint.ESLint();
// 运行 ESLint,并输出结果
(async function() {
const results = await linter.lintText(code, {
fix: true,
});
const formatter = await linter.loadFormatter('stylish');
const resultText = formatter.format(results);
console.log(resultText);
})();
在上面的代码示例中,我们使用了 eslint
这个库对源代码进行了风格检查,并对代码进行了自动修复。可以看到,在运行 eslint
后,我们使用了 stylish
格式来将检查结果进行了输出。
2.代码安全检查
代码安全检查是指对源代码进行安全性检查,并根据安全漏洞修复代码,从而提高代码的安全性。使用 AST 技术可以对代码进行安全检查和修复。
下面是一个使用 AST 技术实现代码安全检查的代码示例:
const parser = require('@babel/parser');
const traverse = require('@babel/traverse').default;
const { transformFromAstSync } = require('@babel/core');
const code = `
function getUserData(data) {
const userId = data.userId;
return fetch(\`/getUserData?id=\${userId}\`);
}
`;
// 将源代码转换为 AST
const ast = parser.parse(code);
// 使用插件来对 AST 进行修改
traverse(ast, {
MemberExpression(path) {
const property = path.get('property');
if (property.isIdentifier({ name: 'fetch' })) {
path.replaceWith(
parser.parseExpression(`myCustomFetchFunction()`)
);
}
},
});
// 将 AST 转换为安全化后的代码
const result = transformFromAstSync(ast).code;
console.log(result);
在上面的代码示例中,我们使用了 @babel/parser
和 @babel/traverse
这两个库来将源代码转换成 AST 并进行修改。在这个例子中,我们使用 MemberExpression
插件将 fetch
函数调用替换为自定义的 myCustomFetchFunction
函数,从而提高代码的安全性。
三、代码编辑器和 IDE 插件
代码编辑器和 IDE 插件是前端开发中常用的工具。使用 AST 技术可以为代码编辑器和 IDE 插件提供更加强大的功能。
1.语法高亮
语法高亮是指对源代码中的不同部分进行不同的颜色标记,以提高代码的可读性。使用 AST 技术可以对代码进行语法分析,从而可以更好地实现语法高亮。
下面是一个使用 AST 技术实现语法高亮的代码示例:
const Prism = require('prismjs');
const code = `
function add(a, b) {
return a + b;
}
`;
// 创建一个词法分析器
const lexer = new Prism.Lexer({
javascript: Prism.languages.javascript,
});
// 进行词法分析,得到 tokens
const tokens = lexer.tokenize(code);
// 输出 tokens,实现语法高亮
console.log(tokens.map(token => token.type));
在上面的代码示例中,我们使用了 prismjs
这个库来实现语法高亮。通过创建一个词法分析器,我们可以对源代码进行词法分析,并得到符号表,从而实现语法高亮。
2.代码提示
代码提示是指在使用代码编辑器或 IDE 时,当用户输入代码时,自动为用户提供相关的代码建议。使用 AST 技术可以对代码进行语法分析,从而可以更好地实现代码提示。
下面是一个使用 AST 技术实现代码提示的代码示例:
const parser = require('@babel/parser');
const traverse = require('@babel/traverse').default;
const code = `
const users = getUserList();
for (const user of users) {
console.log(user.name);
}
`;
// 将源代码转换为 AST
const ast = parser.parse(code);
// 遍历 AST,收集标识符
const identifiers = [];
traverse(ast, {
Identifier(path) {
identifiers.push(path.node.name);
},
});
// 输出标识符,实现代码提示
console.log(identifiers);
在上面的代码示例中,我们使用了 @babel/parser
和 @babel/traverse
这两个库来将源代码转换成 AST 并进行遍历。在遍历的过程中,我们收集了代码中所有的标识符,并将其输出,实现了代码提示的功能。
AST 技术的实践
一、使用 Babel 转换源代码
Babel 是一个用于转换 JavaScript 代码的工具,使用 Babel 可以将 ECMAScript 2015+ 代码转换成向后兼容的 JavaScript 代码。Babel 或许是前端开发中最为常用的 AST 工具之一,下面是一个使用 Babel 转换源代码的示例:
const babel = require('@babel/core');
const code = `
const add = (a, b) => a + b;
const result = add(1, 2);
`;
const result = babel.transformSync(code, {
presets: ['@babel/preset-env'],
});
console.log(result.code);
在上面的代码示例中,我们使用了 @babel/core
这个库将源代码转换为向后兼容的 JavaScript 代码。在使用 babel.transformSync
函数时,我们传入了一个 presets
配置项,该配置项告诉 Babel 使用哪些插件将源代码转换为向后兼容的 JavaScript 代码。
二、使用 ESLint 进行代码检查
ESLint 是一个用于检查 JavaScript 代码的工具,使用 ESLint 可以对代码进行语法分析、代码风格检查和错误检查等操作。下面是一个使用 ESLint 进行代码检查的示例:
const eslint = require('eslint');
const code = `
const a = 1
if(a === 1){
console.log("Hello World!");
}
`;
const linter = new eslint.ESLint();
(async function() {
const results = await linter.lintText(code);
const formatter = await linter.loadFormatter('stylish');
const resultText = formatter.format(results);
console.log(resultText);
})();
在上面的代码示例中,我们使用了 eslint
这个库对源代码进行了检查。使用 ESLint
类可以创建一个 ESLint
实例,然后通过调用 lintText
函数对源代码进行检查。在检查完成后,我们使用 loadFormatter
函数来加载格式化工具,然后通过 format 函数将检查结果输出为指定的格式,这里我们使用了 stylish
格式。
三、使用 AST 技术实现自己的编译器
AST 技术可以用于构建自己的编译器。下面是一个使用 AST 技术实现自己的编译器的示例:
const parser = require('@babel/parser');
const traverse = require('@babel/traverse').default;
const code = `
const add = (a, b) => a + b;
const result = add(1, 2);
`;
// 将源代码转换为 AST
const ast = parser.parse(code);
// 使用插件来对 AST 进行修改
traverse(ast, {
enter(path) {
if (path.isArrowFunctionExpression()) {
path.arrowFunctionToExpression();
}
},
});
// 将 AST 进行代码生成
const output = require('@babel/generator').default(ast, {
comments: false,
jsescOption: {
quotes: 'single',
},
}).code;
console.log(output);
在上面的代码示例中,我们使用了 @babel/parser
将源代码转换为 AST,然后使用 @babel/traverse
对 AST 进行修改,并使用 @babel/generator
将 AST 生成为 JavaScript 代码。在这个例子中,我们使用了 arrowFunctionToExpression
插件将箭头函数表达式转换为函数表达式。
本文介绍了 AST 技术的应用和实践,包括前端代码优化、前端代码检查、代码编辑器和 IDE 插件等多个方面。可以看到,AST 技术在前端开发中有着广泛的应用,可以帮助我们实现很多复杂的功能。因此,学习和掌握 AST 技术对前端开发人员来说尤为重要。
AST 技术的学习可以从以下方面入手:
- 学习 AST 技术的基本概念和原理;
- 学习 AST 技术的相关工具,如 Babel、ESLint 等;
- 学习如何使用 AST 技术来进行前端代码优化、前端代码检查、代码编辑器和 IDE 插件等应用;
- 学习如何使用 AST 技术实现自己的编译器。
总之,AST 技术是一个非常重要的前端开发技术,掌握这个技术不仅可以提高我们的开发效率,还可以帮助我们提高代码的质量和可维护性。
深入浅出:AST 技术的应用与实践的更多相关文章
- 2017.2.9 深入浅出MyBatis技术原理与实践-第八章 MyBatis-Spring(二)-----配置文件详解
深入浅出MyBatis技术原理与实践-第八章 MyBatis-Spring(二) ------配置文件详解 8.2 MyBatis-Spring应用 8.2.1 概述 本文主要讲述通过注解配置MyBa ...
- 学习PHP爬虫--《Webbots、Spiders和Screen Scrapers:技术解析与应用实践(原书第2版)》
<Webbots.Spiders和Screen Scrapers:技术解析与应用实践(原书第2版)> 译者序 前言 第一部分 基础概念和技术 第1章 本书主要内容3 1.1 发现互联网的真 ...
- 20145226夏艺华 网络对抗技术EXP8 WEB基础实践
20145226夏艺华 网络对抗技术EXP8 WEB基础实践 实验问题回答 1.什么是表单? 表单在网页中主要负责数据采集功能.一个表单有三个基本组成部分: 表单标签:这里面包含了处理表单数据所用CG ...
- 2MyBatis入门--深入浅出MyBatis技术原理与实践(笔记)
什么是 MyBatis ? MyBatis 是支持定制化 SQL.存储过程以及高级映射的优秀的持久层框架.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis ...
- 3MyBatis配置--深入浅出MyBatis技术原理与实践(笔记)
XML 映射配置文件 configuration 配置 properties 属性 settings 设置 typeAliases 类型命名 typeHandlers 类型处理器 objectFact ...
- UI自动化技术在高德的实践
一.背景汽车导航作为ToB业务,需要满足不同汽车厂商在功能和风格上体现各自特色的需求.针对这种情况,传统的UI开发方式,基本上是一对一的特别定制.但是这种方式动辄就要500~600人日的工作量投入,成 ...
- 干货 | 京东技术中台的Flutter实践之路
在 2019 年,Flutter 推出了多个正式版本,支持的终端越来越多,使用的项目也越来越多.Flutter 正在经历从小范围尝鲜到大面积应用的过程,越来越多的研发团队加入到 Flutter 的学习 ...
- Java使用FreeMarker模版技术动态生成word实践
一.序言 在日常开发中,常常有动态word文件生成的需求,通过编制模版,然后动态修改word内容以组合成新的文件.报告单.请假单.发票页等都可以使用动态生成word来解决. 笔者总结归纳出通用技术要点 ...
- 《深入浅出MyBatis技术原理与实战》——7. 插件
在第6章讨论了四大运行对象的运行过程,在Configuration对象的创建方法里我们看到了MyBatis用责任链去封装它们. 7.1 插件接口 在MyBatis中使用插件,我们必须使用接口Inter ...
- 《深入浅出MyBatis技术原理与实战》——6. MyBatis的解析和运行原理
MyBatis的运行分为两大部分,第一部分是读取配置文件缓存到Configuration对象,用以创建SqlSessionFactory,第二部分是SqlSession的执行过程. 6.1 涉及的技术 ...
随机推荐
- KUKA库卡机器人常见故障都有哪些解决办法
如何解决机器人线路板维修查不出故障 在进行机器人电路板维修工作时,有时会遇到查不出故障原因的情况,对于这种故障维修业界称之为疑难杂症,这时维修人员该如何应对呢?在这里向朋友们介绍机器人维修一些解决的方 ...
- PIP 换源:提升 Python 包安装速度的秘诀
一.引言 在使用 Python 进行开发时,我们经常需要通过 pip 命令安装各种库和依赖.然而,默认的源可能会因为网络原因导致下载速度缓慢,影响开发效率.这时候,换源就成为了一个非常实用的技巧. 二 ...
- Typecho美化之网页底部增加好久不见的底部样式
好久不见之本站同款网站底部样式,效果见本站. 1.修改footer.php首先,在页脚文件footer.php文件的最下面放入以下代码: <!-- 好久不见 --> <div cla ...
- MySQL - [03] 数据库引擎
所有的数据库文件都存在data目录下,本质还是文件的存储. -- 建表时指定编码格式为中文utf-8 CREATE TABLE IF NOT EXISTS `student` ( `id` INT(4 ...
- docker - [09] 镜像详解
题记部分 一.镜像是什么 镜像是一种轻量级.可执行的独立软件包,用来打包软件运行环境和基于运行环境开发的软件,还包含运行某个软件所需的所有内容,包括代码.运行时.库.环境变量和配置文件. 如果得到 ...
- FormCreate中在事件中获取api
form-create中在事件中获取api FormCreate 是一个可以通过 JSON 生成具有动态渲染.数据收集.验证和提交功能的表单生成组件.支持5个UI框架,并且支持生成任何 Vue 组件. ...
- 添加xxx.so到环境变量里
点击查看代码 libxxx.so 文件位于 /usr/local/lib 目录下,你可以按照以下步骤操作: 创建配置文件: echo "/usr/local/lib" | sudo ...
- 多智能体粒子环境(Multi-Agent Particle Env)食用指南--从入门到入土
0.项目地址: 原地址:openai/multiagent-particle-envs: Code for a multi-agent particle environment used in the ...
- NumPy学习5
今天学习了11, NumPy数组元素增删改查NumPy 数组元素的增删改查操作,主要有以下方法:数组元素操作方法函数名称 描述说明resize 返回指定形状的新数组.append 将元素值添加到数组的 ...
- MaxKB+Ollama 离线部署
主题:在 Centos7 环境部署 MaxKB 以及 Ollama 实现基于离线大模型的的小助手调用. 选择离线部署的原因:原计划是打算直接使用 1Panel 进行 MaxKB 和 Ollama 一键 ...