CMake

文章目录

  • 前言
  • 一、快速开始编译C/C++代码
    • 1. 只有源码的项目
    • 2. 包含库的项目
    • 3. 编译成库给他人使用
    • 使用cmake的流程
      • 1. 生成构建系统
      • 2. 执行构建
      • 3. 执行测试
      • 4. 安装 && 打包
  • 二、cmake 语法简介
    • 1 变量
    • 2 条件语句
    • 3 脚本命令
      • **消息打印**
      • **if-else**:
      • **list命令**:
      • **文件操作**
      • **配置文件生成**
      • 执行系统命令
      • 查找库文件
      • include其他模块
  • 三、配置案例
  • 四、配置说明
    • 配置项目
    • 指定编译器语言版本
    • 配置编译选项
    • debug or release
  • 附录
    • 内建变量
  • reference


前言

CMakeake是什么,是用来做什么的,以及如何?
cmake是一个用于管理和构建源代码的工具。
cmake广泛应用于编译C/C++代码,其实也可以用于编译其他语言。
cmake可实现跨平台构建、编译c/c++源码。
cmake除了用于编译外还提供了ctest, cpack用于测试,安装或打包。


第一章适用于新手,在不了解cmake语言的情况下可快速构建自己的项目

一、快速开始编译C/C++代码

1. 只有源码的项目

.
├── include
│   ├── hea2.h
│   └── head1.h
├── main.c
└── src├── fun1.c└── fun2.c

在根目录下新建CMakeLists.txt文件:

# 新项目必须的
cmake_minimum_required(VERSION 3.10)
# 新项目必须的
project(tutorial VERSION 1.0)
# 生成可执行文件必须的
add_executable(mainmain.csrc/fun1.csrc/fun2.c
)
# 指定头文件搜索路径
target_include_directories(main PUBLIC./include
)

接下来,在根目录下新建build目录,然后运行cmake命令

mkdir build && cd build
cmake ..	     # 生成构建系统
cmake --build .  #执行构建

2. 包含库的项目

和1的区别仅为在CMakeLists.txt中把依赖添加进去:

target_link_libraries(mainlibname1libname2
)
target_link_directories(main PUBLIC./lib
)

然后执行cmake 命令即可

3. 编译成库给他人使用

若需要将源文件编译成库而不是可执行文件,则仅需修改CMakeLists.txt文件即可
生成静态库:

add_library(main STATIC  # STATIC可写可不写main.csrc/fun1.csrc/fun2.c
)

生成动态库:

cmake_minimum_required(VERSION 3.10)project(tutorial VERSION 1.0)add_library(main SHAREDmain.csrc/fun1.csrc/fun2.c
)
target_include_directories(main PUBLIC./include
)

使用cmake的流程

  1. 生成构建系统(buildsystem,比如make工具对应的Makefile文件)
  2. 执行构建(比如make),声明目标文件
  3. 执行测试、安装或打包

1. 生成构建系统

通过cmake 命令生成构建系统

参数含义
-S指定源文件根目录,必须包含一个CMakeLists.txt文件
-B指定构建目录,构建生成的中间文件和目标文件的生成路径
-D指定变量,格式为-D =,-D后面的空格可以省略

指定当前目录为源文件目录,其中包含CMakeLists.txt文件,使用build目录作为构建目录
cmake -S . -B build -DCMAKE_BUILD_TYPE=Debug -DAUTHOR=me

使用-D设置的变量在CMakeLists.txt中生效,可以设置cmake的内置支持的一些变量控制构建的行为;当然也可以使用自定义的变量,在CMakeLists.txt中自行判断做不同的处理。

2. 执行构建

cmake --build .
–buid 后面跟构建系统所在目录

3. 执行测试

如果在CMakeLists.txt中添加如下包含测试的功能则在执行构建后可进行测试

enable_testing()
# include(CTest)  # 添加CTest模块, 可代替enable_testing
add_test(NAME normal_test COMMAND main )
set_tests_properties(normal_test PROPERTIES PASS_REGULAR_EXPRESSION "yes")  # main程序输出“yes”视为通过

4. 安装 && 打包

在CMakeLists.txt中添加如下内容:

install(TARGETS binary_nam DESTINATION ./install/bin)
install(FILES "include/head.h" DESTINATION ./install/inc)
install(EXPORT Json FILE FILE Json.cmake DESTINATION ./install/lib)

