C/C++ 和 Cmake 格式化 ArchLinux 系统级安装 运行以下命令安装 clang
包,其中包含了 clang-format
:
之后你就可以在全局执行 clang-format
命令来格式化代码。可执行文件一般位于 /usr/bin/clang-format
。但是这里安装的而二进制程序使用的是动态库,你把二进制程序分发到其他没有安装 clang 动态库的计算机上,就会报找不到动态库的错误。
如果你有系统权限,或者是在你自己的计算机上开发,这样就可以了即使是把可执行程序拷贝的项目目录下,执行项目目录下的可执行程序,也可以正常运行(动态库还是使用系统目录下的)。
如果你想使用单独的使用静态库的可执行程序,需要手动构建。
手动用户级安装 LLVM 官方提供了单独的 clang-format
二进制文件,可以直接下载并解压到项目目录中。
这个包里面的 clang-format 不知道是如何构建的,笔者没有尝试。
下载 LLVM 源码 从 LLVM GitHub 下载源码,或者直接克隆:
1 2 git clone https://github.com/llvm/llvm-project.git cd llvm-project
配置并仅构建 clang-format
创建构建目录并配置:
1 2 mkdir build && cd buildcmake -G Ninja -DCMAKE_BUILD_TYPE=Release -DLLVM_ENABLE_PROJECTS="clang" -DBUILD_SHARED_LIBS=OFF -DLLVM_BUILD_TOOLS=OFF -DCLANG_BUILD_TOOLS=OFF -DCLANG_ENABLE_FORMAT=ON ../llvm
构建 clang-format :
提取 clang-format
二进制文件 构建完成后,clang-format
二进制文件位于 build/bin/clang-format
。可以将其复制到项目目录中,可以直接使用该可执行程序格式化。
注:即便使用了静态库,但是你在你的机器上构建的 clang-format
依赖于你的版本的 GLIBC 和 libstdc++,而其他的系统上安装的 GLIBC 和 libstdc++ 版本可能较旧,无法满足要求。你可以升级 GLIBC 和 libstdc++ 或者在使用旧 GLIBC 和 libstdc++ 的版本系统上构建。
基本使用 将可执行文件 clang-format
复制到项目目录下后,你可以运行
1 clang-format -i your_code.cpp
来格式化代码。
vscode 扩展 可以在 vscode 中安装 clang-format 扩展来使用在 vscode 中集成的格式化操作。
安装后,可以进行以下配置:
Clang-Format: Path 指定了正确的 clang-format 可执行文件路径(如果 VS Code 没有自动找到它的话)。
Editor: Format On Save:在每次保存文件时,VS Code 会自动运行 clang-format 格式化代码。
Clang-Format: Style:选择格式化风格(例如 Google、LLVM、Chromium 等)。如果你有 .clang-format
文件,它会覆盖此设置。
自定义代码格式化规则 创建配置文件 .clang-format
文件是用来指定代码格式化规则的配置文件。你可以在项目根目录或用户级别目录下创建一个 .clang-format
文件。
你可以手动创建 .clang-format
文件然后自行配置格式化规则。
不过推荐使用以下命令生成默认配置文件,然后在其基础上进行修改:
1 clang-format -style=llvm -dump-config > .clang-format
这会创建一个基于 llvm 风格的默认配置文件。
你也可以选择其他风格,如 LLVM、Microsoft、Google、Chromium 等,例如:
1 clang-format -style=llvm -dump-config > .clang-format
.clang-format
配置文件例子(中文注释):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 --- Language: Cpp AccessModifierOffset: -2 AlignAfterOpenBracket: Align AlignArrayOfStructures: None AlignConsecutiveAssignments: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: true AlignConsecutiveBitFields: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: false AlignConsecutiveDeclarations: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: false AlignConsecutiveMacros: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: false AlignConsecutiveShortCaseStatements: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCaseArrows: false AlignCaseColons: false AlignConsecutiveTableGenBreakingDAGArgColons: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: false AlignConsecutiveTableGenCondOperatorColons: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: false AlignConsecutiveTableGenDefinitionColons: Enabled: false AcrossEmptyLines: false AcrossComments: false AlignCompound: false AlignFunctionPointers: false PadOperators: false AlignEscapedNewlines: Right AlignOperands: Align AlignTrailingComments: Kind: Always OverEmptyLines: 0 AllowAllArgumentsOnNextLine: true AllowAllParametersOfDeclarationOnNextLine: true AllowBreakBeforeNoexceptSpecifier: Never AllowShortBlocksOnASingleLine: Never AllowShortCaseExpressionOnASingleLine: true AllowShortCaseLabelsOnASingleLine: false AllowShortCompoundRequirementOnASingleLine: true AllowShortEnumsOnASingleLine: true AllowShortFunctionsOnASingleLine: All AllowShortIfStatementsOnASingleLine: Never AllowShortLambdasOnASingleLine: All AllowShortLoopsOnASingleLine: false AlwaysBreakAfterDefinitionReturnType: None AlwaysBreakBeforeMultilineStrings: false AttributeMacros: • __capability BinPackArguments: true BinPackParameters: true BitFieldColonSpacing: Both BraceWrapping: AfterCaseLabel: false AfterClass: false AfterControlStatement: Never AfterEnum: false AfterExternBlock: false AfterFunction: false AfterNamespace: false AfterObjCDeclaration: false AfterStruct: false AfterUnion: false BeforeCatch: false BeforeElse: false BeforeLambdaBody: false BeforeWhile: false IndentBraces: false SplitEmptyFunction: true SplitEmptyRecord: true SplitEmptyNamespace: true BreakAdjacentStringLiterals: true BreakAfterAttributes: Leave BreakAfterJavaFieldAnnotations: false BreakAfterReturnType: None BreakArrays: true BreakBeforeBinaryOperators: None BreakBeforeConceptDeclarations: Always BreakBeforeBraces: Attach BreakBeforeInlineASMColon: OnlyMultiline BreakBeforeTernaryOperators: true BreakConstructorInitializers: BeforeColon BreakFunctionDefinitionParameters: false BreakInheritanceList: BeforeColon BreakStringLiterals: true BreakTemplateDeclarations: MultiLine ColumnLimit: 120 CommentPragmas: '^ IWYU pragma:' CompactNamespaces: false ConstructorInitializerIndentWidth: 4 ContinuationIndentWidth: 4 Cpp11BracedListStyle: true DerivePointerAlignment: false DisableFormat: false EmptyLineAfterAccessModifier: Never EmptyLineBeforeAccessModifier: LogicalBlock ExperimentalAutoDetectBinPacking: false FixNamespaceComments: true ForEachMacros: • foreach • Q_FOREACH • BOOST_FOREACH IfMacros: • KJ_IF_MAYBE IncludeBlocks: Preserve IncludeCategories: • Regex: '^"(llvm|llvm-c|clang|clang-c)/' Priority: 2 SortPriority: 0 CaseSensitive: false • Regex: '^(<|"(gtest|gmock|isl|json)/)' Priority: 3 SortPriority: 0 CaseSensitive: false • Regex: '.*' Priority: 1 SortPriority: 0 CaseSensitive: false IncludeIsMainRegex: '(Test)?$' IncludeIsMainSourceRegex: '' IndentAccessModifiers: false IndentCaseBlocks: false IndentCaseLabels: false IndentExternBlock: AfterExternBlock IndentGotoLabels: true IndentPPDirectives: None IndentRequiresClause: true IndentWidth: 2 IndentWrappedFunctionNames: false InsertBraces: false InsertNewlineAtEOF: false InsertTrailingCommas: None IntegerLiteralSeparator: Binary: 0 BinaryMinDigits: 0 Decimal: 0 DecimalMinDigits: 0 Hex: 0 HexMinDigits: 0 JavaScriptQuotes: Leave JavaScriptWrapImports: true KeepEmptyLines: AtEndOfFile: false AtStartOfBlock: true AtStartOfFile: true LambdaBodyIndentation: Signature LineEnding: DeriveLF MacroBlockBegin: '' MacroBlockEnd: '' MainIncludeChar: Quote MaxEmptyLinesToKeep: 1 NamespaceIndentation: None ObjCBinPackProtocolList: Auto ObjCBlockIndentWidth: 2 ObjCBreakBeforeNestedBlockParam: true ObjCSpaceAfterProperty: false ObjCSpaceBeforeProtocolList: true PackConstructorInitializers: BinPack PenaltyBreakAssignment: 2 PenaltyBreakBeforeFirstCallParameter: 19 PenaltyBreakComment: 300 PenaltyBreakFirstLessLess: 120 PenaltyBreakOpenParenthesis: 0 PenaltyBreakScopeResolution: 500 PenaltyBreakString: 1000 PenaltyBreakTemplateDeclaration: 10 PenaltyExcessCharacter: 1000000 PenaltyIndentedWhitespace: 0 PenaltyReturnTypeOnItsOwnLine: 60 PointerAlignment: Right PPIndentWidth: -1 QualifierAlignment: Leave ReferenceAlignment: Pointer ReflowComments: true RemoveBracesLLVM: false RemoveParentheses: Leave RemoveSemicolon: false RequiresClausePosition: OwnLine RequiresExpressionIndentation: OuterScope SeparateDefinitionBlocks: Leave ShortNamespaceLines: 1 SkipMacroDefinitionBody: false SortIncludes: CaseSensitive SortJavaStaticImport: Before SortUsingDeclarations: LexicographicNumeric SpaceAfterCStyleCast: false SpaceAfterLogicalNot: false SpaceAfterTemplateKeyword: true SpaceAroundPointerQualifiers: Default SpaceBeforeAssignmentOperators: true SpaceBeforeCaseColon: false SpaceBeforeCpp11BracedList: false SpaceBeforeCtorInitializerColon: true SpaceBeforeInheritanceColon: true SpaceBeforeJsonColon: false SpaceBeforeParens: ControlStatements SpaceBeforeParensOptions: AfterControlStatements: true AfterForeachMacros: true AfterFunctionDefinitionName: false AfterFunctionDeclarationName: false AfterIfMacros: true AfterOverloadedOperator: false AfterPlacementOperator: true AfterRequiresInClause: false AfterRequiresInExpression: false BeforeNonEmptyParentheses: false SpaceBeforeRangeBasedForLoopColon: true SpaceBeforeSquareBrackets: false SpaceInEmptyBlock: false SpacesBeforeTrailingComments: 1 SpacesInAngles: Never SpacesInContainerLiterals: true SpacesInLineCommentPrefix: Minimum: 1 Maximum: -1 SpacesInParens: Never SpacesInParensOptions: ExceptDoubleParentheses: false InCStyleCasts: false InConditionalStatements: false InEmptyParentheses: false Other: false SpacesInSquareBrackets: false Standard: Latest StatementAttributeLikeMacros: • Q_EMIT StatementMacros: • Q_UNUSED • QT_REQUIRE_VERSION TableGenBreakInsideDAGArg: DontBreak TabWidth: 8 UseTab: Never VerilogBreakBetweenInstancePorts: true WhitespaceSensitiveMacros: • BOOST_PP_STRINGIZE • CF_SWIFT_NAME • NS_SWIFT_NAME • PP_STRINGIZE • STRINGIZE ...
应用配置文件 未指定时自动查找 clang-format
在格式化代码时,会自动查找并使用 .clang-format
文件中的规则。查找规则如下:
当前工作目录 :clang-format
会首先检查当前工作目录下是否存在 .clang-format
文件。
父目录 :如果当前目录没有找到 .clang-format
文件,clang-format
会递归地向父目录查找,直到找到 .clang-format
文件或到达文件系统的根目录。
用户主目录 :如果以上目录都没有找到 .clang-format
文件,clang-format
会检查用户的主目录(~/.clang-format
)。
内置默认规则 :如果以上位置都没有找到 .clang-format
文件,clang-format
会使用其内置的默认格式化规则。
使用指定配置文件 可以通过以下方式明确指定使用某个 .clang-format
文件:
使用 -style=file
参数,并确保 .clang-format
文件位于当前目录或其父目录中。即 yourfile.cpp
要位于 .clang-format
文件所在目录下或者其子目录下。例如:
1 clang-format -style=file -i yourfile.cpp
你也可以使用下面的命令还使用默认规则,若不指定 -style
参数,则默认是使用 LLVM 规则:
1 clang-format -style=LLVM -i yourfile.cpp
-style
可选 LLVM、Microsoft、Google、Chromium 等。
这会强制 clang-format
使用 .clang-format
文件中的规则。
如果你想直接指定某个 .clang-format
文件的路径,可以使用 -style
参数并传递文件内容。例如:
1 clang-format -style="$(cat /path/to/.clang-format) " -i yourfile.cpp
格式化脚本 当项目内代码文件过多时,一个个格式化代码太过繁琐了,我们可以写一个简单的脚本来帮我们遍历处理项目内的代码文件。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 #!/bin/zsh GREEN='\033[0;32m' BLUE='\033[0;34m' NC='\033[0m' SCRIPT_DIR=$(dirname "$0 " ) NEED_FORMAT_FILE="$SCRIPT_DIR /clang-format-files.txt" function get_cpp_files_in_dir { find "$1 " -maxdepth 1 \( -name "*.cpp" -o -name "*.hpp" -o -name "*.h" -o -name "*.cc" \) -type f 2>/dev/null } function get_cpp_files_recursive { find "$1 " \( -name "*.cpp" -o -name "*.hpp" -o -name "*.h" -o -name "*.cc" \) -type f 2>/dev/null } function ask_user { local files=("${(@f)1} " ) local total_lines=0 local lines=0 echo "匹配到以下 ${#files[@]} 个文件:" for file in "${files[@]} " ; do lines=$(wc -l < "$file " 2>/dev/null | tr -d ' ' ) total_lines=$((total_lines + lines)) printf " ${GREEN} %s${NC} \t(${BLUE} %d 行${NC} )\n" "$file " "$lines " done echo "总计:$total_lines 行" echo -n "是否格式化这些文件? (y/n, 默认 y): " read -r answer if [[ -z "$answer " || "$answer " == "y" ]]; then local count=0 for file in "${files[@]} " ; do clang-format -i "$file " count=$((count + 1 )) printf "\r已按顺序格式化 %d 个文件。" "$count " done printf "\n" echo "格式化完成。" else echo "跳过格式化。" fi } if [ -f "$NEED_FORMAT_FILE " ]; then matched_files=() ignored_files=() while IFS= read -r line || [[ -n "$line " ]]; do line=$(echo "$line " | sed -e 's/^[ \t]*//' -e 's/[ \t]*$//' ) if [[ "$line " == +* ]]; then pattern="${line:1} " matched_files+=($(eval "ls -d $SCRIPT_DIR /$pattern " 2>/dev/null | grep -E '\.(cpp|hpp|h|cc)$' )) elif [[ "$line " == -* ]]; then pattern="${line:1} " ignored_files+=($(eval "ls -d $SCRIPT_DIR /$pattern " 2>/dev/null | grep -E '\.(cpp|hpp|h|cc)$' )) fi done < "$NEED_FORMAT_FILE " if [ ${#matched_files[@]} -eq 0 ]; then echo "没有匹配到 clang-format-files.txt 中需要格式化的文件" else echo "已匹配到 clang-format-files.txt 中需要格式化的文件" ask_user "${(F)matched_files} " fi if [ ${#ignored_files[@]} -ne 0 ]; then echo "以下文件被忽略,不会格式化:" for file in "${ignored_files[@]} " ; do printf " ${GREEN} %s${NC} \n" "$file " done fi exit 0 fi current_dir_files=($(get_cpp_files_in_dir "$SCRIPT_DIR " )) if [ ${#current_dir_files[@]} -eq 0 ]; then echo "工作目录内没有 C/C++ 文件。" else echo "当前目录的 C/C++ 文件:" ask_user "${(F)current_dir_files} " fi for dir in "$SCRIPT_DIR " /*(/); do dir_files=($(get_cpp_files_recursive "$dir " )) if [ ${#dir_files[@]} -eq 0 ]; then continue else ask_user "${(F)dir_files} " fi done
脚本执行流程:
当存在配置文件 clang-format-files.txt
时:
解析配置文件的 +
(包含)和 -
(排除)规则
合并所有匹配文件并过滤排除项
单次交互确认 :列出所有目标文件及行数,确认后一次性完成格式化
当不存在配置文件时:
当前目录处理(非递归):
扫描当前层级的 C/C++ 文件
首次交互确认 :用户确认是否格式化这些文件
子目录递归处理:
遍历当前目录的每个子目录
对每个子目录执行递归扫描(含所有嵌套子目录)
逐目录交互确认 :每个子目录的文件列表单独显示并请求确认
clang-format-files.txt
文件说明:
文件路径前的 +
表示需要格式化后面的文件,执行脚本时会进行询问, -
表示不需要询问格式化后面的文件,执行脚本时会忽略。
path/to/directory/**/*
:匹配 path/to/directory
目录及其所有子目录中的所有 C/C++ 相关文件(需要支持 **
的 shell,如 zsh
)。
Arch Linux 系统安装 1 sudo pacman -S cmake-format
cmake-format
是一个 Python 工具,通常以 Python 包的形式分发。要将其打包为独立的二进制文件,可以使用工具如 pyinstaller
。
安装依赖 :
1 2 sudo pacman -S python python-pippip install cmake-format pyinstaller
将 cmake-format
打包为独立二进制 :
1 pyinstaller --onefile --collect-all cmakelang $(which cmake-format)
获取二进制文件 : 打包完成后,独立的 cmake-format
可执行文件位于:
你可以将其复制到项目目录中使用。
注意:pyinstaller
打包时会将动态库(如 libz.so.1
)打包到可执行文件中,但这些动态库可能依赖于较新的 GLIBC 特性(如 GLIBC_ABI_DT_RELR
)。如果使用可执行程序的系统 GLIBC 版本较旧,就会导致运行失败。
生成自定义格式化规则配置文件:
1 cmake-format --dump-config > .cmake-format
.cmake-format
配置文件内容例子(中文注释):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 with section("parse"): additional_commands = { 'foo': { 'flags': ['BAR' , 'BAZ' ], 'kwargs': {'DEPENDS': '*' , 'HEADERS': '*' , 'SOURCES': '*' }}} override_spec = {} vartags = [] proptags = [] with section("format"): disable = False line_width = 120 tab_size = 4 use_tabchars = False fractional_tab_policy = 'use-space' max_subgroups_hwrap = 2 max_pargs_hwrap = 6 max_rows_cmdline = 2 separate_ctrl_name_with_space = False separate_fn_name_with_space = False dangle_parens = False dangle_align = 'prefix' min_prefix_chars = 4 max_prefix_chars = 10 max_lines_hwrap = 2 line_ending = 'unix' command_case = 'canonical' keyword_case = 'unchanged' always_wrap = [] enable_sort = True autosort = False require_valid_layout = False layout_passes = {} with section("markup"): bullet_char = '*' enum_char = '.' first_comment_is_literal = False literal_comment_pattern = None fence_pattern = '^\\s*([`~]{3}[`~]*)(.*)$' ruler_pattern = '^\\s*[^\\w\\s]{3}.*[^\\w\\s]{3}$' explicit_trailing_pattern = '#<' hashruler_min_length = 10 canonicalize_hashrulers = True enable_markup = True with section("lint"): disabled_codes = [] function_pattern = '[0-9a-z_]+' macro_pattern = '[0-9A-Z_]+' global_var_pattern = '[A-Z][0-9A-Z_]+' internal_var_pattern = '_[A-Z][0-9A-Z_]+' local_var_pattern = '[a-z][a-z0-9_]+' private_var_pattern = '_[0-9a-z_]+' public_var_pattern = '[A-Z][0-9A-Z_]+' argument_var_pattern = '[a-z][a-z0-9_]+' keyword_pattern = '[A-Z][0-9A-Z_]+' max_conditionals_custom_parser = 2 min_statement_spacing = 1 max_statement_spacing = 2 max_returns = 6 max_branches = 12 max_arguments = 5 max_localvars = 15 max_statements = 50 with section("encode"): emit_byteorder_mark = False input_encoding = 'utf-8' output_encoding = 'utf-8' with section("misc"): per_command = {}