目录

Visual Studio 使用 clang-format 自定义 C++ 代码默认格式化样式

问题描述

Visual Studio 的 C++ 代码格式化可选使用 clang-format,但它只提供默认样式,如果想使用自定义样式则需要在每个项目目录下放一个 .clang-format_clang-format 文件,没有对全部项目通用的可自定义样式。

解决方法

这里发现 Visual Studio 的格式设置中有一个使用自定义 clang-format.exe 文件选项,所以这里尝试制做一个 clang-format 的包装程序让 VS 使用(来骗)。

包装程序通过其同级目录下的 rules.txt 文件来设置自定义的默认样式,具体设置参照 Clang-Format 官方文档,也可以使用 Clang-Format 交互式构建器。启用这个包装程序后, Visual Studio 内置的默认格式设置样式选项将会失效。

  • 预编译的包装程序和 clang-format.exe 一起打包贴在了文章末尾,可以直接下载使用。
  • 包装程序的代码贴在了文章末尾,如有需要在 Visual Studio 中使用 Std C++17 编译即可。
  • clang-format.exe 可以在 %VS安装目录%\VC\Tools\Llvm\x64\bin 目录中找到,也可以在 LLVM 发布页 下载。

这里贴上作者习惯使用的代码格式化样式(Java 代码的习惯样式),以下是 rules.txt 内容:

 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
# 基于GNU格式
BasedOnStyle: GNU
# 使用支持的最新标准
Standard: Latest
# 行不限制长度
ColumnLimit: 0
# 缩进4个空格
IndentWidth: 4
# 左花括号在名称之后
BreakBeforeBraces: Attach
# 函数参数的小括号与函数名间不空格, 控制语句的小括号与语句间空格
SpaceBeforeParens: ControlStatements
# 换行时将二元运算符置后
BreakBeforeBinaryOperators: None
# 允许空的花括号在一行上
AllowShortBlocksOnASingleLine: Empty
# 函数返回值与函数在同一行
AlwaysBreakAfterDefinitionReturnType: None
AlwaysBreakAfterReturnType: None
# 指针符号贴在类型边
PointerAlignment: Left
# 访问修饰符贴在左边
AccessModifierOffset: -4
# case和default之前缩进
IndentCaseLabels: true
# C式类型转换中间留空格
SpaceAfterCStyleCast: true
# 类继承冒号之前不留空格
SpaceBeforeInheritanceColon: false
# 构造函数初始化冒号之前不留空格
SpaceBeforeCtorInitializerColon: false
# 命名空间全部缩进
NamespaceIndentation: All
# 不排序头文件引用
SortIncludes: Never
# C++11式初始化对象花括号前留空格
SpaceBeforeCpp11BracedList: true

使用之前(二选一)

  1. 确保 clang-format.exePATH 中并且可访问。

    /2024-08-23-visual-studio-%E4%BD%BF%E7%94%A8-clang-format-%E8%87%AA%E5%AE%9A%E4%B9%89-c-%E4%BB%A3%E7%A0%81%E9%BB%98%E8%AE%A4%E6%A0%BC%E5%BC%8F%E5%8C%96%E6%A0%B7%E5%BC%8F/assets/Snipaste_2024-08-21_21-06-49.png
    PATH 中可访问的 clang-format
  2. 确保 clang-format.exe 位于 clang-format-wrapper.exe 的同级目录下。

    /2024-08-23-visual-studio-%E4%BD%BF%E7%94%A8-clang-format-%E8%87%AA%E5%AE%9A%E4%B9%89-c-%E4%BB%A3%E7%A0%81%E9%BB%98%E8%AE%A4%E6%A0%BC%E5%BC%8F%E5%8C%96%E6%A0%B7%E5%BC%8F/assets/Snipaste_2024-08-21_21-09-39.png
    同级目录下的 clang-format

使用方法

  1. 启用 ClangFormat 支持使用自定义 clang-format.exe 文件,选择上一步的 clang-format-wrapper.exe 并确定。

    /2024-08-23-visual-studio-%E4%BD%BF%E7%94%A8-clang-format-%E8%87%AA%E5%AE%9A%E4%B9%89-c-%E4%BB%A3%E7%A0%81%E9%BB%98%E8%AE%A4%E6%A0%BC%E5%BC%8F%E5%8C%96%E6%A0%B7%E5%BC%8F/assets/Snipaste_2024-08-21_21-03-09.png
    配置选项
  2. 现在可以回到编辑器测试一下了,默认格式化快捷键是 Ctrl + K + D

包装程序的代码

 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
#include <filesystem>
#include <optional>
#include <string>

#include <process.h>

namespace fs = std::filesystem;

const wchar_t* clang_format_path = L"clang-format.exe";
const wchar_t* rule_text_path = L"rules.txt";

int wmain(int argc, wchar_t** argv) {
    std::optional<std::wstring> args;
    fs::path path = fs::absolute(argv[0]);
    path.replace_filename(clang_format_path);
    if (fs::exists(path)) {
        args = path.wstring();
    } else {
        args = std::wstring(clang_format_path, 16);
    }
    args->push_back(L' ');
    for (int i = 1; i < argc; i++) {
        if (!wcsstr(argv[i], L"-style=")) {
            args->append(argv[i]);
            args->push_back(L' ');
        }
    }
    args->append(L"-fallback-style=none -style=file:");
    args->append(path.replace_filename(rule_text_path));
    return _wsystem(args->data());
}

资源下载

由于 clang-format.execlang-format-wrapper.exe 依赖 VC++ 140 运行库,如果无法运行请先安装运行库。