然后执行

cmake --install .
或者
cmake --install . --prefix "~/myuser/installdir"

打包部分测试未通过

二、cmake 语法简介

1 变量

CMake中使用setunset命令设置或者取消设置变量

一般变量
设置的变量可以是字符串,数字或者列表(直接设置多个值,或者使用分号隔开的字符串格式为"val1;val2;val3")

# set var
set(val1 a)
set(val2 "hello world")
# set list
set(list1 a b)  # saved as "a;b"
set(list2 a;b)
set(list3 "a;b")set(num 90)  # saved as string, but can compare with other number string
set(falg ON) # bool value
  1. 如果要设置的变量值包含空格,则需要使用双引号或者使用""转义,否则可以省略双引号;
  2. 如果设置多个值或者字符串值的中间有";“,则保存成list,同样是以”;"分割的字符串;
  3. 变量可以被list命令操作,单个值的变量相当于只有一个元素的列表;
  4. 引用变量:${<variable>},在if()条件判断中可以简化为只用变量名<variable>。

Cache变量
Cache变量(缓存条目,cache entries)的作用主要是为了提供用户配置选项,如果用户没有指定,则使用默认值,设置方法如下:

# set(<variable> <value>... CACHE <type> <docstring> [FORCE])
set(CACHE_VAR "Default cache value" CACHE STRING "A sample for cache variable")
  1. 主要为了提供可配置变量,比如编译开关;
  2. 引用CACHE变量:$CACHE{}
  3. Cache变量会被保存在构建目录下的CMakeCache.txt中,缓存起来之后是不变的,除非重新配置更新

环境变量

修改当前处理进程的环境变量,设置和引用格式为:

# set(ENV{<variable>} [<value>])
set(ENV{ENV_VAR} "$ENV{PATH}")
message("Value of ENV_VAR: $ENV{ENV_VAR}")

和CACHE变量类似,要引用环境变量,格式为:$ENV{}。

2 条件语句

  1. 字符串比较,比如:STREQUAL、STRLESS、STRGREATER等;
  2. 数值比较,比如:EQUAL、LESS、GREATER等;
  3. 布尔运算,AND、OR、NOT;
  4. 路径判断,比如:EXISTS、IS_DIRECTORY、IS_ABSOLUTE等;
  5. 版本号判断;
  6. 使用小括号可以组合多个条件语句,比如:(cond1) AND (cond2 OR (cond3))

常量

  1. ON、YES、TRUE、Y和非0值均被视为True;
  2. 0、OFF、NO、FALSE、N、IGNORE、空字符串、NOTFOUND、及以"-NOTFOUND"结尾的字符串均视为False

对于变量,只要其值不是常量中为False的情形,则均视为True。

3 脚本命令

消息打印

message([<mode>] "message text"...)

mode:

  1. 空或者NOTICE:比较重要的信息,如前面演示中的格式
  2. DEBUG:调试信息,主要针对开发者
  3. STATUS:项目使用者可能比较关心的信息,比如提示当前使用的编译器
  4. WARNING:CMake警告,不会打断进程
  5. SEND_ERROR:CMake错误,会继续执行,但是会跳过生成构建系统
  6. FATAL_ERROR:CMake致命错误,会终止进程

if-else:

