前端技术
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
[C10K问题解决方案]的搜索结果
这里是文章列表。热门标签的颜色随机变换,标签颜色没有特殊含义。
点击某个标签可搜索标签相关的文章。
点击某个标签可搜索标签相关的文章。
ClickHouse
...可能会遇到一些常见的问题。这中间啊,有一个问题相当普遍,也是我们需要好好琢磨琢磨的,那就是“表格的列突然自动增长出错了”。 二、问题解析 1. 什么是“表的列出现自动增长错误”? 当我们创建一个表并定义了一个具有自动增长属性的列时,如果我们尝试插入一条数据并且这个列没有被指定为值,则会出现这个错误。 2. 为什么会出现这种错误? 这是因为ClickHouse在处理数据时,需要确保每一行的数据都是完整的。如果你在往数据库里插数据的时候,忘记给自增列填数值了,ClickHouse这个家伙就会觉得这条数据缺胳膊少腿的,不够完整,然后就“怒”了,给你抛出一个错误来。 三、解决方案 1. 使用默认值 如果我们知道某一列的所有数据应该具有相同的初始值,我们可以直接将这个初始值设置为该列的默认值。例如: sql CREATE TABLE test ( id UInt32, value UInt32 DEFAULT 0, name String ) ENGINE = MergeTree() ORDER BY id; 在这个例子中,value列的默认值被设置为了0,这样我们就无需在插入数据时手动指定它的值了。 2. 插入完整数据 另一种避免这种错误的方法是在插入数据时提供所有列的值。例如: sql INSERT INTO test (id, value, name) VALUES (1, 0, 'test'); 在这个例子中,我们在插入数据时提供了value列的值,因此ClickHouse不会抛出错误。 四、总结 通过以上分析,我们可以看出“表的列出现自动增长错误”实际上是因为我们在插入数据时不提供完整的信息导致的。要搞定这个问题,关键点在于得把所有列的数值都清清楚楚地填上,或者,对于那种会自动增长的列,给它设定一个默认的初始值就搞定了。只要我们遵循这些规则,就可以有效地避免这个错误。 五、建议 在使用ClickHouse进行数据分析时,我们应该始终注意保持数据的一致性和完整性。这不仅能让我们彻底告别“表的列自动增长出错”的烦恼,更能实实在在地提升咱们的工作效率,让数据分析的质量蹭蹭上涨。 六、结语 ClickHouse是一款强大的实时数据分析工具,但是在使用它的时候也会遇到各种各样的问题。不过,只要我们把这些小问题背后的“猫腻”摸清楚,再掌握几招解决它们的窍门,那咱们就能更溜地运用ClickHouse,让它帮咱们把数据分析的事儿做得妥妥的。
2023-07-20 08:25:08
553
林中小径-t
HTML
...提供强大的、更简洁的解决方案来处理各种复杂布局问题。在本文的上下文中,通过设置div元素的display属性为flex,我们可以轻松实现其内部子元素(如冰墩墩图片和文字)的水平居中和垂直居中对齐,以及灵活的伸缩空间分配。 alt属性 , alt属性是HTML img标签的一个重要属性,全称为“alternate text”。在网页开发中,当图片无法加载或用户使用屏幕阅读器等辅助技术时,alt属性提供的文本信息将替代图片显示,以确保内容的可访问性和理解性。在这篇文章中,冰墩墩图片的alt属性值设置为“冰墩墩”,确保了即使图片未加载成功,用户也能知道该部分代表的是北京冬奥会吉祥物冰墩墩。 响应式设计 , 响应式设计是一种网页设计方法,使得网站能够根据用户所使用的设备环境(系统平台、屏幕尺寸、屏幕方向等)进行适应性的布局调整。虽然文章并未直接提及响应式设计,但在实际开发过程中,制作冰墩墩网站图标时可能会运用到这一原则,比如利用CSS3媒体查询技术让冰墩墩图像在不同尺寸的屏幕上都能保持合适的大小和显示效果,从而提升用户的浏览体验。
2023-07-30 08:03:59
730
电脑达人
CSS
...访问,成为了当下亟待解决的问题。 不仅如此,在实际项目中,诸如“黑暗模式”切换、滚动监听触发的导航变化等高级交互功能,也正逐步成为提升网站专业度和用户满意度的关键因素。因此,深入研究和实践CSS以及相关的前端技术,结合人性化和美学原则,是每个网页设计师不断提升自身能力、紧跟行业发展的必经之路。 另外,值得关注的是,新的CSS规范和技术如Subgrid、Container Queries等已经开始得到部分浏览器的支持,这些都将为未来的导航菜单设计提供更多可能性和创造性解决方案。通过学习和掌握这些新兴技术,设计师将能够创建出更加高效、美观且易于使用的导航系统,进一步提升用户的浏览体验。
2023-05-12 08:57:33
457
程序媛
JQuery
...高了团队的响应速度和问题解决效率。 由此可见,Echarts已经从单纯的可视化工具进化成为企业数据战略的重要组成部分,它正在推动企业迈向数据驱动的智能运营时代。对于任何寻求提升数据分析能力,优化决策流程的企业来说,Echarts都是值得深入研究和实践的利器。
2024-04-28 16:11:37
299
代码侠
Kylin
...过磁盘分区识别错误的问题?这个问题可能会让你感到困惑和沮丧,因为你可能不知道如何解决它。别担心,我们来一起探讨一下这个问题。 二、问题解析 首先,让我们来看看什么是磁盘分区识别错误。简单来说,当你打算把一个文件从一处搬到另一处,但这两个地方不在同一个磁盘分区上时,你的电脑操作系统就会犯迷糊,认不出磁盘分区,然后给你来个错误提示。这是因为不同的磁盘分区有不同的文件系统,如果你试图将文件从一种文件系统移动到另一种文件系统,操作系统就无法识别这个操作。 三、原因分析 那么,为什么我们在安装Kylin系统时会出现这种问题呢?这可能是由于以下几种原因: 1. 系统资源不足 如果你的计算机硬盘空间不足,系统可能无法正确地进行分区。 2. 文件系统不匹配 如果你试图将文件从一种文件系统移动到另一种文件系统,而这两个文件系统的版本不同,系统就可能出现识别错误。 3. 磁盘损坏 如果你的磁盘出现物理损坏,系统就可能无法正确地读取和写入数据。 四、解决方案 知道了问题的原因,我们就可以开始寻找解决问题的方法了。以下是一些常见的解决办法: 1. 扩展硬盘空间 如果你的硬盘空间不足,你可以尝试扩大硬盘的空间。这可以通过购买一个新的硬盘或者升级现有的硬盘来实现。 2. 更改文件系统 如果你试图将文件从一种文件系统移动到另一种文件系统,你可以尝试更改其中一个文件系统的版本。比如说,你要是想把文件从FAT32格式的盘挪到NTFS格式的盘,完全可以先把这个盘转换成NTFS格式,然后再进行文件搬家的操作。 3. 检查磁盘 如果你的磁盘出现物理损坏,你需要检查磁盘并修复或替换它。 五、实例演示 让我们来看一个具体的例子。假设你在安装Kylin系统时出现了磁盘分区识别错误。你可以按照以下步骤来解决问题: 1. 首先,检查你的硬盘空间。如果你的硬盘空间不足,你需要扩展硬盘空间。你可以通过购买一个新的硬盘或者升级现有的硬盘来实现。 2. 其次,检查你的文件系统。如果你想把文件从一个文件系统搬到另一个文件系统,那就得先瞧准了,这两个系统的版本得对得上号才行。你可以使用命令行工具来查看和更改文件系统的版本。例如,在Windows系统中,你可以使用fsutil fsinfo diskvolume信息来查看和更改文件系统的版本。 3. 最后,如果你的磁盘出现物理损坏,你需要检查磁盘并修复或替换它。你可以使用各种磁盘检测和修复工具来帮助你完成这个任务。 六、总结 总的来说,磁盘分区识别错误是一个比较常见的问题,但是只要你知道了它的原因,并且采取了正确的解决办法,你就能够成功地解决这个问题。记住了啊,不论你碰到啥困难、挑战,都要稳住心态,乐观面对,坚信自己肯定有办法把问题给解决了。别忘了,你可是个解决问题的小能手呢!
2023-04-06 20:16:18
186
雪域高原-t
VUE
...构建基于Web的开源解决方案,如PDF.js、pdfmake等项目,使得开发在线PDF编辑器和生成器变得更加简易高效。这些服务不仅支持从文本内容快速创建PDF,还能够处理表格、图像、列表等复杂结构,并确保跨平台兼容性良好。 此外,针对数据安全性和隐私保护问题,一些在线PDF工具也推出了加密生成、权限设置等功能,以满足企业和个人用户对于敏感信息处理的安全需求。因此,了解并合理利用这些在线PDF工具和技术,不仅可以提升工作效率,也是紧跟数字化时代发展步伐的重要表现。通过深入研究和实践,我们可预见在未来,更多创新的在线文档处理方案将不断涌现,持续推动无界办公和远程协作的新常态。
2023-11-07 11:10:47
80
程序媛
Datax
...遇到一些磕磕绊绊的小问题,就比如这次我要和大家伙儿深入探讨的“连接源数据库时授权不给力”的状况。 二、授权失败的原因分析 当我们尝试使用Datax连接源数据库时,如果出现授权失败的情况,可能是因为以下几个原因: 1. 数据库用户名或密码错误 这是最常见的原因,也是最容易检查和修复的问题。 2. 数据库权限不足 例如,没有执行某些特定操作的权限(如INSERT, UPDATE, DELETE等)。 3. 数据库服务器设置问题 例如,数据库服务器的安全策略设置过严格,不允许从指定IP地址进行连接。 4. 数据库防火墙设置问题 例如,数据库防火墙阻止了Datax的连接请求。 三、解决方案 针对以上问题,我们可以采取以下措施来解决: 1. 检查并确认数据库用户名和密码是否正确。比如,咱们可以试试直接在数据库客户端里把这些信息敲进去登录一下,看看能不能顺利连上数据库。 2. 检查并确认Datax连接字符串中的用户名和密码是否正确。例如: python sourceDB = "mysql://username:password@host/database" 这里,username和password需要替换为你的实际用户名和密码,host需要替换为你的数据库服务器地址,database需要替换为你的目标数据库名称。 3. 如果数据库服务器设置了安全策略,需要确保你使用的用户名具有执行所需操作的权限。要解决这个问题,你只需要在数据库客户端里动动手,新建一个用户账号,然后给这个账号分配它所需要的权限就搞定了。就像是在手机上注册个新用户,然后赋予它特定的使用权限一样简单易懂。 4. 如果数据库防火墙阻止了Datax的连接请求,你需要调整防火墙规则,允许来自Datax运行机器的连接请求。 四、结论 总的来说,当我们在使用Datax连接源数据库时遇到授权失败的问题时,我们需要仔细检查我们的数据库配置和安全策略,以及我们的Datax配置文件。同时呢,我们还得翻翻Datax的官方文档,逛逛社区论坛啥的,这样才能捞到更多的帮助和解决方案。希望这篇文章能对你有所帮助!
2023-05-11 15:12:28
564
星辰大海-t
Python
...人的智能,实现对复杂问题的解决与决策。Python作为一种强大的编程语言,在AI领域被广泛应用,包括但不限于机器学习、深度学习、自然语言处理等方面,为构建智能算法和模型提供便捷高效的工具。 数据挖掘(Data Mining) , 数据挖掘是通过运用统计学、机器学习等方法从大量数据中抽取有价值的信息和知识的过程。在Python的学习与应用中,它扮演了重要角色,例如使用Pandas库进行数据清洗与预处理,利用Scikit-learn等库进行数据建模与分析,从而帮助用户发现数据背后的模式和规律。 网络开发(Web Development) , 网络开发指的是创建和维护网站或网络应用程序的一系列活动,包括前端设计、后端逻辑编写以及数据库管理等多个方面。Python在网络开发中的作用主要体现在其丰富的Web框架上,如Django和Flask,这些框架简化了开发者的工作流程,提供了快速搭建稳定高效网站的解决方案。 实际项目(Real-world Project) , 在本文中,“实际项目”指的是将Python编程知识应用于解决现实生活或工作场景中的具体问题的实践活动。比如,用Python开发一个数据分析项目、建立一个基于网络的应用程序或者编写自动化脚本来提升工作效率等。通过参与实际项目,学习者能够在实践中深化对Python的理解,并锻炼自身解决问题的能力。
2023-09-23 08:54:15
330
电脑达人
Tornado
...经常会遇到各种各样的问题。而其中,“Tornado服务器无法启动”就是一种非常常见的问题。Tornado是一个Python Web框架和异步网络库,由FriendFeed开发,并于2009年开源。然而,在实际操作的时候,我们可能会遇到这么个情况:咱们的Tornado服务器突然不听话了,死活启动不了。 二、什么是Tornado? Tornado是一种用于构建可伸缩Web应用程序和非阻塞网络服务的Python库。它超级灵活,能够轻松应对海量的同时连接请求,而且在I/O操作这方面可是精心优化过的,所以特别适合那些需要实时交互的应用和服务场景。然而,跟其他软件一样,Tornado这家伙有时候也会闹点小脾气,比如它可能会出现个常见的问题——“Tornado服务器启动不起来啦”。 三、为什么会出现“Tornado服务器无法启动”的问题? 当我们在运行Tornado服务器时,如果出现“Tornado服务器无法启动”的错误,那么这通常意味着我们的服务器遇到了某种问题,无法正常启动并提供服务。这种情况可能有很多原因,以下是一些最常见的可能性: 1. 依赖包缺失 Tornado是一个依赖众多Python库的程序,如果我们没有正确安装或者缺少某些必要的依赖,那么就可能出现这个问题。 2. 路径配置错误 在运行Tornado服务器之前,我们需要进行一些路径配置,如果这些配置不正确,也可能导致服务器无法启动。 3. 系统资源不足 如果我们的系统资源(如内存、CPU等)不足以支持Tornado服务器的运行,那么服务器也可能无法启动。 四、如何解决“Tornado服务器无法启动”的问题? 当我们遇到“Tornado服务器无法启动”的问题时,我们应该首先尝试找出具体的原因,然后根据具体情况来解决问题。以下是一些可能的解决方案: 1. 检查依赖包 我们可以检查一下是否已经正确安装了所有的依赖包。如果没有,我们就需要安装它们。例如,我们可以通过pip来安装: python pip install tornado 2. 检查路径配置 我们需要确保我们的路径配置是正确的。例如,我们可以在代码中这样设置路径: python import os os.chdir("/path/to/your/project") 3. 检查系统资源 我们需要确保我们的系统资源足够支持Tornado服务器的运行。要是资源不够使了,咱们可能得考虑升级一下硬件设备,或者把咱们的代码整得更精简些,好让资源能省着点用。 五、总结 “Tornado服务器无法启动”是我们经常遇到的一个问题,但是只要我们找到了具体的原因,并采取相应的措施,就可以很容易地解决这个问题。另外呢,咱们也得学点日常的故障排除小窍门儿,这样一旦碰上问题,就能立马找到解冑方案,省得干着急。 六、参考资料 [1] Tornado官方文档: [2] Stack Overflow上的相关讨论: 注意:以上内容仅供参考,具体的操作方法需要根据实际情况进行调整。
2023-12-23 10:08:52
157
落叶归根-t
Maven
...用原理及其可能遇到的问题后,我们进一步探讨Maven构建工具的最新发展动态和高级应用场景。 近期,Apache Maven 4.0-alpha-1版本已发布,其中对构建生命周期管理和插件执行逻辑进行了优化升级。新版本改进了对execution-id的解析机制,使得开发者在命令行中指定特定execution时更为便捷高效。同时,Maven团队正积极探索通过配置文件或环境变量来动态控制execution-id的可能性,以适应云原生、持续集成/持续部署(CI/CD)等现代开发场景的需求。 此外,为了提升Maven构建过程中的灵活性和可维护性,业界提倡采用Profile和Build Profiles的概念,通过这些功能可以基于不同的环境和条件激活预设的execution-id集合,从而实现更加精细的构建流程控制。 深入研究Maven构建过程的最佳实践,例如ThoughtWorks公司的技术博客曾就如何合理组织plugin executions进行过深度剖析,强调了正确配置execution-id对于项目模块化构建的重要性,并结合实际案例提供了详尽的解决方案。 因此,在实际开发中,不仅需要掌握execution-id的基本用法,更要关注Maven社区的发展动态与最佳实践,以便更高效地利用这一强大工具管理复杂的Java项目构建流程。
2023-12-11 19:41:15
108
月影清风_t
HTML
...作的时候碰到了这么个问题:如果我们只是简单粗暴地把新的HTML文档直接塞到标签里面去的话,那么这个新的HTML文档可不会被人家当作一个完整的网页结构来看待,而是会被理解成一段普通的文本内容。这就意味着它的内容不会被正确解析和显示。 在本篇文章中,我们将深入探讨这个问题,并给出解决方案。同时,我也会通过实际的例子来帮助你更好地理解和应用这些知识。 问题解析 首先,让我们来看看为什么在标签内插入一个新的HTML文档时,如果未指定其内容是HTML文档的部分,它将被视为文本而不是一个完整的HTML文档。 这是因为浏览器在解析HTML文档时,会从上到下逐行扫描文档,遇到标签时就会开始解析该HTML文档。然后,它会在找到标签之前一直解析这个HTML文档。因此,如果你在一个标签内插入一个新的HTML文档,而这个新的HTML文档没有标签,那么浏览器就会将这个新的HTML文档视为文本,而不是一个完整的HTML文档。 解决方案 那么,如何解决这个问题呢?一种常见的方法是在新的HTML文档中添加一个标签。例如: html New HTML Document This is the content of the new HTML document. 这样,浏览器就可以正确地解析和显示这个新的HTML文档了。 除了这种方法之外,还有一些其他的解决方案。例如,你可以使用JavaScript或者其他编程语言来动态生成新的HTML文档。这个方法的好处在于,它赋予了你更大的灵活性去随心所欲地掌控新HTML文档的内容布局和结构设计,就像你亲手捏泥巴一样自由自在。 总的来说,无论你选择哪种方法,都需要确保你的新的HTML文档有一个完整的HTML结构,包括、和等标签。这样才能让浏览器正确地解析和显示你的新HTML文档。 结论 在本文中,我们讨论了一个常见的问题:在标签内插入一个新的HTML文档时,如果未指定其内容是HTML文档的部分,它将被视为文本而不是一个完整的HTML文档。然后,我们提供了一些解决方案,并给出了实际的例子来帮助你更好地理解和应用这些知识。 在进行网页开发时,我们需要时刻注意这些问题,以便能够编写出高质量的HTML文档。同时呢,我们也要不断充电学习、积极摸索,这样才能时刻准备好,去应对各种意想不到的挑战和问题!
2023-04-15 17:36:32
543
岁月如歌-t
Python
...法在实际运用中存在的问题,如对初始质心敏感、容易陷入局部最优等,学者们不断提出新的优化策略与变种算法,如自适应模糊C均值算法、概率模糊C均值算法等,这些研究成果不仅丰富了聚类理论,也为实际问题解决提供了更多选择(可查阅最新的国际人工智能与数据挖掘会议或期刊论文获取最新动态)。 总之,FCM算法作为经典且灵活的聚类工具,在不断发展的数据科学领域中持续焕发活力,并通过与新兴技术结合及自身的迭代优化,展现出广阔的应用前景。读者可通过追踪最新的科研成果和实践案例,深入理解并掌握这一算法在现实世界中的具体应用与价值。
2023-07-03 21:33:00
63
追梦人_t
转载文章
...d-api │ 0.10.4 │ │ Uninstalled │ odl-neutron-northbound-api-0.10.4 │ OpenDaylight :: Neutron :: Northbound feature:install odl-neutron-northbound-api feature:install odl-netvirt-openstack odl-dlux-core odl-mdsal-apidocs feature:install odl-ovsdb-openstack odl-netvirt-sfc JAVA_HOME=/usr/lib/jvm/java-1.8.0-openjdk CLASSPATH=.:$JAVA_HOME/lib/tools.jar PATH=$JAVA_HOME/bin:$PATH JVM_OPTS="-Xms256m -XX:PermSize=256m -XX:MaxPermSize=512m" MAVEN_OPTS="$MAVEN_OPTS -Xms512m -Xmx1024m -XX:PermSize=256m -XX:MaxPermSize=512m" export MAVEN_OPTS JAVA_HOME CLASSPATH JVM_OPTS PATH [root@localhost ~] netstat -ntpl Active Internet connections (only servers) Proto Recv-Q Send-Q Local Address Foreign Address State PID/Program name tcp 0 0 0.0.0.0:22 0.0.0.0: LISTEN 3327/sshd tcp 0 0 127.0.0.1:25 0.0.0.0: LISTEN 3620/master tcp6 0 0 :::6633 ::: LISTEN 868/java tcp6 0 0 127.0.0.1:1099 ::: LISTEN 868/java tcp6 0 0 :::6640 ::: LISTEN 868/java tcp6 0 0 127.0.0.1:6644 ::: LISTEN 868/java tcp6 0 0 :::8181 ::: LISTEN 868/java tcp6 0 0 127.0.0.1:2550 ::: LISTEN 868/java tcp6 0 0 :::22 ::: LISTEN 3327/sshd tcp6 0 0 :::8185 ::: LISTEN 868/java tcp6 0 0 127.0.0.1:44601 ::: LISTEN 868/java tcp6 0 0 :::33273 ::: LISTEN 868/java tcp6 0 0 ::1:25 ::: LISTEN 3620/master tcp6 0 0 :::44444 ::: LISTEN 868/java tcp6 0 0 :::6653 ::: LISTEN 868/java tcp6 0 0 :::39169 ::: LISTEN 868/java tcp6 0 0 :::8101 ::: LISTEN 868/java tcp6 0 0 :::6886 ::: LISTEN 868/java openstack配置 openstack的networking-odl插件安装方式 https://docs.openstack.org/networking-odl/latest/install/installation.htmlodl-installation yum install python-networking-odl.noarch -y https://docs.openstack.org/networking-odl/latest/install/installation.htmlnetworking-odl-configuration systemctl restart neutron-server /etc/neutron/plugins/ml2 测试端口可连接性 curl -u admin:admin http://10.13.80.34:8181/controller/nb/v2/neutron/networks odl配置文件修改 etc/custom.properties ovsdb.l3.fwd.enabled=yes ovsdb.l3gateway.mac=0a:00:27:00:00:0d telnet 10.13.80.34 8181 netstat -nlp | grep 8181 telnet 127.0.0.1 8181 telnet 10.13.80.34 8181 systemctl status firewall iptables iptables -nvL iptables -F 清空iptables openstack server create --flavor tiny --image cirros --nic net-id=24449ee2-b84e-493f-8d76-139ac3e4f3cd --key-name mykey provider-instance nova service-list nova show ae5e26d1-c84d-40fa-bb27-f0b46d6a7061 查看虚机详情 ovs-vsctl set Open_vSwitch 89444614-3bf8-4d7a-b3a0-df5d20b48b7a other_config={'local_ip'='192.168.56.102'} ovs-vsctl set Open_vSwitch b084eccf-b92e-470c-8dff-8549e92c2104 other_config={'local_ip'='192.168.56.122'} ovs-vsctl list interface eth0 ovs-appctl fdb/show br-int [root@rcontroller01 ~] openstack security group rule list 2e19a748-9086-49f8-9498-01abc1a964fe 一个神奇的命令 +--------------------------------------+-------------+-----------+------------+--------------------------------------+ | ID | IP Protocol | IP Range | Port Range | Remote Security Group | +--------------------------------------+-------------+-----------+------------+--------------------------------------+ | 0184e6b3-4f7f-4fd5-8125-b80682e7ee48 | None | None | | 2e19a748-9086-49f8-9498-01abc1a964fe | | 1e0bfedc-8f25-408a-9328-708113bbbc52 | icmp | 0.0.0.0/0 | | None | | 39116d39-454b-4d82-867e-bbfd3ea63182 | None | None | | None | | 4032366f-3ac9-4862-85a7-c7411a8b7678 | None | None | | 2e19a748-9086-49f8-9498-01abc1a964fe | | dc7bc251-f0d0-456a-9102-c5b66646aa84 | tcp | 0.0.0.0/0 | 22:22 | None | | ddacf7ea-57ea-4c8a-8b68-093766284595 | None | None | | None | +--------------------------------------+-------------+-----------+------------+--------------------------------------+ dpif/dump-flows dp 想控制端打印dp中流表的所有条目。 这个命令主要来与debugOpen Vswitch.它所打印的流表不是openFlow的流条目。 它打印的是由dp模块维护的简单的流。 如果你想查看OpenFlow条目,请使用ovs-ofctl dump-flows。dpif/del-fow dp 删除指定dp上所有流表。同上所述,这些不是OpenFlow流表。 ovs-appctl dpif/dump-flows br-int 创建网络 openstack network create --share --external --provider-physical-network provider --provider-network-type flat provider $ openstack subnet create --network provider \ --allocation-pool start=192.168.56.100,end=192.168.56.200 \ --dns-nameserver 8.8.8.8 --gateway 192.168.56.1 \ --subnet-range 192.168.56.0/24 provider openstack network create selfservice $ openstack subnet create --network selfservice \ --dns-nameserver 8.8.8.8 --gateway 192.168.1.1 \ --subnet-range 192.168.1.0/24 selfservice openstack router create router openstack router add subnet router selfservice openstack router set router --external-gateway provider openstack port list --router router +--------------------------------------+------+-------------------+-------------------------------------------------------------------------------+--------+ | ID | Name | MAC Address | Fixed IP Addresses | Status | +--------------------------------------+------+-------------------+-------------------------------------------------------------------------------+--------+ | bff6605d-824c-41f9-b744-21d128fc86e1 | | fa:16:3e:2f:34:9b | ip_address='172.16.1.1', subnet_id='3482f524-8bff-4871-80d4-5774c2730728' | ACTIVE | | d6fe98db-ae01-42b0-a860-37b1661f5950 | | fa:16:3e:e8:c1:41 | ip_address='203.0.113.102', subnet_id='5cc70da8-4ee7-4565-be53-b9c011fca011' | ACTIVE | +--------------------------------------+------+-------------------+-------------------------------------------------------------------------------+--------+ $ ping -c 4 203.0.113.102 创建虚机 openstack keypair list $ ssh-keygen -q -N "" $ openstack keypair create --public-key ~/.ssh/id_rsa.pub mykey openstack flavor list openstack image list openstack network list openstack server create --flavor tiny --image cirros --nic net-id=27616098-0374-4ab4-95a8-b5bf4839dcf8 --key-name mykey provider-instance 网络配置 python /usr/lib/python2.7/site-packages/networking_odl/cmd/set_ovs_hostconfigs.py --ovs_hostconfigs='{ "ODL L2": { "allowed_network_types": [ "flat", "vlan", "vxlan" ], "bridge_mappings": { "provider": "br-int" }, "supported_vnic_types": [ { "vnic_type": "normal", "vif_type": "ovs", "vif_details": {} } ] }, "ODL L3": {} }' ovs-vsctl list open . [2019/1/16 19:09] 高正伟: ovs-vsctl set Open_vSwitch . other_config:local_ip=hostip ovs-vsctl set Open_vSwitch . other_config:local_ip=192.168.56.122 ovs-vsctl set Open_vSwitch . other_config:remote_ip=192.168.56.122 ovs-vsctl remove interface tunca7b782f232 options remote_ip ovs-vsctl set Open_vSwitch . other_config:provider_mappings=provider:br-ex ovs-vsctl set Open_vSwitch . external_ids:provider_mappings="{\"provider\": \"br-ex\"}" 清空 ovs-vsctl clear Open_vSwitch . external_ids ovs-vsctl set-manager tcp:10.13.80.34:6640 ovs-vsctl set-controller br-ex tcp:10.13.80.34:6640 ovs-vsctl del-controller br-ex sudo neutron-odl-ovs-hostconfig ovs-vsctl show ovs-vsctl add-port <bridge name> <port name> ovs-vsctl add-port br-ex enp0s10 ovs-vsctl del-port br-ex phy-br-ex ovs-vsctl del-port br-ex tun2ad7e9e91e4 重启odl后 systemctl restart openvswitch.service systemctl restart neutron-server.service systemctl stop neutron-server.service 创建虚机 openstack network create --share --external --provider-physical-network provider --provider-network-type flat provider openstack subnet create --network provider --allocation-pool start=192.168.56.2,end=192.168.56.100 --dns-nameserver 8.8.8.8 --gateway 192.168.56.1 --subnet-range 192.168.56.0/24 provider nova boot --image cirros --flavor tiny --nic net-id= --availability-zone nova:rcontroller01 vm-01 openstack server create --flavor tiny --image cirros --nic net-id= --key-name mykey test nova boot --image cirros --flavor tiny --nic net-id=0fe983c2-8178-403b-a00e-e8561580b210 --availability-zone nova:rcontroller01 vm-01 虚机可以学习到mac但是ping不通 抓包,先在虚机网卡上抓包, 然后在br-int上抓包 发现虚拟网卡上是发送了icmp请求报文的,但是br-int上没有 查看报文情况 [root@rcontroller01 ~] ovs-appctl dpif/dump-flows br-int recirc_id(0),tunnel(tun_id=0x0,src=192.168.56.102,dst=192.168.56.122,flags(-df-csum+key)),in_port(4),eth(),eth_type(0x0800),ipv4(proto=17,frag=no),udp(dst=3784), packets:266436, bytes:17584776, used:0.591s, actions:userspace(pid=4294962063,slow_path(bfd)) recirc_id(0xa0),in_port(5),ct_state(+new-est-rel-inv+trk),ct_mark(0/0x1),eth(),eth_type(0x0800),ipv4(frag=no), packets:148165, bytes:14520170, used:0.566s, actions:drop recirc_id(0),in_port(3),eth(),eth_type(0x0806), packets:1, bytes:60, used:5.228s, actions:drop recirc_id(0),tunnel(tun_id=0xb,src=192.168.56.102,dst=192.168.56.122,flags(-df-csum+key)),in_port(4),eth(dst=fa:16:3e:ab:ba:7e),eth_type(0x0806), packets:0, bytes:0, used:never, actions:5 recirc_id(0),in_port(5),eth(src=fa:16:3e:ab:ba:7e),eth_type(0x0800),ipv4(src=192.168.0.16,proto=1,frag=no), packets:148165, bytes:14520170, used:0.566s, actions:ct(zone=5004),recirc(0xa0) recirc_id(0),in_port(3),eth(),eth_type(0x0800),ipv4(frag=no), packets:886646, bytes:316947183, used:0.210s, flags:SFPR., actions:drop recirc_id(0),in_port(5),eth(src=fa:16:3e:ab:ba:7e,dst=fa:16:3e:7d:95:75),eth_type(0x0806),arp(sip=192.168.0.16,tip=192.168.0.5,op=1/0xff,sha=fa:16:3e:ab:ba:7e), packets:0, bytes:0, used:never, actions:userspace(pid=4294961925,controller(reason=4,dont_send=0,continuation=0,recirc_id=4618,rule_cookie=0x822002d,controller_id=0,max_len=65535)),set(tunnel(tun_id=0xb,src=192.168.56.122,dst=192.168.56.102,ttl=64,tp_dst=4789,flags(df|key))),4 安全组设置 openstack security group rule create --proto tcp 2e19a748-9086-49f8-9498-01abc1a964fe openstack security group rule create --proto tcp 6095293d-c2cd-433d-8a8f-e77ecb03609e openstack security group rule create --proto udp 2e19a748-9086-49f8-9498-01abc1a964fe openstack security group rule create --proto udp 6095293d-c2cd-433d-8a8f-e77ecb03609e ovs-vsctl add-port br-ex "ex-patch-int" ovs-vsctl set interface "ex-patch-int" type=patch ovs-vsctl set interface "ex-patch-int" options:peer=int-patch-ex ovs-vsctl add-port br-int "int-patch-ex" ovs-vsctl set interface "int-patch-ex" type=patch ovs-vsctl set interface "int-patch-ex" options:peer=ex-patch-int ovs-vsctl del-port br-ex "ex-patch-int" ovs-vsctl del-port br-int "int-patch-ex" ovs-vsctl del-port br-ex enp0s9 ovs-vsctl add-port br-int enp0s9 ovs-appctl ofproto/trace 重要命令 sudo ovs-ofctl -O OpenFlow13 show br-int sudo ovs-appctl ofproto/trace br-int "in_port=5,ip,nw_src=192.168.0.16,nw_dst=192.168.0.5" ovs-appctl dpctl/dump-conntrack 11.查看接口id等 ovs-appctl dpif/show 12.查看接口统计 ovs-ofctl dump-ports br-int 查看接口 sudo ovs-ofctl show br-int -O OpenFlow13 ovs常用命令 控制管理类 1.查看网桥和端口 ovs-vsctl show 1 2.创建一个网桥 ovs-vsctl add-br br0 ovs-vsctl set bridge br0 datapath_type=netdev 1 2 3.添加/删除一个端口 for system interfaces ovs-vsctl add-port br0 eth1 ovs-vsctl del-port br0 eth1 for DPDK ovs-vsctl add-port br0 dpdk1 -- set interface dpdk1 type=dpdk options:dpdk-devargs=0000:01:00.0 for DPDK bonds ovs-vsctl add-bond br0 dpdkbond0 dpdk1 dpdk2 \ -- set interface dpdk1 type=dpdk options:dpdk-devargs=0000:01:00.0 \ -- set interface dpdk2 type=dpdk options:dpdk-devargs=0000:02:00.0 1 2 3 4 5 6 7 8 9 4.设置/清除网桥的openflow协议版本 ovs-vsctl set bridge br0 protocols=OpenFlow13 ovs-vsctl clear bridge br0 protocols 1 2 5.查看某网桥当前流表 ovs-ofctl dump-flows br0 ovs-ofctl -O OpenFlow13 dump-flows br0 ovs-appctl bridge/dump-flows br0 1 2 3 6.设置/删除控制器 ovs-vsctl set-controller br0 tcp:1.2.3.4:6633 ovs-vsctl del-controller br0 1 2 7.查看控制器列表 ovs-vsctl list controller 1 8.设置/删除被动连接控制器 ovs-vsctl set-manager tcp:1.2.3.4:6640 ovs-vsctl get-manager ovs-vsctl del-manager 1 2 3 9.设置/移除可选选项 ovs-vsctl set Interface eth0 options:link_speed=1G ovs-vsctl remove Interface eth0 options link_speed 1 2 10.设置fail模式,支持standalone或者secure standalone(default):清除所有控制器下发的流表,ovs自己接管 secure:按照原来流表继续转发 ovs-vsctl del-fail-mode br0 ovs-vsctl set-fail-mode br0 secure ovs-vsctl get-fail-mode br0 1 2 3 11.查看接口id等 ovs-appctl dpif/show 1 12.查看接口统计 ovs-ofctl dump-ports br0 1 流表类 流表操作 1.添加普通流表 ovs-ofctl add-flow br0 in_port=1,actions=output:2 1 2.删除所有流表 ovs-ofctl del-flows br0 1 3.按匹配项来删除流表 ovs-ofctl del-flows br0 "in_port=1" 1 匹配项 1.匹配vlan tag,范围为0-4095 ovs-ofctl add-flow br0 priority=401,in_port=1,dl_vlan=777,actions=output:2 1 2.匹配vlan pcp,范围为0-7 ovs-ofctl add-flow br0 priority=401,in_port=1,dl_vlan_pcp=7,actions=output:2 1 3.匹配源/目的MAC ovs-ofctl add-flow br0 in_port=1,dl_src=00:00:00:00:00:01/00:00:00:00:00:01,actions=output:2 ovs-ofctl add-flow br0 in_port=1,dl_dst=00:00:00:00:00:01/00:00:00:00:00:01,actions=output:2 1 2 4.匹配以太网类型,范围为0-65535 ovs-ofctl add-flow br0 in_port=1,dl_type=0x0806,actions=output:2 1 5.匹配源/目的IP 条件:指定dl_type=0x0800,或者ip/tcp ovs-ofctl add-flow br0 ip,in_port=1,nw_src=10.10.0.0/16,actions=output:2 ovs-ofctl add-flow br0 ip,in_port=1,nw_dst=10.20.0.0/16,actions=output:2 1 2 6.匹配协议号,范围为0-255 条件:指定dl_type=0x0800或者ip ICMP ovs-ofctl add-flow br0 ip,in_port=1,nw_proto=1,actions=output:2 7.匹配IP ToS/DSCP,tos范围为0-255,DSCP范围为0-63 条件:指定dl_type=0x0800/0x86dd,并且ToS低2位会被忽略(DSCP值为ToS的高6位,并且低2位为预留位) ovs-ofctl add-flow br0 ip,in_port=1,nw_tos=68,actions=output:2 ovs-ofctl add-flow br0 ip,in_port=1,ip_dscp=62,actions=output:2 8.匹配IP ecn位,范围为0-3 条件:指定dl_type=0x0800/0x86dd ovs-ofctl add-flow br0 ip,in_port=1,ip_ecn=2,actions=output:2 9.匹配IP TTL,范围为0-255 ovs-ofctl add-flow br0 ip,in_port=1,nw_ttl=128,actions=output:2 10.匹配tcp/udp,源/目的端口,范围为0-65535 匹配源tcp端口179 ovs-ofctl add-flow br0 tcp,tcp_src=179/0xfff0,actions=output:2 匹配目的tcp端口179 ovs-ofctl add-flow br0 tcp,tcp_dst=179/0xfff0,actions=output:2 匹配源udp端口1234 ovs-ofctl add-flow br0 udp,udp_src=1234/0xfff0,actions=output:2 匹配目的udp端口1234 ovs-ofctl add-flow br0 udp,udp_dst=1234/0xfff0,actions=output:2 11.匹配tcp flags tcp flags=fin,syn,rst,psh,ack,urg,ece,cwr,ns ovs-ofctl add-flow br0 tcp,tcp_flags=ack,actions=output:2 12.匹配icmp code,范围为0-255 条件:指定icmp ovs-ofctl add-flow br0 icmp,icmp_code=2,actions=output:2 13.匹配vlan TCI TCI低12位为vlan id,高3位为priority,例如tci=0xf123则vlan_id为0x123和vlan_pcp=7 ovs-ofctl add-flow br0 in_port=1,vlan_tci=0xf123,actions=output:2 14.匹配mpls label 条件:指定dl_type=0x8847/0x8848 ovs-ofctl add-flow br0 mpls,in_port=1,mpls_label=7,actions=output:2 15.匹配mpls tc,范围为0-7 条件:指定dl_type=0x8847/0x8848 ovs-ofctl add-flow br0 mpls,in_port=1,mpls_tc=7,actions=output:2 1 16.匹配tunnel id,源/目的IP 匹配tunnel id ovs-ofctl add-flow br0 in_port=1,tun_id=0x7/0xf,actions=output:2 匹配tunnel源IP ovs-ofctl add-flow br0 in_port=1,tun_src=192.168.1.0/255.255.255.0,actions=output:2 匹配tunnel目的IP ovs-ofctl add-flow br0 in_port=1,tun_dst=192.168.1.0/255.255.255.0,actions=output:2 一些匹配项的速记符 速记符 匹配项 ip dl_type=0x800 ipv6 dl_type=0x86dd icmp dl_type=0x0800,nw_proto=1 icmp6 dl_type=0x86dd,nw_proto=58 tcp dl_type=0x0800,nw_proto=6 tcp6 dl_type=0x86dd,nw_proto=6 udp dl_type=0x0800,nw_proto=17 udp6 dl_type=0x86dd,nw_proto=17 sctp dl_type=0x0800,nw_proto=132 sctp6 dl_type=0x86dd,nw_proto=132 arp dl_type=0x0806 rarp dl_type=0x8035 mpls dl_type=0x8847 mplsm dl_type=0x8848 指令动作 1.动作为出接口 从指定接口转发出去 ovs-ofctl add-flow br0 in_port=1,actions=output:2 1 2.动作为指定group group id为已创建的group table ovs-ofctl add-flow br0 in_port=1,actions=group:666 1 3.动作为normal 转为L2/L3处理流程 ovs-ofctl add-flow br0 in_port=1,actions=normal 1 4.动作为flood 从所有物理接口转发出去,除了入接口和已关闭flooding的接口 ovs-ofctl add-flow br0 in_port=1,actions=flood 1 5.动作为all 从所有物理接口转发出去,除了入接口 ovs-ofctl add-flow br0 in_port=1,actions=all 1 6.动作为local 一般是转发给本地网桥 ovs-ofctl add-flow br0 in_port=1,actions=local 1 7.动作为in_port 从入接口转发回去 ovs-ofctl add-flow br0 in_port=1,actions=in_port 1 8.动作为controller 以packet-in消息上送给控制器 ovs-ofctl add-flow br0 in_port=1,actions=controller 1 9.动作为drop 丢弃数据包操作 ovs-ofctl add-flow br0 in_port=1,actions=drop 1 10.动作为mod_vlan_vid 修改报文的vlan id,该选项会使vlan_pcp置为0 ovs-ofctl add-flow br0 in_port=1,actions=mod_vlan_vid:8,output:2 1 11.动作为mod_vlan_pcp 修改报文的vlan优先级,该选项会使vlan_id置为0 ovs-ofctl add-flow br0 in_port=1,actions=mod_vlan_pcp:7,output:2 1 12.动作为strip_vlan 剥掉报文内外层vlan tag ovs-ofctl add-flow br0 in_port=1,actions=strip_vlan,output:2 1 13.动作为push_vlan 在报文外层压入一层vlan tag,需要使用openflow1.1以上版本兼容 ovs-ofctl add-flow -O OpenFlow13 br0 in_port=1,actions=push_vlan:0x8100,set_field:4097-\>vlan_vid,output:2 1 ps: set field值为4096+vlan_id,并且vlan优先级为0,即4096-8191,对应的vlan_id为0-4095 14.动作为push_mpls 修改报文的ethertype,并且压入一个MPLS LSE ovs-ofctl add-flow br0 in_port=1,actions=push_mpls:0x8847,set_field:10-\>mpls_label,output:2 1 15.动作为pop_mpls 剥掉最外层mpls标签,并且修改ethertype为非mpls类型 ovs-ofctl add-flow br0 mpls,in_port=1,mpls_label=20,actions=pop_mpls:0x0800,output:2 1 16.动作为修改源/目的MAC,修改源/目的IP 修改源MAC ovs-ofctl add-flow br0 in_port=1,actions=mod_dl_src:00:00:00:00:00:01,output:2 修改目的MAC ovs-ofctl add-flow br0 in_port=1,actions=mod_dl_dst:00:00:00:00:00:01,output:2 修改源IP ovs-ofctl add-flow br0 in_port=1,actions=mod_nw_src:192.168.1.1,output:2 修改目的IP ovs-ofctl add-flow br0 in_port=1,actions=mod_nw_dst:192.168.1.1,output:2 17.动作为修改TCP/UDP/SCTP源目的端口 修改TCP源端口 ovs-ofctl add-flow br0 tcp,in_port=1,actions=mod_tp_src:67,output:2 修改TCP目的端口 ovs-ofctl add-flow br0 tcp,in_port=1,actions=mod_tp_dst:68,output:2 修改UDP源端口 ovs-ofctl add-flow br0 udp,in_port=1,actions=mod_tp_src:67,output:2 修改UDP目的端口 ovs-ofctl add-flow br0 udp,in_port=1,actions=mod_tp_dst:68,output:2 18.动作为mod_nw_tos 条件:指定dl_type=0x0800 修改ToS字段的高6位,范围为0-255,值必须为4的倍数,并且不会去修改ToS低2位ecn值 ovs-ofctl add-flow br0 ip,in_port=1,actions=mod_nw_tos:68,output:2 1 19.动作为mod_nw_ecn 条件:指定dl_type=0x0800,需要使用openflow1.1以上版本兼容 修改ToS字段的低2位,范围为0-3,并且不会去修改ToS高6位的DSCP值 ovs-ofctl add-flow br0 ip,in_port=1,actions=mod_nw_ecn:2,output:2 1 20.动作为mod_nw_ttl 修改IP报文ttl值,需要使用openflow1.1以上版本兼容 ovs-ofctl add-flow -O OpenFlow13 br0 in_port=1,actions=mod_nw_ttl:6,output:2 1 21.动作为dec_ttl 对IP报文进行ttl自减操作 ovs-ofctl add-flow br0 in_port=1,actions=dec_ttl,output:2 1 22.动作为set_mpls_label 对报文最外层mpls标签进行修改,范围为20bit值 ovs-ofctl add-flow br0 in_port=1,actions=set_mpls_label:666,output:2 1 23.动作为set_mpls_tc 对报文最外层mpls tc进行修改,范围为0-7 ovs-ofctl add-flow br0 in_port=1,actions=set_mpls_tc:7,output:2 1 24.动作为set_mpls_ttl 对报文最外层mpls ttl进行修改,范围为0-255 ovs-ofctl add-flow br0 in_port=1,actions=set_mpls_ttl:255,output:2 1 25.动作为dec_mpls_ttl 对报文最外层mpls ttl进行自减操作 ovs-ofctl add-flow br0 in_port=1,actions=dec_mpls_ttl,output:2 1 26.动作为move NXM字段 使用move参数对NXM字段进行操作 将报文源MAC复制到目的MAC字段,并且将源MAC改为00:00:00:00:00:01 ovs-ofctl add-flow br0 in_port=1,actions=move:NXM_OF_ETH_SRC[]-\>NXM_OF_ETH_DST[],mod_dl_src:00:00:00:00:00:01,output:2 1 2 ps: 常用NXM字段参照表 NXM字段 报文字段 NXM_OF_ETH_SRC 源MAC NXM_OF_ETH_DST 目的MAC NXM_OF_ETH_TYPE 以太网类型 NXM_OF_VLAN_TCI vid NXM_OF_IP_PROTO IP协议号 NXM_OF_IP_TOS IP ToS值 NXM_NX_IP_ECN IP ToS ECN NXM_OF_IP_SRC 源IP NXM_OF_IP_DST 目的IP NXM_OF_TCP_SRC TCP源端口 NXM_OF_TCP_DST TCP目的端口 NXM_OF_UDP_SRC UDP源端口 NXM_OF_UDP_DST UDP目的端口 NXM_OF_SCTP_SRC SCTP源端口 NXM_OF_SCTP_DST SCTP目的端口 27.动作为load NXM字段 使用load参数对NXM字段进行赋值操作 push mpls label,并且把10(0xa)赋值给mpls label ovs-ofctl add-flow br0 in_port=1,actions=push_mpls:0x8847,load:0xa-\>OXM_OF_MPLS_LABEL[],output:2 对目的MAC进行赋值 ovs-ofctl add-flow br0 in_port=1,actions=load:0x001122334455-\>OXM_OF_ETH_DST[],output:2 1 2 3 4 28.动作为pop_vlan 弹出报文最外层vlan tag ovs-ofctl add-flow br0 in_port=1,dl_type=0x8100,dl_vlan=777,actions=pop_vlan,output:2 1 meter表 常用操作 由于meter表是openflow1.3版本以后才支持,所以所有命令需要指定OpenFlow1.3版本以上 ps: 在openvswitch-v2.8之前的版本中,还不支持meter 在v2.8版本之后已经实现,要正常使用的话,需要注意的是datapath类型要指定为netdev,band type暂时只支持drop,还不支持DSCP REMARK 1.查看当前设备对meter的支持 ovs-ofctl -O OpenFlow13 meter-features br0 2.查看meter表 ovs-ofctl -O OpenFlow13 dump-meters br0 3.查看meter统计 ovs-ofctl -O OpenFlow13 meter-stats br0 4.创建meter表 限速类型以kbps(kilobits per second)计算,超过20kb/s则丢弃 ovs-ofctl -O OpenFlow13 add-meter br0 meter=1,kbps,band=type=drop,rate=20 同上,增加burst size参数 ovs-ofctl -O OpenFlow13 add-meter br0 meter=2,kbps,band=type=drop,rate=20,burst_size=256 同上,增加stats参数,对meter进行计数统计 ovs-ofctl -O OpenFlow13 add-meter br0 meter=3,kbps,stats,band=type=drop,rate=20,burst_size=256 限速类型以pktps(packets per second)计算,超过1000pkt/s则丢弃 ovs-ofctl -O OpenFlow13 add-meter br0 meter=4,pktps,band=type=drop,rate=1000 5.删除meter表 删除全部meter表 ovs-ofctl -O OpenFlow13 del-meters br0 删除meter id=1 ovs-ofctl -O OpenFlow13 del-meter br0 meter=1 6.创建流表 ovs-ofctl -O OpenFlow13 add-flow br0 in_port=1,actions=meter:1,output:2 group表 由于group表是openflow1.1版本以后才支持,所以所有命令需要指定OpenFlow1.1版本以上 常用操作 group table支持4种类型 all:所有buckets都执行一遍 select: 每次选择其中一个bucket执行,常用于负载均衡应用 ff(FAST FAILOVER):快速故障修复,用于检测解决接口等故障 indirect:间接执行,类似于一个函数方法,被另一个group来调用 1.查看当前设备对group的支持 ovs-ofctl -O OpenFlow13 dump-group-features br0 2.查看group表 ovs-ofctl -O OpenFlow13 dump-groups br0 3.创建group表 类型为all ovs-ofctl -O OpenFlow13 add-group br0 group_id=1,type=all,bucket=output:1,bucket=output:2,bucket=output:3 类型为select ovs-ofctl -O OpenFlow13 add-group br0 group_id=2,type=select,bucket=output:1,bucket=output:2,bucket=output:3 类型为select,指定hash方法(5元组,OpenFlow1.5+) ovs-ofctl -O OpenFlow15 add-group br0 group_id=3,type=select,selection_method=hash,fields=ip_src,bucket=output:2,bucket=output:3 4.删除group表 ovs-ofctl -O OpenFlow13 del-groups br0 group_id=2 5.创建流表 ovs-ofctl -O OpenFlow13 add-flow br0 in_port=1,actions=group:2 goto table配置 数据流先从table0开始匹配,如actions有goto_table,再进行后续table的匹配,实现多级流水线,如需使用goto table,则创建流表时,指定table id,范围为0-255,不指定则默认为table0 1.在table0中添加一条流表条目 ovs-ofctl add-flow br0 table=0,in_port=1,actions=goto_table=1 2.在table1中添加一条流表条目 ovs-ofctl add-flow br0 table=1,ip,nw_dst=10.10.0.0/16,actions=output:2 tunnel配置 如需配置tunnel,必需确保当前系统对各tunnel的remote ip网络可达 gre 1.创建一个gre接口,并且指定端口id=1001 ovs-vsctl add-port br0 gre1 -- set Interface gre1 type=gre options:remote_ip=1.1.1.1 ofport_request=1001 2.可选选项 将tos或者ttl在隧道上继承,并将tunnel id设置成123 ovs-vsctl set Interface gre1 options:tos=inherit options:ttl=inherit options:key=123 3.创建关于gre流表 封装gre转发 ovs-ofctl add-flow br0 ip,in_port=1,nw_dst=10.10.0.0/16,actions=output:1001 解封gre转发 ovs-ofctl add-flow br0 in_port=1001,actions=output:1 vxlan 1.创建一个vxlan接口,并且指定端口id=2001 ovs-vsctl add-port br0 vxlan1 -- set Interface vxlan1 type=vxlan options:remote_ip=1.1.1.1 ofport_request=2001 2.可选选项 将tos或者ttl在隧道上继承,将vni设置成123,UDP目的端为设置成8472(默认为4789) ovs-vsctl set Interface vxlan1 options:tos=inherit options:ttl=inherit options:key=123 options:dst_port=8472 3.创建关于vxlan流表 封装vxlan转发 ovs-ofctl add-flow br0 ip,in_port=1,nw_dst=10.10.0.0/16,actions=output:2001 解封vxlan转发 ovs-ofctl add-flow br0 in_port=2001,actions=output:1 sflow配置 1.对网桥br0进行sflow监控 agent: 与collector通信所在的网口名,通常为管理口 target: collector监听的IP地址和端口,端口默认为6343 header: sFlow在采样时截取报文头的长度 polling: 采样时间间隔,单位为秒 ovs-vsctl -- --id=@sflow create sflow agent=eth0 target=\"10.0.0.1:6343\" header=128 sampling=64 polling=10 -- set bridge br0 sflow=@sflow 2.查看创建的sflow ovs-vsctl list sflow 3.删除对应的网桥sflow配置,参数为sFlow UUID ovs-vsctl remove bridge br0 sflow 7b9b962e-fe09-407c-b224-5d37d9c1f2b3 4.删除网桥下所有sflow配置 ovs-vsctl -- clear bridge br0 sflow 1 QoS配置 ingress policing 1.配置ingress policing,对接口eth0入流限速10Mbps ovs-vsctl set interface eth0 ingress_policing_rate=10000 ovs-vsctl set interface eth0 ingress_policing_burst=8000 2.清除相应接口的ingress policer配置 ovs-vsctl set interface eth0 ingress_policing_rate=0 ovs-vsctl set interface eth0 ingress_policing_burst=0 3.查看接口ingress policer配置 ovs-vsctl list interface eth0 4.查看网桥支持的Qos类型 ovs-appctl qos/show-types br0 端口镜像配置 1.配置eth0收到/发送的数据包镜像到eth1 ovs-vsctl -- set bridge br0 mirrors=@m \ -- --id=@eth0 get port eth0 \ -- --id=@eth1 get port eth1 \ -- --id=@m create mirror name=mymirror select-dst-port=@eth0 select-src-port=@eth0 output-port=@eth1 2.删除端口镜像配置 ovs-vsctl -- --id=@m get mirror mymirror -- remove bridge br0 mirrors @m 3.清除网桥下所有端口镜像配置 ovs-vsctl clear bridge br0 mirrors 4.查看端口镜像配置 ovs-vsctl get bridge br0 mirrors Open vSwitch中有多个命令,分别有不同的作用,大致如下: ovs-vsctl用于控制ovs db ovs-ofctl用于管理OpenFlow switch 的 flow ovs-dpctl用于管理ovs的datapath ovs-appctl用于查询和管理ovs daemon 转载于:https://www.cnblogs.com/liuhongru/p/10336849.html 本篇文章为转载内容。原文链接:https://blog.csdn.net/weixin_30876945/article/details/99916308。 该文由互联网用户投稿提供,文中观点代表作者本人意见,并不代表本站的立场。 作为信息平台,本站仅提供文章转载服务,并不拥有其所有权,也不对文章内容的真实性、准确性和合法性承担责任。 如发现本文存在侵权、违法、违规或事实不符的情况,请及时联系我们,我们将第一时间进行核实并删除相应内容。
2023-06-08 17:13:19
295
转载
Nginx
...口时超时丢包的原因及解决策略之后,我们不妨将视线转向网络性能优化和服务器配置的最新实践与研究。近期,随着云计算和大数据应用的飞速发展,网络环境的复杂性与服务器负载压力显著增加,这对网络连接稳定性和响应速度提出了更高要求。 例如,2022年的一项技术报告中,研究者们探讨了在大规模分布式系统环境下,如何通过深度调优Nginx及其他网络服务组件,以适应高并发、低延迟的需求。他们不仅关注到了proxy_connect_timeout等关键参数的设置,还提出了一套动态调整策略,可以根据实时网络状况进行智能适配,从而有效减少超时丢包现象。 同时,在网络架构层面,边缘计算和5G技术的发展为改善网络环境提供了新的解决方案。通过在更接近用户的边缘节点部署服务,可以大幅度降低网络延迟并缓解拥塞问题,从而避免tcping测试过程中可能出现的超时丢包情况。 此外,心跳包机制的实际运用也在不断丰富和完善。在某些前沿应用场景中,如物联网(IoT)设备通信,已经采用更为先进的双向心跳检测机制,并结合TCP keepalive特性,实现了对长连接状态的高效维护,进一步提升了服务可靠性。 综上所述,无论是从服务器配置的精细化管理,还是从网络基础设施的升级换代,都为我们应对tcping Nginx端口超时丢包等问题提供了有力武器。紧跟行业发展趋势和技术研究成果,将有助于我们在实际工作中更好地诊断并解决这类网络通讯难题。
2023-12-02 12:18:10
193
雪域高原_t
转载文章
在处理子集和问题时,深度优先搜索(DFS)与动态规划(DP)是两种常用的算法策略。实际上,在计算机科学和算法竞赛领域中,对于这类决策性问题的探讨持续不断。最近的一次国际编程大赛上,就有参赛者利用类似题目展示了如何灵活运用DFS进行状态搜索,并对小规模数据实现了高效求解。 同时,随着计算资源的增长和优化技术的进步,动态规划方法在解决背包问题等组合优化问题上的应用也在不断拓展。例如,一篇2023年发表于《ACM Transactions on Algorithms》的研究论文,深入研究了在物品价值与体积相等情况下背包问题的特殊结构,揭示了其恰好装满状态下的复杂性和最优解特性。 此外,针对更大数据规模的问题,一些研究者正探索结合贪心策略、剪枝技术和近似算法以降低时间复杂度。比如,一项最新研究成果提出了一种基于分支限界法和预处理技巧改进的搜索算法,能够有效应对大规模子集和问题,为实际应用提供了新的解决方案。 在实际编程实践中,数组排序往往是提高搜索效率的关键步骤,通过合理排序可以减少不必要的搜索空间。而在教育领域,诸如LeetCode、Codeforces等在线平台上的相关题目讨论和解题报告,也为我们理解此类问题提供了丰富的实例参考和实战经验。 综上所述,无论是在学术研究前沿还是编程实战层面,对“能否从数组中选择若干个数使其和为目标值”的问题探究,都在持续推动着算法设计与优化技术的发展,展现了算法在解决实际问题中的强大生命力。
2023-02-03 18:37:40
76
转载
c#
...发中,难免会遇到一些问题,尤其是在使用C进行编程的时候。尤其是当我们在运行程序时,总会遇到各种各样的错误。今天我们就来聊聊如何解决这些常见的错误。 二、错误分析与解决 首先我们要知道的是,任何错误都是可以通过分析找到解决办法的。所以,当我们遇到错误时,首先要做的就是找出错误的原因。而这就需要我们对代码有深入的理解和掌握。 三、常见错误类型及解决方案 1. 异常错误 这是最常见的错误类型,通常是由于代码中的逻辑错误或者数据异常引起的。例如: csharp int i = 10; int j = "hello"; int result = i + j; // 这里就会抛出一个异常,因为不能将字符串和整数相加 为了解决这种类型的错误,我们需要仔细检查代码,确保所有的数据类型都正确无误。如果需要的话,我们还能给程序加个异常处理机制,这样一来,就算遇到点儿小差错,程序也能稳稳当当地运行下去,不至于突然崩掉。 2. 资源泄露错误 这种错误通常发生在我们使用了某个资源(如文件、网络连接等)后忘记关闭的情况下。例如: csharp FileStream fs = new FileStream("test.txt", FileMode.Open); // ... 程序在这里做了一些操作 ... fs.Close(); // 忘记关闭流 为了解决这个问题,我们需要养成良好的编程习惯,在使用完资源后立即关闭。同时,我们也可以使用using语句块来自动管理资源,如下所示: csharp using (FileStream fs = new FileStream("test.txt", FileMode.Open)) { // ... 程序在这里做了一些操作 ... } 3. 编译错误 这种错误通常是由于语法错误或者编译器无法识别的语句引起的。例如: csharp public class MyClass { public void MyMethod() { System.out.println("Hello, World!"); // 这里就有一个编译错误,因为System.out.println是Java语言的语句,而不是C } } 为了解决这个问题,我们需要仔细检查我们的代码,并确保使用的语句是正确的。同时,我们还需要注意不同编程语言之间的差异。 四、总结 总的来说,解决编程错误并不是一件难事,只要我们有足够的耐心和细心,就一定能找到解决问题的方法。同时,我们也应该养成良好的编程习惯,避免出现不必要的错误。 最后,我希望这篇文章能够帮助你解决你在使用C编程时遇到的问题。如果你有任何疑问,欢迎留言讨论,我会尽力为你解答。 希望这篇文章对你有所帮助,也希望大家多多支持我!
2023-11-12 22:43:56
550
林中小径_t
Struts2
...'execute'”问题解析与解决方案 在我们深入使用Struts2框架进行Java Web开发时,偶尔会遇到一种常见的运行时异常——Java.lang.NullPointerException,尤其在Action类执行execute方法时。这篇东西,咱们就来点儿接地气的,从实际动手干的视角,一边瞅着代码实例,一边掰扯这个问题是怎么冒出来的、怎么把它摆平的,还有怎样提前给它上个“紧箍咒”,预防它再出来闹腾。 1. 异常现象分析 首先,让我们通过一个示例来直观感受一下这个问题。假设我们有一个简单的Struts2 Action类: java public class UserAction extends ActionSupport { private UserService userService; // 这是一个依赖注入的对象 public String execute() { User user = userService.getUserById(1); // 假设这里调用服务层获取用户信息 // ... 其他业务逻辑 return SUCCESS; } // getter 和 setter 方法省略... } 当执行上述execute方法时,如果出现NullPointerException,则意味着在执行userService.getUserById(1)这行代码时,userService对象未被正确初始化,其值为null。 2. 问题根源探究 原因一:依赖注入失败 在Struts2中,我们通常利用框架的依赖注入功能来实现Action和Service之间的解耦。就像刚才举的例子那样,如果咱们没有给userService这个家伙喂饱饭(也就是没有正确注入它),或者在喂饭的过程中出了岔子,那么到执行execute方法的时候,userService就会变成一个空肚子(null),这样一来,就难免会闹肚子(引发异常)了。 原因二:实例化时机不当 另一种可能的情况是,尽管在配置文件中设置了依赖注入,但可能由于某些原因(例如配置错误或加载顺序问题),导致注入的服务对象尚未初始化完成,此时访问也会抛出空指针异常。 3. 解决方案及示例 解决方案一:确保依赖注入生效 在Struts2的配置文件中(通常是struts.xml),我们需要明确指定Action类中需要注入的属性和服务对象的关系: xml /success.jsp userServiceBean 解决方案二:检查并修正实例化顺序 如果确认了依赖注入配置无误,但仍出现空指针异常,则应检查应用启动过程中相关Bean的加载顺序,确保在Action类执行execute方法之前,所有依赖的对象已经成功初始化。 解决方案三:防御性编程 无论何种情况,我们在编码时都应当遵循防御性编程原则,对可能为null的对象进行判空处理: java public class UserAction extends ActionSupport { private UserService userService; public String execute() { if (userService != null) { // 防御性判空 User user = userService.getUserById(1); // ... 其他业务逻辑 } else { System.out.println("userService is not initialized correctly!"); // 打印日志或采取其他容错处理 } return SUCCESS; } // getter 和 setter 方法省略... } 4. 总结与思考 面对“Java.lang.NullPointerException in Action class while executing method 'execute'”这样的问题,我们需要从多方面进行排查和解决。不仅仅是对Struts2框架的依赖注入机制了如指掌,更要像侦探一样时刻保持警惕,做好咱们的防御性编程工作。为啥呢?这就像是给程序穿上防弹衣,能有效防止那些突如其来的运行时异常搞崩我们的程序,让程序稳稳当当地跑起来,不尥蹶子。在实际做项目的时候,把这些技巧学懂了、用溜了,那咱们的开发速度和代码质量绝对会嗖嗖往上涨,没跑儿!
2023-06-26 11:07:11
70
青春印记
Docker
...,因此出现了操作超时问题;解决。 为了解决这个问题,我们可以采取以下方法;增加: 1. 增加时间限制;避免:通过修改 Docker 的设置文件;修改,可以增加 Docker 的时间限制;避免来避免使用;超时已超时的错误。比如;包含,在/etc/docker/daemon.json文件中添加以下内容: { "live-restore": true, "storage-driver": "overlay2", "iptables": false, "max-concurrent-downloads": 10, "max-concurrent-uploads": 10, "registry-mirrors": [ "http://dockerhub.azk8s.cn", "http://hub-mirror.c.163.com" ], "debug": false, "experimental": true, "log-driver": "json-file", "log-level": "warn", "metrics-addr": "0.0.0.0:9323", "default-shm-size": "8G" } 其中,max-concurrent-downloads和max-concurrent-uploads可以根据现实情况;相应进行校准;解决方法。 2. 改进;网络环境网络环境:在虚拟环境;任何地方与网络之间的通信方面,可以改进;网络环境网络环境来避免操作超时问题;解决。比如;包含,可以增加带宽资源;更改或者更改虚拟环境;任何地方所在的网络位置。 总而言之;需要,解决 Docker 使用;超时已超时的问题需要综合考虑多个要素;进行,并根据现实情况;相应进行相应的校准;解决方法。通过这些方法;增加,我们可以更好地利用 Docker 的虚拟环境;任何地方化发布;多个,增强;系统系统的稳定性和可用性。
2023-10-26 09:32:48
557
电脑达人
SeaTunnel
...用户可能会碰上这么个问题:SeaTunnel这小家伙,没法帮咱们截取屏幕或者视频画面。这篇文章将尝试解答这个问题,并提供可能的解决方案。 二、为什么SeaTunnel无法截取屏幕或视频? 有几个可能的原因导致SeaTunnel无法截取屏幕或视频: 1. SeaTunnel版本过旧 2. 操作系统兼容性问题 3. 权限设置限制 4. 屏幕分辨率过高或过低 5. 音频输入设备问题 三、如何解决SeaTunnel无法截取屏幕或视频的问题? 以下是一些可能的解决方案: 1. 更新SeaTunnel到最新版本 如果您的SeaTunnel版本过旧,可能会出现一些已知的问题,包括无法截取屏幕或视频。您可以访问SeaTunnel的官方网站下载最新版本的软件。 2. 确保操作系统兼容性 SeaTunnel需要与您的操作系统兼容才能正常工作。如果你正在用的是Windows 7或是更老的操作系统,碰到了些头疼的问题,那我建议你考虑一下给电脑升个级,换上个更新的操作系统版本吧。就像是给你的旧电脑换个新内核,让它重新焕发活力。 3. 检查权限设置 在某些情况下,SeaTunnel可能因为权限设置问题而无法截取屏幕或视频。试试看,先用鼠标右键点一下SeaTunnel的小图标,然后在弹出的菜单里选中“属性”这个选项。接下来,你会发现一个新页面跳出来了,这时候别慌,找到并切换到“安全”这个标签页。最后一步,留心检查一下是不是所有用户的权限都已经开启,都可以顺利访问。 4. 调整屏幕分辨率 如果您的屏幕分辨率过高或过低,可能会影响SeaTunnel的工作。您可以尝试调整屏幕分辨率来解决问题。 5. 检查音频输入设备 如果SeaTunnel无法截取视频,但可以截取屏幕和音频,那么问题可能出在音频输入设备上。您可以尝试重新连接音频输入设备,或者更换其他设备进行测试。 四、代码示例 以下是一个使用SeaTunnel截取屏幕的例子: python from selenium import webdriver import time driver = webdriver.Chrome() driver.get("http://www.google.com") time.sleep(5) 让页面加载完成 使用海隧道开始录制 driver.execute_script("seattlerecorder.start('output.mp4')") time.sleep(10) 录制10秒 结束录制 driver.execute_script("seattlerecorder.stop()") driver.quit() 以上代码使用了Selenium库来控制Chrome浏览器,首先打开Google首页,然后等待5秒钟让页面加载完成,然后开始使用SeaTunnel录制输出为'mp4'格式的屏幕,最后停止录制并关闭浏览器。 五、结论 SeaTunnel是一款强大的屏幕录制工具,但是在使用过程中可能会遇到一些问题,如无法截取屏幕或视频。经过这篇内容的详细介绍,相信你现在对这个问题可能出现的各种原因以及相应的解决办法已经心里有谱了。希望这些信息能帮您搞定SeaTunnel无法捕捉屏幕或视频的问题,让您顺利畅行无阻。
2023-10-29 17:27:43
78
青山绿水-t
Apache Lucene
...到下一个阈值(默认为10GB)。这种方式的好处是能相当给力地把控内存使用,不过呢,也可能让搜索速度没那么快了。 3. ConcurrentMergeScheduler:这个策略是并发的,它可以在不同的线程上同时进行合并,从而提高合并的速度。不过要注意,要是咱们把并发数量调得太大,可能会让CPU过于忙碌,忙到“火力全开”,这样一来,CPU使用率就嗖嗖地往上升啦。 四、如何优化Lucene索引段合并策略? 那么,我们如何根据自己的需求,选择合适的合并策略呢?以下是一些优化建议: 1. 根据内存大小调整合并阈值 如果你的服务器内存较小,可以考虑使用LogByteSizeMergePolicy,并降低其合并阈值,以减少内存占用。 2. 根据查询频率调整并发数量 如果你的应用程序需要频繁地进行搜索,可以考虑使用ConcurrentMergeScheduler,并增加其并发数量,以加快搜索速度。 3. 使用自定义的合并策略 如果你想实现更复杂的合并策略,例如先合并某些特定的段,再合并其他段,你可以编写自己的合并策略,并将其注册给Lucene。 总的来说,Lucene的索引段合并策略是一个复杂但又非常重要的问题。了解并巧妙运用合并策略后,咱们就能让Lucene这位搜索大神发挥出更强大的威力,这样一来,应用程序的性能也能蹭蹭地往上提升,用起来更加流畅顺滑,一点儿也不卡壳。
2023-03-19 15:34:42
397
岁月静好-t
NodeJS
...eJS版本>=10中尝试访问一个不是目录的文件时,就会出现这个问题。比如说,当我们试着把一个文件当作流来读取,但实际上人家是个文本文件的时候,就可能会碰上这个问题。那么,如何避免这个错误呢? 二、问题原因分析 “ENOENT: no such file or directory”错误的主要原因是我们的程序试图访问的文件或目录不存在。这可能是因为我们在编写代码时,不小心把文件或者目录的名字给写错了,要么就是那个文件或者目录被我们无意中删除了,或者它自己“跑路”去了其他地方。 在NodeJS版本>=10中,如果我们尝试将一个不是目录的文件作为目录来访问,就会出现“ENOTDIR: Not a directory”错误。这是因为,在NodeJS的世界里,甭管啥文件,统统都被视为普普通通的文件,而不是什么高大上的目录。因此,如果我们试图将一个文件作为目录来访问,就会出现这个错误。 三、解决方案 那么,如何解决“ENOTDIR: Not a directory”错误呢?下面是一些可能的解决方案: 1. 检查文件或目录是否存在 在访问文件或目录之前,我们需要先检查它们是否存在。如果它们不存在,我们就不能访问它们,否则就会出现“ENOENT: no such file or directory”错误。 示例代码如下: javascript let exists = fs.existsSync('file.txt'); if (!exists) { console.error('File not found!'); } 如果文件存在,我们就继续访问它。如果文件不存在,我们就输出一个错误消息。 2. 将文件视为普通文件,而不是目录 在NodeJS中,所有的文件都被视为普通文件,而不是目录。所以,如果我们心血来潮,硬要把一个文件当成文件夹来打开,系统就会抛出个“ENOTDIR:这不是个目录”的错误给我们,意思是它压根不是我们想找的文件夹。 因此,我们需要确保我们在访问文件时,将其视为普通文件,而不是目录。 示例代码如下: javascript fs.readFile('file.txt', 'utf8', function(err, data) { if (err) { if (err.code === 'EISDIR') { console.error('Cannot read from a directory!'); } else { console.error('An error occurred:', err); } } else { console.log(data); } }); 在这段代码中,我们首先尝试读取文件的内容。如果读取过程中发生错误,我们就检查错误代码。要是你遇到个错误代码"EISDIR",那咱就给用户撂个明白话儿:你这会儿是想从一个文件夹里头读取东西呢,这操作可不行。 3. 使用fs.stat()方法检查文件类型 我们也可以使用fs.stat()方法检查文件的类型。如果文件是一个目录,我们就不能将其作为普通文件来访问。 示例代码如下: javascript fs.stat('file.txt', function(err, stats) { if (err) { if (err.code === 'EISDIR') { console.error('Cannot read from a directory!'); } else { console.error('An error occurred:', err); } } else { if (stats.isDirectory()) { console.error('Cannot read from a directory!'); } else { console.log('Reading file...'); } } }); 在这段代码中,我们首先使用fs.stat()方法获取文件的统计信息。然后,我们检查文件的类型。如果文件是一个目录,我们就输出一个错误消息。否则,我们就开始读取文件的内容。 四、总结 总的来说,“ENOTDIR: Not a directory”错误是由于我们试图访问一个不是目录的文件或目录导致的。为了避免犯这个错误,咱们得保证自家的程序够机灵,能够准确地核实文件或者目录是不是真的存在。而且啊,它还要能聪明地分辨出啥时候该把一个东西看成普通的文件,而不是个目录。另外,咱们还可以用fs.stat()这个小技巧来瞅瞅文件的真身,确保咱不会把文件错认成目录,闹出乌龙。
2023-04-14 13:43:40
118
青山绿水-t
Bootstrap
...下拉菜单后无法收回?问题解析与解决之道 引言 Bootstrap,这个广受欢迎的前端框架以其强大的响应式设计和丰富的组件库深受开发者喜爱。不过,在实际用起来的时候,咱们可能会碰到一些小状况,就像这样:当用户点击创建的那个下拉菜单,菜单是会顺利打开,但是呢,它却不太听话,不会自己乖乖地收回去。这无疑影响了用户体验,让人略感困扰。本文将深入探讨这一现象,并通过实例代码一步步带你找到解决方案。 问题描述与重现 1. 下拉菜单的基本实现 首先,我们先来看看如何用 Bootstrap 5 创建一个基础的下拉菜单: html 下拉菜单 选项一 选项二 选项三 这段代码会生成一个按钮,点击后会展开下拉菜单,但如果没有正确的 JavaScript 配置,菜单可能无法在点击外部区域或选择菜单项后自动收回。 2. 无法收回的问题重现 当你尝试以上代码并发现下拉菜单在打开后无法自动关闭时,那很可能是因为你尚未引入或者正确配置 Bootstrap 的 JavaScript 插件。Bootstrap 的很多交互功能都需要依赖 jQuery 和 Popper.js 来实现动态效果。 解决方案 3. 引入必要的 JavaScript 库 确保你的项目已经正确引入了 jQuery、Popper.js 以及 Bootstrap 的 JavaScript 文件。例如: html 4. 初始化下拉菜单插件 Bootstrap 5 中的下拉菜单需要手动初始化其 JavaScript 功能。你可以在文档加载完毕后通过调用 bootstrap.Dropdown.getInstance 或 bootstrap.Dropdown.getOrCreateInstance 方法来初始化下拉菜单: javascript document.addEventListener('DOMContentLoaded', function () { var dropdowns = document.querySelectorAll('.dropdown-toggle') Array.from(dropdowns).forEach(function (dropdown) { bootstrap.Dropdown.getOrCreateInstance(dropdown) }) }) 上述代码会在页面加载完成后对所有带有 .dropdown-toggle 类名的元素进行下拉菜单初始化操作,这样一来,下拉菜单就可以正常地展开和收回了。 总结 通过上面的示例代码和解析,我们可以看到,使用 Bootstrap 创建下拉菜单时,不仅需要注意 HTML 结构,还需正确引入并初始化相关的 JavaScript 插件。当碰到“下拉菜单顽固不肯收回去”的状况时,咱们得淡定地、一步步地审查脚本的引用情况和初始化步骤,这样才能准确无误地找到问题的藏身之处。在编程这个领域里,每一个小细节都像一块积木一样重要,你可别小瞧了那些看似不起眼的小问题,它们就像隐藏在机器王国里的捣蛋鬼,随时可能给你惹出大乱子来。因此,让我们在探索与实践中,不断积累经验,提升技能,享受解决问题的乐趣吧!
2023-11-22 18:24:59
482
寂静森林_
站内搜索
用于搜索本网站内部文章,支持栏目切换。
知识学习
实践的时候请根据实际情况谨慎操作。
随机学习一条linux命令:
ln -s target link
- 创建符号链接。
推荐内容
推荐本栏目内的其它文章,看看还有哪些文章让你感兴趣。
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
历史内容
快速导航到对应月份的历史文章列表。
随便看看
拉到页底了吧,随便看看还有哪些文章你可能感兴趣。
时光飞逝
"流光容易把人抛,红了樱桃,绿了芭蕉。"