前端技术
HTML
CSS
Javascript
前端框架和UI库
VUE
ReactJS
AngularJS
JQuery
NodeJS
JSON
Element-UI
Bootstrap
Material UI
服务端和客户端
Java
Python
PHP
Golang
Scala
Kotlin
Groovy
Ruby
Lua
.net
c#
c++
后端WEB和工程框架
SpringBoot
SpringCloud
Struts2
MyBatis
Hibernate
Tornado
Beego
Go-Spring
Go Gin
Go Iris
Dubbo
HessianRPC
Maven
Gradle
数据库
MySQL
Oracle
Mongo
中间件与web容器
Redis
MemCache
Etcd
Cassandra
Kafka
RabbitMQ
RocketMQ
ActiveMQ
Nacos
Consul
Tomcat
Nginx
Netty
大数据技术
Hive
Impala
ClickHouse
DorisDB
Greenplum
PostgreSQL
HBase
Kylin
Hadoop
Apache Pig
ZooKeeper
SeaTunnel
Sqoop
Datax
Flink
Spark
Mahout
数据搜索与日志
ElasticSearch
Apache Lucene
Apache Solr
Kibana
Logstash
数据可视化与OLAP
Apache Atlas
Superset
Saiku
Tesseract
系统与容器
Linux
Shell
Docker
Kubernetes
[Docker中自动创建本地挂载目录机制]的搜索结果
这里是文章列表。热门标签的颜色随机变换,标签颜色没有特殊含义。
点击某个标签可搜索标签相关的文章。
点击某个标签可搜索标签相关的文章。
转载文章
... 进入snowboy目录,执行以下命令,进行编译 cd /snowboy/swig/Python3make 得到了编译好的文件_snowboydetect.so 新建自己文件夹,将snowboy/example/Python3下的文件全复制到自己文件夹下,并将上一步编译后得到的_snowboydetect.so放到自己的文件夹中。 9、生成自己的唤醒词 训练模型:参考https://github.com/Kitt-AI/snowboy/ 10、将自己的模型.pmdl放到自己创建的文件夹snowboy里。 11、使用以下代码运行 注意:需要将官方案例中的 snowboydecoder.py 文件修改一下,把from . import snowboydetect 改为 import snowboydetect然后再运行。并将编译后的swig/Python3目录下的snowboydetect.py复制到自己的目录中。 python3 demo.py .pmdl 听到叮的一声,代表成功了。 完整参考文档:http://docs.kitt.ai/snowboy/downloads 本篇文章为转载内容。原文链接:https://blog.csdn.net/qq_43556844/article/details/113617602。 该文由互联网用户投稿提供,文中观点代表作者本人意见,并不代表本站的立场。 作为信息平台,本站仅提供文章转载服务,并不拥有其所有权,也不对文章内容的真实性、准确性和合法性承担责任。 如发现本文存在侵权、违法、违规或事实不符的情况,请及时联系我们,我们将第一时间进行核实并删除相应内容。
2023-03-05 08:57:02
123
转载
HTML
...能超级强大,而且插件机制灵活得不得了,这让我们能够轻轻松松应对各种千奇百怪、复杂的构建需求,一点儿也不费劲儿。今天,咱们要聊一聊一个实际操作的问题,就是在用 webpack --watch 实时监控文件变动并自动重新编译之后,怎么才能顺手牵羊地执行一个我们自定义的回调函数,把部分文件悄无声息地搬到我们指定的目录里去。这个功能在我们日常开发里头,尤其给力。比如当你需要同步更新那些静态资源、模板文件啥的,它就能派上大用场,超级实用嘞! 1. 理解webpack-watch模式 首先,我们需要理解 webpack --watch 命令的作用。当你在项目根目录运行 webpack --watch 时,webpack 将持续监听你的源代码文件,一旦检测到有改动,它会立即重新进行编译打包。这是一种实时反馈开发成果的高效工作模式。 2. 使用webpack插件实现回调功能 webpack 的强大之处在于它的插件系统。我们可以编写自定义插件来扩展其功能。下面,我们将创建一个自定义webpack插件,用于在每次编译完成后执行文件拷贝操作。 javascript class CopyAfterCompilePlugin { constructor(options) { this.options = options || {}; } apply(compiler) { compiler.hooks.done.tap('CopyAfterCompilePlugin', (stats) => { if (!stats.hasErrors()) { const { copyFrom, copyTo } = this.options; // 这里假设copyFrom和copyTo是待拷贝文件和目标路径 fs.copyFileSync(copyFrom, copyTo); console.log(已成功将${copyFrom}拷贝至${copyTo}); } }); } } // 在webpack配置文件中引入并使用该插件 const CopyWebpackPlugin = require('./CopyAfterCompilePlugin'); module.exports = { // ... 其他webpack配置项 plugins: [ new CopyWebpackPlugin({ copyFrom: 'src/assets/myfile.js', copyTo: 'dist/static/myfile.js' }), ], }; 上述代码中,我们定义了一个名为 CopyAfterCompilePlugin 的webpack插件,它会在编译过程结束后触发 done 钩子,并执行文件拷贝操作。这里使用了 Node.js 的 fs 模块提供的 copyFileSync 方法进行文件拷贝。 3. 插件应用与思考 在实际开发中,你可能需要拷贝多个文件或整个目录,这时可以通过遍历文件列表或者递归调用 copyFileSync 来实现。同时,为了提高健壮性,可以增加错误处理逻辑,确保拷贝失败时能给出友好的提示信息。 通过这种方式,我们巧妙地利用了webpack的生命周期钩子,实现了编译完成后的自动化文件管理任务。这种做法,可不光是让手动操作变得省心省力,工作效率嗖嗖往上升,更重要的是,它让构建流程变得更聪明、更自动化了。就好比给生产线装上了智能小助手,让webpack插件系统那灵活多变、随时拓展的特性展现得淋漓尽致。 总结一下,面对“webpack --watch 编译完成之后执行一个callback,将部分文件拷贝到指定目录”的需求,通过编写自定义webpack插件,我们可以轻松解决这个问题,这也是前端工程化实践中的一个小技巧,值得我们在日常开发中加以运用和探索。当然啦,每个项目的个性化需求肯定是各不相同的,所以呢,咱们就可以在这个基础上灵活变通,根据实际情况来个“私人订制”,把咱们的构建过程打磨得更贴合项目的独特需求,让每一个环节都充满浓浓的人情味儿,更有温度。
2023-12-07 22:55:37
690
月影清风_
NodeJS
...心组件,其对垃圾回收机制的持续优化对JavaScript开发者至关重要。最近,V8团队发布了新版本更新,引入了更智能的内存管理策略和更高效的垃圾回收算法,这有助于减少内存泄漏的可能性,并提高大型应用程序的性能表现。 同时,Node.js社区也在不断推出新的工具和服务来帮助开发者更好地进行内存分析和优化。诸如Node.js内置的process.memoryUsage() API、第三方模块如memory-leak-detector等工具,可以帮助开发者实时监控应用内存使用情况,快速定位潜在的内存泄漏问题。 此外,针对Node.js的长期运行服务场景,有专家建议采用最新的架构模式,比如利用worker_threads或多进程模型避免长时间运行任务导致的内存积压,或结合容器化技术(如Docker)实现资源限制与自动重启策略,以从系统层面防止内存泄漏带来的影响。 综上所述,在实际开发中,紧跟JavaScript引擎的演进步伐,掌握并运用最新的内存管理工具与策略,将有助于我们打造更为健壮且高性能的Node.js应用。
2023-12-25 21:40:06
75
星河万里-t
Gradle
...助开发者高效地管理和自动化项目构建流程。在这篇文章里,我们要好好唠一唠在用Gradle打包项目时,如何稳稳地把依赖包给正确塞进去这个核心环节。咱不仅会摆出一堆实用的代码实例,还会带着大家伙儿一起脑洞大开,进行一番深度探索和思考。 1. 理解Gradle依赖管理 首先,我们需要理解Gradle依赖管理的基本原理。Gradle依赖可以分为两种类型:本地依赖和远程依赖。本地依赖这个概念,就像是你项目里的“自给自足小菜园”,通常是指那些项目内部或者在你电脑本地文件系统中的jar包,它们就在你触手可及的地方,随用随取。而远程依赖呢,就好比是“远方的超市”,你需要从Maven仓库、Ivy仓库或者其他类似的远程仓库中把这些依赖项下载下来才能使用。这就像是你开车去超市采购食材一样,虽然不是家门口就有,但种类丰富,能满足更多样的需求。在实际项目中,我们更多时候是处理远程依赖。 例如,在Gradle脚本(build.gradle)中声明一个远程依赖,如添加对spring-boot-starter-web的依赖: groovy dependencies { implementation 'org.springframework.boot:spring-boot-starter-web:2.5.0' } 上述代码中,implementation是配置作用域,用于指定该依赖在编译和运行时的行为;'org.springframework.boot:spring-boot-starter-web:2.5.0'则遵循“group:module:version”的格式,分别表示组织名、模块名和版本号。 2. 配置依赖源与仓库 为了能够成功下载远程依赖,需要在Gradle脚本中配置依赖源(Repository)。一般来说,Gradle这家伙默认会先去Maven Central这个大仓库里找你需要的依赖项。但如果它发现你要的东西在这个仓库里找不到的话,你就得告诉它其他可以淘宝的地方,也就是添加其他的仓库地址啦。以下是如何添加JCenter仓库的例子: groovy repositories { mavenCentral() jcenter() // 或者maven { url 'https://jcenter.bintray.com/' } } 3. 特殊依赖处理 传递依赖与排除依赖 - 传递依赖:当你直接依赖某个库时,Gradle也会自动引入该库的所有依赖项(即传递依赖)。这虽然方便,但也可能带来版本冲突的问题。此时,Gradle允许你查看并管理这些传递依赖: groovy configurations.compileClasspath.resolvedConfiguration.resolvedArtifacts.each { artifact -> println "Dependency: ${artifact.name} - ${artifact.moduleVersion.id}" } - 排除依赖:对于不希望引入的传递依赖,可以通过exclude关键字来排除: groovy dependencies { implementation('com.example.library:A') { exclude group: 'com.example', module: 'B' } } 这段代码表示在引入A库的同时,明确排除掉来自同一组织的B模块。 4. 打包时包含依赖 当使用Gradle打包项目(如创建可执行的jar/war文件)时,确保所有依赖都被正确包含至关重要。Gradle提供了多种插件支持这种需求,比如在Spring Boot项目中,我们可以使用bootJar或bootWar任务: groovy plugins { id 'org.springframework.boot' version '2.5.0' } jar { archiveBaseName = 'my-project' archiveVersion = '1.0.0' } task bootJar(type: BootJar) { classifier = 'boot' } 在这个例子中,BootJar任务会自动将所有必需的依赖项打入到生成的jar文件中,使得应用具备自包含、独立运行的能力。 总结来说,Gradle打包时正确包含依赖包是一个涉及依赖声明、仓库配置以及特殊依赖处理的过程。经过对Gradle依赖管理机制的深入理解和亲手实践,我们不仅能够轻而易举地搞定那些恼人的依赖问题,更能进一步把项目构建过程玩转得溜溜的,从而大大提升开发效率,让工作效率飞起来。同时,在不断摸爬滚打、亲自上手实践的过程中,我们越发能感受到Gradle设计的超级灵活性和满满的人性化关怀,这也是为啥众多开发者对它爱得深沉,情有独钟的原因所在。
2023-12-14 21:36:07
336
柳暗花明又一村_
Go Iris
...;=1.11,Go会自动将源码存放在用户主目录下的go文件夹中 - (2)设置GOPROXY(可选):在国内网络环境下,为了加速依赖包的下载,通常建议设置GOPROXY代理。 bash export GOPROXY=https://goproxy.cn,direct 2. 安装Iris 当准备工作完成后,即可开始安装Iris。在终端输入以下命令进行安装: bash go get -u github.com/kataras/iris/v12@latest 问题1:安装失败或超时 有时,由于网络状况或其他原因,你可能会遇到安装超时或者失败的情况。这时候,请尝试以下解决办法: - (3)检查网络连接:确保网络通畅,如需可更换稳定的网络环境。 - (4)重新安装并清除缓存:有时候,Go的模块缓存可能导致问题,可以先清理缓存再尝试安装。 bash go clean -modcache go get -u github.com/kataras/iris/v12@latest 3. 使用Iris创建项目 安装完成后,让我们通过一段简单的代码实例来验证Iris是否正常工作: go package main import ( "github.com/kataras/iris/v12" ) func main() { app := iris.New() // 设置默认路由 app.Get("/", func(ctx iris.Context) { ctx.HTML(" Welcome to Iris! ") }) // 启动服务器监听8080端口 app.Listen(":8080") } 问题2:运行程序时报错找不到Iris包 如果在运行上述代码时遇到找不到Iris包的错误,这通常是由于Go环境路径配置不正确导致的。确认go.mod文件中是否包含正确的Iris依赖信息,若没有,请执行如下命令添加依赖: bash cd your_project_directory go mod tidy 以上就是关于Go Iris安装过程中可能出现的问题以及对应的解决方法。安装与配置虽看似琐碎,但却是构建强大应用的基础。希望这些分享能帮助你在探索Go Iris的路上少走弯路,顺利开启高效编程之旅。接下来,尽情享受Iris带来的极致性能与便捷开发体验吧!
2023-07-12 20:34:37
347
山涧溪流
Struts2
...ruts2框架在尝试创建指定的Action类时遇到了点状况。就像这次,它正努力生成一个名叫com.example.MyAction的家伙,结果却不那么顺利。这不仅影响到我们的业务逻辑执行,也阻碍了页面跳转等一系列交互过程。这篇东西,咱们会手把手地通过实实在在的代码实例,一起抽丝剥茧,探究这个问题背后的真相,同时还会给你献上一些实用的解决妙招。 2. 问题剖析 情景还原 假设你正在使用Struts2构建一个用户登录功能,并定义了一个处理登录请求的Action类MyAction: java package com.example; public class MyAction extends ActionSupport { private String username; private String password; // Getter and Setter methods for username and password... @Override public String execute() throws Exception { // Your login logic here... return "success"; } } 然后在struts.xml配置文件中映射该Action: xml /success.jsp 当用户发起登录请求访问login.action时,如果出现“Unable to instantiate action”错误,意味着Struts2在尝试创建MyAction实例时出现了异常。 3. 原因分析 导致此类错误的原因可能有以下几点: - Action类未正确编译或部署:确保你的Action类已经被成功编译并且包含在WEB-INF/classes目录下,或者被正确的打包到WAR文件中。 - Action类没有默认构造函数:Struts2通过反射机制来创建Action对象,所以必须存在无参数的构造函数。 java // 正确示例 - 提供默认构造函数 public class MyAction extends ActionSupport { public MyAction() { // ... } // 其他代码... } - 依赖注入问题:如果你在Action类中使用了@Autowired等注解进行依赖注入,但在Spring容器还未完全初始化时就尝试实例化Action,也可能引发此问题。 - 类路径问题:检查你的类路径设置是否正确,确保Struts2能找到并加载对应的Action类。 4. 解决方案 针对上述原因,我们可以采取如下措施: (1) 检查编译和部署情况 确保你的Java源码已成功编译并部署到正确的目录结构中。 (2) 添加默认构造函数 无论你的Action类是否有自定义构造函数,都应添加一个默认构造函数以满足Struts2的实例化需求。 (3) 确保依赖注入顺序 如果是Spring与Struts2整合的问题,需要调整配置以保证Spring容器在Struts2开始实例化Action之前完成初始化。 (4) 核对类路径 确认web应用的类路径设置正确无误,确保能够找到并加载到com.example.MyAction类。 5. 总结与探讨 遇到“Unable to instantiate action”这类错误时,切勿慌乱,它通常是由于一些基础设置或编码规范问题所引起的。作为一个开发者,在我们每天敲代码的过程中,真的得对这些问题上点心,就像侦探破案一样,得仔仔细细地排查、调试。这样咱们才能真正摸清Struts2框架是怎么工作的,把它玩转起来,以后类似的错误才不会找上门来。同时呢,不断回顾、归纳总结这些经验教训,并且乐于分享给大伙儿,这对我们个人技术能力的提升,以及整个团队协作效率的提高,那可是大有裨益,可以说帮助不要太大!让我们携手共进,在实践中深化对Struts2框架的理解,共同面对并解决各种技术挑战!
2023-04-28 14:54:56
67
寂静森林
Flink
...码示例帮助你理解这一机制。 1. Apache Flink 批流一体的统一计算引擎 (1)Flink的设计哲学 Apache Flink的核心理念是将批视为一种特殊的流——有限流,从而实现了一种基于流处理的架构去同时处理无限流数据和有界数据集。这种设计简直让开发者们乐开了花,从此以后再也不用头疼选择哪种处理模型了。无论是对付那些堆积如山的历史数据,还是实时流动的数据流,都能轻松驾驭,只需要同一套API就能搞定编写工作。这样一来,不仅开发效率噌噌噌地往上飙,连资源利用率也得到了前所未有的提升,真可谓是一举两得的超级福利! (2)批流一体的实现原理 在Flink中,所有的数据都被视作数据流,即便是静态的批数据,也被看作是无界流的一个切片。这就意味着,批处理的任务其实可以理解为流处理的一个小弟,只需要在数据源那里设定一个特定的边界条件,就一切搞定了。这么做的优点就在于,开发者能够用一个统一的编程套路,来应对各种不同的应用场景,轻轻松松实现批处理和流处理之间的无缝切换。就像是你有了一个万能工具箱,甭管是组装家具还是修理电器,都能游刃有余地应对,让批处理和流处理这两种模式切换起来就像换扳手一样自然流畅。 2. 切换批处理与流处理模式的实战演示 (1)定义DataStream API java import org.apache.flink.api.common.functions.MapFunction; import org.apache.flink.streaming.api.datastream.DataStream; import org.apache.flink.streaming.api.environment.StreamExecutionEnvironment; public class BatchToStreamingExample { public static void main(String[] args) throws Exception { // 创建流处理环境 final StreamExecutionEnvironment env = StreamExecutionEnvironment.getExecutionEnvironment(); // 假设这是批处理数据源(实际上Flink也支持批处理数据源) DataStream text = env.fromElements("Hello", "World", "Flink", "is", "awesome"); // 流处理操作(映射函数) DataStream mappedStream = text.map(new MapFunction() { @Override public String map(String value) { return value.toUpperCase(); } }); // 在流处理环境中提交作业(这里也可以切换到批处理模式下运行) env.execute("Batch to Streaming Example"); } } (2)从流处理模式切换到批处理模式 上述代码是在流处理环境下运行的,但实际上,只需简单改变数据源,我们就可以轻松地处理批数据。例如,我们可以使用readTextFile方法读取文件作为批数据源: java DataStream text = env.readTextFile("/path/to/batch/data.txt"); 在实际场景中,Flink会根据数据源的特性自动识别并调整内部执行策略,实现批处理模式下的优化执行。 3. 深入探讨批流一体的价值 批处理和流处理模式的无缝切换,不仅简化了编程模型,更使资源调度、状态管理以及故障恢复等底层机制得以统一,极大地提高了系统的稳定性和性能表现。同时呢,这也意味着当业务需求风吹草动时,咱能更灵活地扭动数据处理策略,不用大费周章重构大量代码。说白了,就是“一次编写,到处运行”,真正做到灵活应变,轻松应对各种变化。 总结来说,Apache Flink凭借其批流一体的设计理念和技术实现,让我们在面对复杂多变的大数据应用场景时,拥有了更为强大且高效的武器。无论你的数据是源源不断的实时流,还是静待处理的历史批数据,Flink都能游刃有余地完成使命。这就是批流一体的魅力所在,也是我们深入探索和研究它的价值所在。
2023-04-07 13:59:38
504
梦幻星空
Greenplum
...示gpfdist将在目录/data/to/import下监听8081端口,并将日志输出至/var/log/gpfdist.log。 接下来,我们可以创建一个外部表指向gpfdist服务中的数据文件,实现数据的导入: sql CREATE EXTERNAL TABLE my_table (id int, name text) LOCATION ('gpfdist://localhost:8081/datafile.csv') FORMAT 'CSV' (DELIMITER ',', HEADER); 这段SQL语句定义了一个外部表my_table,其数据来源是通过gpfdist服务提供的CSV文件,数据按照逗号分隔,并且文件包含表头信息。 0 4. 数据导出 COPY命令的应用 4.1 COPY命令简介 Greenplum提供了强大的COPY命令,可以直接将数据从表中导出到本地文件或者从文件导入到表中,执行效率极高。 4.2 COPY命令实战示例 假设我们有一个名为sales_data的表,需要将其内容导出为CSV文件,可以使用如下命令: sql COPY sales_data TO '/path/to/export/sales_data.csv' WITH (FORMAT csv, HEADER); 这条命令会把sakes_data表中的所有数据以CSV格式(包含表头)导出到指定路径的文件中。 反过来,如果要从CSV文件导入数据到Greenplum表,可以这样做: sql COPY sales_data FROM '/path/to/import/sales_data.csv' WITH (FORMAT csv, HEADER); 以上命令将读取指定CSV文件并将数据加载到sakes_data表中。 0 5. 总结与思考 通过实践证明,不论是借助gpfdist工具进行数据导入,还是运用COPY命令完成数据导出,Greenplum都以其简单易用的特性,使得大规模数据的传输变得相对轻松。不过,在实际动手干的时候,咱们还需要瞅准不同的业务场景,灵活地调整各种参数配置。就像数据格式啦、错误处理的方式这些小细节,都得灵活应变,这样才能保证数据的导入导出既稳又快,不掉链子。同时,当我们对Greenplum越来越了解、越用越溜的时候,会惊喜地发现更多既巧妙又高效的管理数据的小窍门,让数据的价值妥妥地发挥到极致。
2023-06-11 14:29:01
469
翡翠梦境
Kubernetes
...aemonSet都会自动调度并管理这些Pod,确保集群内所有节点的服务一致性。 Kubernetes节点 , 在Kubernetes集群中,节点是一个工作机器,可以是物理机或虚拟机,它负责运行容器化应用。节点由Master组件管理,并通过kubelet代理与Master通信。节点包含运行Pod所需的环境和服务,如Docker、kube-proxy和kubelet等。 Taint 和 Label , 在Kubernetes中,Label是用于对对象(如节点、Pods、服务等)进行分类和组织的关键字/值对,使得用户可以根据标签来选择和操作资源。而Taint则是节点的一个属性,带有特定taint的节点只能调度接受相应toleration(即能容忍该taint)的Pod,用以实现节点的亲和性和反亲和性策略。 Pod , 在Kubernetes中,Pod是最小的部署单元,它是容器的逻辑分组,代表集群上运行的一个进程及其存储资源。一个Pod中可以包含一个或多个紧密相关的容器,这些容器共享网络命名空间、IP地址以及存储卷,从而形成一个协同工作的应用程序单元。 kubectl , kubectl是Kubernetes提供的命令行工具,用于与集群进行交互,执行各种操作,例如创建、修改、删除资源对象,检查集群状态,以及获取日志和监控信息等。在处理Pod不在预期节点上运行的问题时,运维人员会频繁使用kubectl执行诸如查看节点状态、编辑DaemonSet配置、调整Pod数量等相关操作。
2023-04-13 21:58:20
207
夜色朦胧-t
转载文章
...的功能支持,比如复制目录、移动文件、删除非空目录等,这些在原生fs模块中可能需要编写更多代码才能实现的功能,在fs-extra中都能轻松调用。 同时,对于持续集成(CI/CD)和自动化部署场景,通过结合Node.js的文件操作能力与其他工具(如Git、Webpack等),能够高效完成资源打包、版本控制以及自动化发布流程中的文件处理任务。 此外,随着Node.js在服务器端应用场景的拓展,如静态网站生成器(如Hugo、Gatsby)、服务端渲染框架(Next.js)等都深度依赖于文件系统的操作,深入学习和掌握Node.js的文件系统API,将有助于开发者更好地应对实际开发需求,提升工作效率。 在安全方面,Node.js文件系统操作也需注意权限管理和异常处理机制,以防止潜在的安全风险,确保数据安全和系统稳定性。因此,理解并遵循最佳实践来执行文件操作是每个Node.js开发者必备技能之一。
2023-12-30 19:15:04
67
转载
Tomcat
...ubernetes和Docker。这促使了对Tomcat的API Gateway和服务发现机制的重新思考。 首先,微服务架构下的Tomcat管理强调的是服务化和API化,如使用Spring Cloud Gateway提供统一的API入口,使得远程管理更加模块化和灵活。同时,容器化技术如Docker的使用,使得Tomcat可以在多个环境中无缝部署,简化了版本管理和部署流程。 其次,云原生集成带来了新的安全挑战和解决方案。比如,Kubernetes的Service Account和Role-Based Access Control(RBAC)可以帮助管理远程对Tomcat的访问权限,同时,云平台的自动扩缩容功能也减轻了运维压力。 此外,Kubernetes的Ingress Controller和TLS Termination在HTTPS流量管理上提供了新的可能性,使得Tomcat在云端的性能和安全性得到提升。 总的来说,现代Tomcat的远程管理已经从单一服务器扩展到整个微服务生态,这不仅需要开发者掌握新的工具和技术,也需要理解和适应云原生的思维模式。持续关注云原生技术的发展和最佳实践,对于提升Tomcat管理的效率和安全性至关重要。
2024-06-17 11:00:56
264
翡翠梦境
Struts2
...emplates/目录下,就会导致模板加载失败。 ② 模板引擎初始化异常 Struts2在启动时需要对FreeMarker或Velocity引擎进行初始化,如果相关配置如类加载器、模板路径等出现问题,也会引发模板加载失败。例如,对于Velocity,我们需要确保其资源配置正确: xml ③ 文件编码不一致 若模板文件的编码格式与应用服务器或模板引擎默认编码不匹配,也可能造成模板加载失败。例如,FreeMarker的默认编码是ISO-8859-1,如果我们创建的ftl文件是UTF-8编码,就需要在配置中明确指定编码: properties 在freemarker.properties中配置 default_encoding=UTF-8 3. 解决方案及实战演示 ① 核实并修正模板路径 检查并确认struts.xml中的结果类型配置是否指向正确的模板文件位置。如果你把模板放在了其他地方,记得及时更新路径。 ② 正确初始化模板引擎 确保配置文件(如velocity.properties和toolbox.xml)的位置和内容无误,并在Struts2配置中正确引用。如遇异常,可通过日志排查具体错误信息以定位问题。 ③ 统一文件编码 根据实际情况,调整模板文件编码或者模板引擎的默认编码设置,确保二者一致。 4. 结语 模板加载失败背后的人工智能思考 在面对模板加载失败这类看似琐碎却影响项目运行的问题时,我们需要像侦探一样细心观察、抽丝剥茧,找出问题的根本原因。同时呢,咱也要真正认识到,甭管是挑FreeMarker还是Velocity,重点不在选哪个工具,而在于怎么把它们配置得恰到好处,编码要规规矩矩的,还有就是深入理解这些框架背后的运行机制,这才是王道啊!在这个过程中,我们就像在升级打怪一样,不断从实践中汲取经验,让解决各种问题的能力蹭蹭上涨。同时呢,也像是挖掘宝藏一般,对Struts2框架以及整个Web开发大世界有了更深入、更接地气的理解和实践操作。 以上内容,我试图以一种更为口语化、情感化的表达方式,带您走过排查和解决Struts2框架中模板加载失败问题的全过程。希望通过这些实实在在的例子和我们互动式的讨论,让您不仅能摸清表面现象,更能洞察背后的原因,这样一来,在未来的开发工作中您就能更加得心应手,挥洒自如啦!
2024-03-07 10:45:28
175
风轻云淡
Gradle
...项。例如,如果你正在创建一个Java项目,并需要添加Apache Commons Lang库作为依赖,你可以这样做: groovy // 在你的module级别的build.gradle文件中 dependencies { implementation 'org.apache.commons:commons-lang3:3.12.0' // 这是一个示例依赖,版本号请根据实际情况调整 } 这里的implementation是Gradle的一种依赖范围,表示该依赖对于当前模块内部是可见的,但在编译生成的库或应用中将不会暴露给其他依赖此模块的项目。当然,还有其他的依赖范围,如api、compileOnly等,具体选择哪种取决于你的项目需求。 2. 使用Gradle命令同步依赖 添加了依赖后,我们需要让Gradle下载并同步这些依赖到本地仓库。这可以通过运行以下命令实现: bash $ gradle build --refresh-dependencies --refresh-dependencies标志会强制Gradle重新下载所有依赖,即使它们已经在本地缓存中存在。当首次添加依赖或更新依赖版本时,这个步骤至关重要。 3. 配置打包插件以包含依赖 为了确保依赖包能够被打包进最终的产品(如jar或war),你需要配置对应的打包插件。例如,对于Java项目,我们通常会用到java或application插件,而对于Web应用,可能会用到war插件。 groovy // 应用application插件以创建可执行的JAR,其中包含了所有依赖 apply plugin: 'application' // 或者,对于web应用,应用war插件 apply plugin: 'war' // 配置mainClass(仅对application插件有效) mainClassName = 'com.example.Main' // 确保构建过程包含所有依赖 jar { from { configurations.runtimeClasspath.collect { it.isDirectory() ? it : zipTree(it) } } } // 对于war插件,无需特殊配置,它会自动包含所有依赖 这段代码的作用是确保在构建JAR或WAR文件时,不仅包含你自己的源码编译结果,还包含所有runtimeClasspath上的依赖。 4. 深入理解依赖管理和打包机制 当你完成上述步骤后,Gradle将会在打包过程中自动处理依赖关系,并将必要的依赖包含在内。不过,在实际动手操作的时候,免不了会碰到些复杂状况。就好比在多个模块的项目间,它们之间的依赖关系错综复杂,像传球一样互相传递;又或者有时候你得像个侦探,专门找出并排除那些特定的、不需要的依赖项,这些情况都是有可能出现的。 这里有一个思考点:Gradle的强大之处在于其智能的依赖解析和冲突解决机制。当你在为各个模块设定依赖关系时,Gradle这个小帮手会超级聪明地根据每个依赖的“身份证”(也就是group、name和version)以及它们的依赖范围,精心挑选出最合适、最匹配的版本,然后妥妥地将它打包进构建出来的最终产物里。所以呢,摸清楚Gradle里面的依赖管理和生命周期这俩玩意儿,就等于在打包的时候给咱装上了一双慧眼,能更溜地驾驭这些依赖项的行为,让它们乖乖听话。 总结来说,通过在build.gradle文件中明确声明依赖、适时刷新依赖、以及合理配置打包插件,我们可以确保Gradle在打包阶段能准确无误地包含所有必要的依赖包。在实际动手捣鼓和不断尝试的过程中,你会发现Gradle这个超级灵活、威力强大的构建神器,不知不觉间已经给我们的工作带来了很多意想不到的便利,让事情变得更加轻松简单。
2023-08-27 09:07:13
471
人生如戏_
HTML
...别的信息。这些日志会自动乖乖地蹦进默认的日志文件里头,这个文件一般都藏在你电脑的AppData目录下,具体哪个小角落就得看你的操作系统啦。 3. 自定义日志文件路径及格式 如果你希望自定义日志文件的位置和名称,可以通过以下方式设置: javascript log.transports.file.getFile().path = path.join(app.getPath('userData'), 'custom-log.log'); 同时,electron-log也支持多种格式化选项,包括JSON、pretty-print等,可以根据需求调整: javascript log.transports.file.format = '{h}:{i}:{s} {level}: {text}'; 4. 思考与讨论 值得注意的是,虽然我们在渲染进程中直接调用了electron-log,但实际上所有的日志都通过IPC通信机制传递给主进程,再由主进程负责实际的写入文件操作。这么干,既能确保安全,防止渲染进程直接去摆弄磁盘,还能让日志管理变得简单省事儿多了。 在整个过程中,electron-log不仅充当了开发者的眼睛,洞察每一处可能的问题点,还像一本详尽的操作手册,忠实记录着应用运行的每一步足迹。这种实时、细致入微的日志系统,绝对是我们Electron应用背后的强大后盾,让我们的应用跑得既稳又强。 总的来说,通过electron-log,我们在 Electron 渲染进程中记录和输出日志变得轻松易行,大大提高了调试效率和问题定位的速度。每一个开发者都该好好利用这些工具,让咱们的应用程序像人一样“开口说话”,把它们的“心里话”都告诉我们。
2023-10-02 19:00:44
552
岁月如歌_
SpringBoot
...下我们会设置一个重试机制,以应对可能出现的各种网络、服务器等不可控因素导致的消息发送失败。但是,如果不加把劲儿控制一下,这种重试机制就很可能像一群疯狂的粉丝不断涌向同一个明星那样,让同一台Broker承受不住压力,这样一来,严重的性能问题也就随之爆发喽。所以呢,我们得在重试这套流程里头动点脑筋,加点策略进去。这样一来,当生产者小哥遇到状况失败了,就能尽可能地绕开那些已经闹情绪的Broker家伙,不让它们再添乱。 三、解决方案 为了解决这个问题,我们可以采用以下两种方案: 1. 设置全局的Broker列表 在创建Producer实例时,我们可以指定一个包含所有Broker地址的列表,然后在每次重试时随机选择一个Broker进行发送。这样可以有效地避免过多的请求集中在某一台Broker上,从而降低对Broker的压力。以下是具体的代码实现: java List brokers = Arrays.asList("broker-a", "broker-b", "broker-c"); Set failedBrokers = new HashSet<>(); public void sendMessage(String topic, String body) { for (int i = 0; i < RETRY_TIMES; i++) { Random random = new Random(); String broker = brokers.get(random.nextInt(brokers.size())); if (!failedBrokers.contains(broker)) { try { producer.send(topic, new MessageQueue(topic, broker, 0), new DefaultMQProducer.SendResultHandler() { @Override public void onSuccess(SendResult sendResult) { System.out.println("Message send success"); } @Override public void onException(Throwable e) { System.out.println("Message send exception: " + e.getMessage()); failedBrokers.add(broker); } }); return; } catch (Exception e) { System.out.println("Message send exception: " + e.getMessage()); failedBrokers.add(broker); } } } System.out.println("Message send fail after retrying"); } 在上述代码中,我们首先定义了一个包含所有Broker地址的列表brokers,然后在每次重试时随机选择一个Broker进行发送。如果该Broker在之前已经出现过错误,则将其添加到已失败的Broker集合中。在下一次重试时,我们不再选择这个Broker。 2. 利用RocketMQ提供的重试机制 除了手动设置Broker列表之外,我们还可以利用RocketMQ自带的重试机制来达到相同的效果。简单来说,我们可以搞个“RetryMessageListener”这个小家伙来监听一下,它的任务就是专门盯着RocketMQ发出的消息。一旦消息发送失败,它就负责把这些失败的消息重新拉出来再试一次,确保消息能顺利送达。在用这个监听器的时候,我们就能知道当前的Broker是不是还在重试列表里混呢。如果发现它在的话,那咱们就麻利地把它从列表里揪出来;要是不是,那就继续让它“回炉重造”,执行重试操作呗。以下是具体的代码实现: java public class RetryMessageListener implements MQListenerMessageConsumeOrderlyCallback { private Set retryBrokers = new HashSet<>(); private List brokers = Arrays.asList("broker-a", "broker-b", "broker-c"); @Override public ConsumeConcurrentlyStatus consumeMessage(List msgs, ConsumeConcurrentlyContext context) { for (String broker : brokers) { if (retryBrokers.contains(broker)) { retryBrokers.remove(broker); } } for (String broker : retryBrokers) { try { producer.send(msgs.get(0).getTopic(), new MessageQueue(msgs.get(0).getTopic(), broker, 0),
2023-06-16 23:16:50
39
梦幻星空_t
ClickHouse
...量备份,并保存到指定目录。你还可以通过配置文件或命令行参数指定要备份的具体数据库或表。 1.2 恢复备份 当发生数据丢失时,可以利用备份文件进行恢复: bash clickhouse-backup restore backup_name 执行上述命令后,ClickHouse将会从备份中恢复所有数据。千万要注意啊,伙计,在你动手进行恢复操作之前,得先瞧瞧目标集群是不是空空如也,或者你是否能接受数据被覆盖这个可能的结果。 2. 使用Replication(复制)机制 2.1 配置Replicated表 ClickHouse支持ZooKeeper或Raft协议实现的多副本复制功能。例如,创建一个分布式且具有复制特性的表: sql CREATE TABLE replicated_table ( ... ) ENGINE = ReplicatedMergeTree('/clickhouse/tables/{database}/{table}', 'replica1') PARTITION BY ... ORDER BY ... 这里,/clickhouse/tables/{database}/{table}是一个 ZooKeeper 路径,用于协调多个副本之间的数据同步;'replica1'则是当前副本标识符。 2.2 数据自动同步与容灾 一旦某台服务器上的数据出现异常,其他拥有相同Replicated表的服务器仍保留完整的数据。当有新的服务器小弟加入集群大家庭,或者主节点大哥不幸挂掉的时候,Replication机制这个超级替补队员就会立马出动,自动把数据同步得妥妥的,确保所有数据都能保持一致性、完整性,一个字都不会少。 3. 数据一致性检查与修复 3.1 使用checksum函数 ClickHouse提供checksum函数来计算表数据的校验和,可用于验证数据是否完整: sql SELECT checksum() FROM table_name; 定期执行此操作并记录结果,以便在后续时间点对比校验和的变化,从而发现可能的数据丢失问题。 3.2 表维护及修复 若发现数据不一致,可以尝试使用OPTIMIZE TABLE命令进行表维护和修复: sql OPTIMIZE TABLE table_name FINAL; 该命令会重新整理表数据,并尝试修复任何可能存在的数据损坏问题。 4. 实践思考与探讨 尽管我们可以通过上述方法来减少和应对ClickHouse中的数据丢失风险,但防患于未然总是最优策略。在搭建和运用ClickHouse系统的时候,千万记得要考虑让它“坚如磐石”,也就是要设计出高可用性方案。比如说,我们可以采用多副本这种方式,就像备份多个小帮手一样,让数据安全无忧;再者,跨地域冗余存储也是一招妙计,想象一下,即使地球另一边的机房挂了,这边的数据也能照常运作,这样就大大提升了系统的稳健性和可靠性啦!同时,建立一个完善、接地气的数据监控系统,能够灵敏捕捉并及时解决那些可能冒头的小问题,这绝对是一个无比关键的步骤。 总结起来,面对ClickHouse数据丢失问题,我们需采取主动防御和被动恢复相结合的方式,既要做好日常的数据备份和Replication配置,也要学会在问题发生后如何快速有效地恢复数据,同时结合数据一致性检查以及表维护等手段,全面提升数据的安全性和稳定性。在实践中不断优化和完善,才能真正发挥出ClickHouse在海量数据分析领域的强大威力。
2023-01-20 13:30:03
445
月影清风
Nacos
...户或应用程序对文件、目录或系统资源的操作能力。在计算机系统中,权限管理是一项重要的安全措施,用于控制谁可以执行特定的操作,如读取、写入或执行文件。权限通常分为多个级别,如只读、读写等。在文章中,权限问题是指应用程序没有足够的权限去修改或创建配置文件,从而导致配置信息无法正确写入本地存储。确保应用程序具有适当的文件权限是保证配置信息正确写入的重要步骤。
2024-11-26 16:06:34
158
秋水共长天一色
Shell
...单,只需进入你的脚本目录,运行: bash git init 这样,Git就会在这个目录里创建一个隐藏的.git文件夹,用来存储所有版本信息。 4.2 添加文件并提交 接着,你需要把你的Shell脚本添加到Git仓库,并进行第一次提交。假设你的脚本叫myscript.sh,你可以这样做: bash git add myscript.sh git commit -m "Initial commit of myscript.sh" 这里,-m后面跟着的是这次提交的信息,简短明了地描述了这次改动的内容。 4.3 操作示例 假设你已经有一个名为backup.sh的脚本,想要加入版本控制,你可以这么做: bash cd /path/to/your/script git init git add backup.sh git commit -m "Add backup script" 这样,你就有了一个基础的Git仓库,可以开始跟踪你的脚本变化了。 4.4 使用别名简化命令 为了方便操作,我们可以给常用的Git命令设置别名。在你的~/.bashrc或~/.zshrc文件中添加如下内容: bash alias gs='git status' alias gc='git commit -m' 这样,以后只需要输入gs就能查看状态,gc "Your commit message"就可以直接提交了,是不是很方便? 5. 高级技巧 5.1 分支管理 分支是Git的一大特色,可以让你在同一项目中同时处理多个功能。例如,你想尝试一个新的特性,但又不想影响主分支上的稳定代码,可以创建一个新的分支: bash git checkout -b feature-branch 然后在这个分支上做任何你想做的改动,最后合并回主分支: bash git checkout main git merge feature-branch 5.2 远程仓库与GitHub 如果你需要与他人协作,或者想备份你的代码,可以将本地仓库推送到远程服务器,比如GitHub。首先,你需要在GitHub上创建一个仓库,然后添加远程仓库地址: bash git remote add origin https://github.com/yourusername/yourrepo.git git push -u origin main 这样,你的代码就安全地保存在云端了。 6. 结语 通过这篇文章,我希望你对如何在Shell脚本中集成版本控制系统有了更深的理解。记住,版本控制不只是技术活儿,它还是咱们好好工作的习惯呢!从今天起,让我们一起养成良好的版本控制习惯吧! 如果你有任何疑问或想了解更多细节,请随时留言交流。我们一起探索更多的技术奥秘!
2025-01-26 15:38:32
50
半夏微凉
转载文章
...17的发布,对类加载机制和程序入口点有了更灵活的处理方式。例如,Java平台模块系统(JPMS)允许开发者定义模块化的组件,并通过明确指定模块间的依赖关系实现自动编译和加载,这使得即使没有传统意义上的main方法,也能构建可运行的Java应用程序。 同时,对于微服务架构和容器化部署场景,通常采用框架或容器(如Spring Boot、Docker等)来管理应用的生命周期,它们提供了自定义启动器和引导过程,不再强制要求每个服务包含一个main方法。在这种情况下,业务逻辑被封装在服务类中,由框架统一调度执行。 此外,随着函数式编程思想在Java领域的普及,Java开发者开始更多地利用Lambda表达式和函数接口,甚至借助第三方库(如JavaFX、Quarkus、Vert.x等)提供的无main方法运行模式,简化了小型脚本和事件驱动型应用的编写和执行流程。 总而言之,在当今Java开发领域中,虽然main方法仍然是独立Java应用程序的标准入口点,但随着技术进步和编程范式的演变,Java代码的执行和编译机制正变得日益丰富和多元化。为了紧跟这一发展步伐,开发者需要不断学习和掌握新的工具、框架及编程模式,以应对日益复杂的应用场景需求。
2023-08-16 23:56:55
366
转载
Docker
为什么很多Docker容器中的用户uid默认是999? 在探索Docker容器世界的旅程中,我们经常会发现一个有趣的现象:不少Docker镜像或容器内运行的进程,默认情况下其用户的uid(User ID)被设置为999。你可能心里正犯嘀咕,为啥我们偏偏对这个数字情有独钟,而不是其他的呢?在这篇文里,咱们就一起手拉手,像解密探险一样揭开这个谜团吧!我会带着大伙儿,通过实实在在的例子和深入的讨论,来摸清楚这背后究竟藏着啥讲究。 1. Docker容器与用户权限 首先,让我们简要回顾一下Docker容器内的用户权限模型。你知道吗,Docker那个小家伙,默认情况下启动容器时,会直接动用到root大权限,这在安全性和隔离性方面,可不是什么顶呱呱的优秀操作。为了让大家用得更安心,我常常建议这样做:别让你在容器里运行的应用权限太高了,最好能把它们映射到宿主机上的普通用户级别,这样一来就更加安全啦。就像是让这些应用从VIP房间搬到了经济舱,虽然待遇没那么高,但是安全性却大大提升,避免惹出什么乱子来。这就引出了uid的概念——它是Unix/Linux系统中标识用户身份的重要标识符。 2. 默认uid的选择 999的秘密 那么,为什么许多Docker官方或社区制作的镜像倾向于将应用运行时的用户uid设为999呢?答案其实并不复杂: - 避免冲突:在大多数Linux发行版中,系统用户的uid从100开始分配给普通用户,因此选取大于100但又不是特别大的数字(如999),可以最大程度地减少与宿主机现有用户的uid冲突的可能性。 - 保留空间:选择一个高于常规uid范围的值,确保了不会意外覆盖宿主机上的任何重要用户账号。 - 一致性与约定俗成:随着时间推移,选用999作为非root用户的uid逐渐成为一种行业惯例和最佳实践,尤其是在创建需要低权限运行的应用程序镜像时。 3. 实践示例 自定义uid的Dockerfile 下面是一个简单的Dockerfile片段,展示如何在构建镜像时创建并使用uid为999的用户: dockerfile 首先,基于某个基础镜像 FROM ubuntu:latest 创建一个新的系统用户,指定uid为999 RUN groupadd --gid 999 appuser && \ useradd --system --uid 999 --gid appuser appuser 设置工作目录,并确保所有权归新创建的appuser所有 WORKDIR /app RUN chown -R appuser:appuser /app 以后的所有操作均以appuser身份执行 USER appuser 示例安装和运行一个应用程序 RUN npm install 假设我们要运行一个Node.js应用 CMD ["node", "index.js"] 在这个例子中,我们创建了一个名为appuser的新用户,其uid和gid都被设置为999。然后呢,咱就把容器里面的那个 /app 工作目录的所有权,给归到该用户名下啦。这样一来,应用在跑起来的时候,就能够顺利地打开、编辑和保存文件,不会因为权限问题卡壳。 4. 深入思考 uid映射与安全策略 虽然999是一个常见选项,但它并不是硬性规定。实际上,根据具体的部署环境和安全需求,你可以灵活调整uid。比如,在某些情况下,可能需要把容器里面的用户uid,对应到宿主机上的某个特定用户,这样一来,我们就能对文件系统的权限进行更精准的调控了,就像拿着钥匙开锁那样,该谁访问就给谁访问的权利。这时,可以通过Docker的--user参数或者在Dockerfile中定义用户来实现uid的精确映射。 总而言之,Docker容器中用户uid为999这一现象,体现了开发者们在追求安全、便捷和兼容性之间所做的权衡和智慧。随着我们对容器技术的领悟越来越透彻,这些原则就能被我们玩转得更加游刃有余,随时适应各种实际场景下的需求变化,就像是给不同的应用场景穿上量身定制的衣服一样。而这一切的背后,都离不开我们持续的探索、试错和优化的过程。
2023-05-11 13:05:22
463
秋水共长天一色_
HessianRPC
...户端可以通过动态加载机制,根据服务端返回的版本信息加载对应的接口实现类,从而实现自动适配新版本服务。 java // 动态加载示例(伪代码) String serviceUrl = "http://server:port/myService"; HessianProxyFactory factory = new HessianProxyFactory(); MyService myService; try { // 获取服务端版本信息 VersionInfo versionInfo = getVersionFromServer(serviceUrl); // 根据版本创建代理对象 if (versionInfo.isV1()) { myService = (MyService) factory.create(MyService.class, serviceUrl + "?version=v1"); } else if (versionInfo.isV2()) { myService = (MyService) factory.create(MyService.class, serviceUrl + "?version=v2"); } } catch (Exception e) { // 错误处理 } // 调用对应版本的方法 String result = myService.newMethod(1, "newParam"); - 客户端版本迭代:对于无法通过兼容性设计解决的重大变更,客户端也需要同步更新以适应新接口。这时候,咱们得好好策划一个详尽的升级计划和方案出来,并且要赶紧给所有客户端开发的大哥们发个消息,让他们麻溜地进行更新工作。 总结起来,要保证Hessian服务端更新后与客户端的无缝对接,关键在于合理的设计和服务管理策略,包括但不限于版本控制、接口向后兼容性设计、双重部署及灰度发布以及客户端的灵活适配升级。在整个过程中,不断沟通、思考和实践,才能确保每一次迭代都平稳顺利地完成。
2023-10-30 17:17:18
495
翡翠梦境
转载文章
...查询优化以及内存分配机制等核心内容至关重要。例如,如何根据业务场景合理设计索引,能显著提高查询效率;而通过定期分析并调整MySQL配置参数,如innodb_buffer_pool_size,可以帮助系统更好地利用硬件资源,提升整体性能。 此外,在当前云原生与容器化技术盛行的时代背景下,学习如何在Docker或Kubernetes环境中部署和管理MySQL也极为重要。MySQL官方已提供适用于多种容器平台的镜像,便于用户快速搭建高可用、弹性伸缩的数据库集群。 同时,随着数据安全问题日益凸显,MySQL数据库的安全加固措施同样值得重点关注。包括但不限于使用SSL加密传输数据、设置复杂的账户权限体系、定期审计与备份数据库,以及采用诸如防火墙规则限制访问来源等多种手段,确保数据库系统的安全稳定运行。 综上所述,无论是紧跟MySQL最新版本特性、深入钻研数据库内部原理,还是关注新技术环境下的部署实践与安全防护策略,都是每一位数据库管理人员持续进阶的必修课程。
2023-12-22 19:36:20
117
转载
站内搜索
用于搜索本网站内部文章,支持栏目切换。
知识学习
实践的时候请根据实际情况谨慎操作。
随机学习一条linux命令:
watch -n 5 command
- 每隔5秒执行一次指定命令并更新输出。
推荐内容
推荐本栏目内的其它文章,看看还有哪些文章让你感兴趣。
2023-04-28
2023-08-09
2023-06-18
2023-04-14
2023-02-18
2023-04-17
2024-01-11
2023-10-03
2023-09-09
2023-06-13
2023-08-07
2023-03-11
历史内容
快速导航到对应月份的历史文章列表。
随便看看
拉到页底了吧,随便看看还有哪些文章你可能感兴趣。
时光飞逝
"流光容易把人抛,红了樱桃,绿了芭蕉。"