set(emp_str "")
if (NOT emp_str AND flag and num less 50 and not not_define_var)message(STATUS "first ")
elseif (emp_str)message(STATUS "second")
else()message("third)
endif()

此外还有 for/while等

list命令

  1. APPEND,往列表中添加元素;
  2. LENGTH,获取列表元素个数;
  3. JOIN,将列表元素用指定的分隔符连接起来;
list(APPEND <var> <val>)

文件操作

CMake的file命令支持的操作比较多,可以读写、创建或复制文件和目录、计算文件hash、下载文件、压缩文件等等。

file(GLOB_RECURSE ALL_SRCsrc/module1/*.csrc/module2/*.c)

GLOB_RECURSE表示执行递归查找,查找目录下所有符合指定正则表达式的文件。

配置文件生成

使用configure_file命令可以将配置文件模板中的特定内容替换,生成目标文件。 输入文件中的内容@VAR@或者${VAR}在输出文件中将被对应的变量值替换。 使用方式为:

PROJECT(tutorial VERSION 1.0.1.1)
configure_file(version.h.in "${PROJECT_BINARY_DIR}/version.h")

version.h.in

#define VERSION "@VERSION@"
#define MAJOR @tutorial_VERSION_MAJOR@
#define MINOR @tutorial_VERSION_MINOR@
#define PATCH @tutorial_VERSION_PATCH@

执行系统命令

使用execute_process命令可以执行一条或者顺序执行多条系统命令,对于需要使用系统命令获取一些变量值是有用的。比如获取当前仓库最新提交的commit的commit id:

execute_process(COMMAND bash "-c" "git rev-parse --short HEAD" OUTPUT_VARIABLE COMMIT_ID)

查找库文件

通过find_library在指定的路径和相关默认路径下查找指定名字的库,常用的格式如下:

find_library (<VAR> name1 [path1 path2 ...])

找到的库就可以被其他target使用,表明依赖关系。

include其他模块

include命令将cmake文件或者模块加载并执行

include(CPack) # 开启打包功能
include(CTest) # 开启测试相关功能

CMake自带有很多有用的模块,可以看看官网的链接:cmake-modules,对支持的功能稍微有所了解,后续有需要再细看文档。当然,如果感兴趣,也可以直接看CMake安装路径下的目录CMake\share\cmake-\Modules中的模块源文件。

关于CMake脚本源文件的示例位于路径:cmake/script_demo.cmake,可以使用cmake -P cmake/script_demo.cmake执行查看结果; 关于配置文件生成的操作在项目根目录的CMakeLists.txt中也有示例。

三、配置案例

main CMakeLists.txt

cmake_minimum_required(VERSION 3.15)
project(tutorial VERSION 2.3)
set(binary_name haha)
add_executable(${binary_name}main.cpp)
message("project_source_dir: ${PROJECT_SOURCE_DIR}, binary: ${PROJECT_BINARY_DIR}")
configure_file(tutorialconfig.h.in tutorialconfig.h)  # comm cmake and src
#########################  change c++ standard  ################################
add_library(tutorial_compiler_flags INTERFACE)
target_compile_features(tutorial_compiler_flags INTERFACE cxx_std_17)
target_link_libraries(${binary_name} PUBLIC tutorial_compiler_flags)
#-------------------------------------------------------------------------------
add_subdirectory(libs/json)  # subproj CMakeLists.txt
target_link_libraries(${binary_name} PUBLIC Json)
target_include_directories(${binary_name} PUBLIC "${PROJECT_BINARY_DIR}""${PROJECT_SOURCE_DIR}""${PROJECT_SOURCE_DIR}/libs/json""${PROJECT_SOURCE_DIR}/libs/math")
target_link_options(${binary_name} PUBLIC "-D_BUG_")
# cmake . -DUSE_MYLIB=OFF
option(USE_MYLIB "Use providesd imt XXX" ON)
if (USE_MYLIB)target_compile_definitions(${binary_name} PRIVATE "USE_MYLIB")message("oo USE_MYLIB macro used")
else()message("oo USE_MYLIB macro not used")
endif()
#########################  Generator Expression  ###############################
set(gcc_like_cxx "$<COMPILE_LANG_AND_ID:CXX,ARMClang,AppleClang,Clang,GNU,LCC>")
set(msvc_cxx "$<COMPILE_LANG_AND_ID:CXX,MSVC>")
target_compile_options(tutorial_compiler_flags INTERFACE# "$<${gcc_like_cxx}:-Wall;-Wextra;-Wshadow;-Wformat=2;-Wunused>"# "$<${msvc_cxx}:-W3>"# 仅在构建时启用,在作为项目的模块时不启用"$<${gcc_like_cxx}:$<BUILD_INTERFACE:-Wall;-Wextra;-Wshadow;-Wformat=2;-Wunused>>""$<${msvc_cxx}:$<BUILD_INTERFACE:-W3>>"
)
###########################  install & ctest ############################
# cmake --install .
# cmake --install . --prefix "/home/myuser/installdir"
install(TARGETS ${binary_name} DESTINATION ${PROJECT_SOURCE_DIR}/install/bin)
install(FILES "${PROJECT_BINARY_DIR}/tutorialconfig.h" DESTINATION ${PROJECT_SOURCE_DIR}/install/include)
# install(EXPORT Json FILE Json.cmake DESTINATION ${PROJECT_SOURCE_DIR}/install/lib)
enable_testing()  # enable testing
add_test(NAME Runs COMMAND ${binary_name} 25)
add_test(NAME Usage COMMAND ${binary_name} 4)
set_tests_properties(Usage PROPERTIES PASS_REGULAR_EXPRESSION "Usage:.*number")
add_test(NAME StandardUse COMMAND ${binary_name} 4)
set_tests_properties(StandardUse PROPERTIES PASS_REGULAR_EXPRESSION "4 is 2")
function(do_test target arg result)add_test(NAME Comp${arg} COMMAND ${target} ${arg})set_tests_properties(Comp${arg} PROPERTIES PASS_REGULAR_EXPRESSION ${result})
endfunction()
do_test(${binary_name} 4 "4 is 2")
do_test(${binary_name} 5 "5 is 2")
do_test(${binary_name} 6 "6 is 2")
do_test(${binary_name} 7 "7 is 2")
do_test(${binary_name} 8 "8 is 2")
do_test(${binary_name} 9 "9 is 2")
#############################  Dashboard  ######################
include(CTest)  # include CTest module
if(UNIX)message(hello------------)
endif()
##########################  system introspection  ##############################
include(CheckCXXSourceCompiles)  # include this module, check_XXX_compiles
check_cxx_source_compiles("#include <cmath>int main() {std::log(1.0);return 0;}"HAVE_LOG
)
check_cxx_source_compiles("#include <cmath>int main() {std::exp(1.0);return 0;}"HAVE_EXP
)
if (HAVE_LOG AND HAVE_EXP)target_compile_definitions(${binary_name} PRIVATE "HAVE_LOG" "HAVE_EXP")message("support exp and log function")
else()message("not support exp or log function")
endif()
#########################  packaging && installer ? failed  #############################
# This module will include any runtime libraries that are needed by the project for the current platform
include(InstallRequiredSystemLibraries)
# set(CPACK_RESOURCE_FILE_LICENSE "${CMAKE_CURRENT_SOURCE_DIR}/License.txt")
set(CPACK_PACKAGE_VERSION_MAJOR "${tutorial_VERSION_MAJOR}")
set(CPACK_PACKAGE_VERSION_MINOR "${tutorial_VERSION_MINOR}")
set(CPACK_SOURCE_GENERATOR "TGZ")
include(CPack)
# then cpack / cpack -G ZIP -C Debug  -G:generator,-C:multi-config specify configuration

sub CMakeLists.txt

include(MakeTable.cmake)  # include and execute
# cmake_minimum_required(VERSION 3.0)
# project(cppjsonrw VERSION 1.9)
add_library(Json json_reader.cc json_value.cc json_writer.cc ${CMAKE_CURRENT_BINARY_DIR}/table.h)
target_include_directories(JsonINTERFACE  # consumers require but the producer doesn't# if not exprot${CMAKE_CURRENT_SOURCE_DIR}  ${CMAKE_CURRENT_BINARY_DIR}# else export# $<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR}># $<INSTALL_INTERFACE:include>
)
target_link_libraries(Json PUBLIC tutorial_compiler_flags)
message("Jproject source dir: ${PROJECT_SOURCE_DIR}")
message("Jcmake source dir: ${CMAKE_SOURCE_DIR}")
message("Jcmake current source dir: ${CMAKE_CURRENT_SOURCE_DIR}")
#######################  static or dynamic lib  ################################
set(CMAKE_ARCHIVE_OUTPUT_DIRECTORY "${PROJECT_BINARY_DIR}")
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY "${PROJECT_BINARY_DIR}")
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY "${PROJECT_BINARY_DIR}")
option(BUILD_SHARED_LIBS "Build using shared libraries" ON)
###########################  install  ############################
set(installable_libs tutorial_compiler_flags Json)
# if (TARGET Json)
#     list(APPEND installable_libs Json)
# endif()
install(TARGETS ${installable_libs} EXPORT Json DESTINATION ${PROJECT_SOURCE_DIR}/install/lib)
install(FILES json.h DESTINATION ${PROJECT_SOURCE_DIR}/install/include)
#######################  custom command and generated file  ####################

四、配置说明

配置项目

# 配置项目
project(tutorial VERSION 1.0.1 LANGUAGES C CXX)

指定编译器语言版本

set(CMAKE_C_STANDARD 99)
set(CMAKE_CXX_STANDARD 17)

配置编译选项

通过命令add_compile_options可为所有编译器配置选项(同时对多个编译器生效);通过设置变量CMAKE_C_FLAGS可以配置C编译器选项;设置变量CMAKE_CXX_FLAGS可配置C++编译器的编译选项。

add_compile_options(-Wall -g)
// add_compile_options("-g;-Wall")
set(CMAKE)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -pipe -std=c99")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -pipe -std=c++17")

debug or release

cmake --build . -DCMAKE_BUILD_TYPE=Debug
set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -g -O0")
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -g -O0")
set(CMAKE_C_FLAGS_RELEASE "${CMAKE_C_FLAGS_RELEASE} -O2")
set(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -O2")

附录

内建变量

PROJECT_SOURCE_DIR最近的CMakeLists.txt中含project命令所在目录
PROJECT_BINARY_DIR最近的含有project的对应构建树中所在目录
CMAKE_SOURCE_DIR顶层CMakeLists.txt所在目录
CMAKE_BINARY_DIR对应构建树顶层目录
CMAKE_CURRENT_SOURCE_DIR当前CMakeLists.txt所在目录
CMAKE_CURRENT_BINARY_DIR对应构建树当前所在目录
<PROJECT-NAME>_VERSION_MAJOR

reference

https://cmake.org/cmake/help/latest/
cmake-commands
cmake-modules(7)-https://cmake.org/cmake/help/latest/manual/cmake-modules.7.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/116289.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

从VTI7064与W25Qxx了解SPI通信协议

在学习过程中记录。 学习背景 最近在做的项目需要设计电路包含外扩FLASH&#xff08;W25Q128&#xff09;与SRAM(VTI7064)&#xff0c;二者都用到了SPI通信协议&#xff0c;之前没学过&#xff0c;学习记录一下。 顺便说一下这次学习中发现的好用工具WPS AI。可以对文档进行…

【STM32】时钟设置函数(寄存器版)

一、STM32时钟设置函数移植 1.时钟模块回顾 一个疑问 前面代码并没有设置时钟为什么可以直接使用。 2.时钟树 3.时钟树分析 1.内部晶振&#xff08;HSI&#xff09; 内部晶振不稳定&#xff0c;当我们上电后&#xff0c;会自动产生振动&#xff0c;自动产生时钟&#xff0c;…

git常用的几条命令介绍

必须了解的命令整理 1&#xff0c;git init 初始化一个新的Git仓库。 这将在当前目录中创建一个名为".git"的子目录&#xff0c;Git会将所有仓库的元数据存储在其中。 2&#xff0c;git clone 克隆一个已存在的仓库。 这会创建一个本地仓库的副本&#xff0c;包…

【java爬虫】使用selenium获取某交易所公司半年报数据

引言 上市公司的财报数据一般都会进行公开&#xff0c;我们可以在某交易所的官方网站上查看这些数据&#xff0c;由于数据很多&#xff0c;如果只是手动收集的话可能会比较耗时耗力&#xff0c;我们可以采用爬虫的方法进行数据的获取。 本文就介绍采用selenium框架进行公司财…

2023年10月24日程序员节

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

SQL server中内连接和外连接的区别、表达(表的连接)

SQL server中内连接与外连接的区别、表达 区别表达内连接外连接 待续 首先&#xff0c;内连接和外连接都是对表的连接操作 区别 内连接&#xff1a;连接结果仅包含符合连接条件的行&#xff0c;其中至少一个属性是共同的&#xff1b;注意区分在嵌套查询时使用的any以及all的区…

基于Tucker分解的时序知识图谱补全10.23

基于Tucker分解的时序知识图谱补全 摘要引言相关工作静态知识图谱补全时序知识图谱补全 背景提出的模型学习时间复杂度和参数增长表达能力分析 实验 摘要 知识图谱已被证明是众多智能应用的有效工具。然而&#xff0c;大量有价值的知识仍然隐含在知识图谱中。为了丰富现有的知…

[python 刷题] 19 Remove Nth Node From End of List

[python 刷题] 19 Remove Nth Node From End of List 题目&#xff1a; Given the head of a linked list, remove the nth node from the end of the list and return its head. 题目说的是就是移除倒数第 n 个结点&#xff0c;如官方给的案例&#xff1a; 这里提供的 n 就是…

实时配送跟踪功能的实现:外卖跑腿小程序的技术挑战

在当今数字化时代&#xff0c;外卖和跑腿服务已经成为了生活中不可或缺的一部分。为了提供更好的用户体验&#xff0c;外卖跑腿小程序越来越注重实时配送跟踪功能的实现。这项技术挑战旨在确保顾客可以方便地跟踪他们的订单&#xff0c;以及配送员可以高效地完成送货任务。本文…

经典卷积神经网络 - VGG

使用块的网络 - VGG。 使用多个 3 3 3\times 3 33的要比使用少个 5 5 5\times 5 55的效果要好。 VGG全称是Visual Geometry Group&#xff0c;因为是由Oxford的Visual Geometry Group提出的。AlexNet问世之后&#xff0c;很多学者通过改进AlexNet的网络结构来提高自己的准确…

一年一度的1024程序员节

前言 1024 程序员节是中国程序员的节日&#xff0c;于每年的 10 月 24 日庆祝。这个节日旨在纪念和表彰程序员对科技和社会发展所做的贡献。 1024 程序员节最早由中国互联网公司 CSDN&#xff08;中国软件开发者网&#xff09;发起&#xff0c;自然而然地成为了中国程序员社区…

element-ui封装loading,便于在拦截请求或其他场景使用

loading.js import { Loading } from element-uilet loadingCount 0 let loadingconst startLoading () > {loading Loading.service({lock: true,text: 加载中,background: rgba(0, 0, 0, 0.7)}) }const endLoading () > {loading.close() }export const showLoadi…

TDengine小知识-数据文件命名规则

TDengine 时序数据库对数据文件有自己的命名规则&#xff0c;文件名中包含了vnodeID、时间范围、版本、文件类型等多种信息。了解数据文件命名规则&#xff0c;可以让运维工作更简单。 废话不多说&#xff0c;直接上图&#xff1a; v4&#xff1a;文件所属 Vgroup 组&#xf…

leetcode:2347. 最好的扑克手牌(python3解法)

难度&#xff1a;简单 给你一个整数数组 ranks 和一个字符数组 suit 。你有 5 张扑克牌&#xff0c;第 i 张牌大小为 ranks[i] &#xff0c;花色为 suits[i] 。 下述是从好到坏你可能持有的 手牌类型 &#xff1a; "Flush"&#xff1a;同花&#xff0c;五张相同花色的…

奶萨看到你那看到教大家

从前阿萨姆调换四大家各不卡速卡大家

华为企业AP开启IPV6包转发

现象&#xff1a; 华为企业AP默认关闭IPV6转发&#xff0c;影响是即便是桥接模式下客户端无法与IPV6网关等设备通信。 web页面无任何相关配置项。 解决&#xff1a; ssh或串口登录&#xff0c;wlan视图下执行sta-ipv6-service enable 开启即可。 <HUAWEI> system-vi…

Flask四种配置方式

Flask是一个轻量级的Python Web框架&#xff0c;被广泛应用于Web开发中。Flask提供了多种配置方式&#xff0c;可以根据不同的需求和场景进行选择。本篇博客将介绍Flask的几种配置方式&#xff0c;并提供相关代码示例。 Flask应用程序的配置对象 在Flask中&#xff0c;应用程序…

安装visual studio报错“无法安装msodbcsql“

在安装visual studio2022时安装完成后提示无法安装msodbcsql, 查看日志文件详细信息提示&#xff1a;指定账户已存在。 未能安装包“msodbcsql,version17.2.30929.1,chipx64,languagezh-CN”。 搜索 URL https://aka.ms/VSSetupErrorReports?qPackageIdmsodbcsql;PackageActi…

用matlab求解线性规划

文章目录 1、用单纯形表求解线性规划绘制单纯形表求解&#xff1a; 2、用matlab求解线性规划——linprog()函数问题&#xff1a;补充代码&#xff1a;显示出完整的影子价格向量 1、用单纯形表求解线性规划 求解线性规划 m i n − 3 x 1 − 4 x 2 x 3 min -3x_1-4x_2x_3 min−…

【ArcGIS模型构建器】04:根据矢量范围批量裁剪影像栅格数据

本文以中国2000-2010-2020年3期GLC30土地覆盖数据为例,演示用模型构建器批量裁剪出四川省3年的数据。 文章目录 一、结果预览二、模型构建三、运行模型四、注意事项一、结果预览 用四川省行政区数据裁剪出的3年Globeland30(配套实验数据data04.rar中有三年中国区域成品数据)…