MCP 客户端
了解如何使用 Model Context Protocol (MCP) 客户端与 MCP 服务器进行交互
Model Context Protocol 客户端
MCP 客户端是 Model Context Protocol (MCP) 架构中的关键组件,负责与 MCP 服务器建立和管理连接。它实现了协议的客户端部分,处理以下内容:
- 协议版本协商以确保与服务器的兼容性
- 能力协商以确定可用功能
- 消息传输和 JSON-RPC 通信
- 工具发现和执行
- 资源访问和管理
- 提示系统交互
- 可选功能如根目录管理和采样支持
核心 io.modelcontextprotocol.sdk:mcp
模块提供了 STDIO 和 SSE 客户端传输实现,无需外部 Web 框架。
对于 Spring Framework 用户,Spring 特定的传输实现作为可选依赖 io.modelcontextprotocol.sdk:mcp-spring-webflux
提供。
客户端为不同应用场景提供了同步和异步 API。
// 使用自定义配置创建同步客户端
McpSyncClient client = McpClient.sync(transport)
.requestTimeout(Duration.ofSeconds(10))
.capabilities(ClientCapabilities.builder()
.roots(true) // 启用根目录功能
.sampling() // 启用采样功能
.build())
.sampling(request -> new CreateMessageResult(response))
.build();
// 初始化连接
client.initialize();
// 列出可用工具
ListToolsResult tools = client.listTools();
// 调用工具
CallToolResult result = client.callTool(
new CallToolRequest("calculator",
Map.of("operation", "add", "a", 2, "b", 3))
);
// 列出和读取资源
ListResourcesResult resources = client.listResources();
ReadResourceResult resource = client.readResource(
new ReadResourceRequest("resource://uri")
);
// 列出和使用提示
ListPromptsResult prompts = client.listPrompts();
GetPromptResult prompt = client.getPrompt(
new GetPromptRequest("greeting", Map.of("name", "Spring"))
);
// 添加/删除根目录
client.addRoot(new Root("file:///path", "description"));
client.removeRoot("file:///path");
// 关闭客户端
client.closeGracefully();
// 使用自定义配置创建同步客户端
McpSyncClient client = McpClient.sync(transport)
.requestTimeout(Duration.ofSeconds(10))
.capabilities(ClientCapabilities.builder()
.roots(true) // 启用根目录功能
.sampling() // 启用采样功能
.build())
.sampling(request -> new CreateMessageResult(response))
.build();
// 初始化连接
client.initialize();
// 列出可用工具
ListToolsResult tools = client.listTools();
// 调用工具
CallToolResult result = client.callTool(
new CallToolRequest("calculator",
Map.of("operation", "add", "a", 2, "b", 3))
);
// 列出和读取资源
ListResourcesResult resources = client.listResources();
ReadResourceResult resource = client.readResource(
new ReadResourceRequest("resource://uri")
);
// 列出和使用提示
ListPromptsResult prompts = client.listPrompts();
GetPromptResult prompt = client.getPrompt(
new GetPromptRequest("greeting", Map.of("name", "Spring"))
);
// 添加/删除根目录
client.addRoot(new Root("file:///path", "description"));
client.removeRoot("file:///path");
// 关闭客户端
client.closeGracefully();
// 使用自定义配置创建异步客户端
McpAsyncClient client = McpClient.async(transport)
.requestTimeout(Duration.ofSeconds(10))
.capabilities(ClientCapabilities.builder()
.roots(true) // 启用根目录功能
.sampling() // 启用采样功能
.build())
.sampling(request -> Mono.just(new CreateMessageResult(response)))
.toolsChangeConsumer(tools -> Mono.fromRunnable(() -> {
logger.info("工具更新: {}", tools);
}))
.resourcesChangeConsumer(resources -> Mono.fromRunnable(() -> {
logger.info("资源更新: {}", resources);
}))
.promptsChangeConsumer(prompts -> Mono.fromRunnable(() -> {
logger.info("提示更新: {}", prompts);
}))
.build();
// 初始化连接并使用功能
client.initialize()
.flatMap(initResult -> client.listTools())
.flatMap(tools -> {
return client.callTool(new CallToolRequest(
"calculator",
Map.of("operation", "add", "a", 2, "b", 3)
));
})
.flatMap(result -> {
return client.listResources()
.flatMap(resources ->
client.readResource(new ReadResourceRequest("resource://uri"))
);
})
.flatMap(resource -> {
return client.listPrompts()
.flatMap(prompts ->
client.getPrompt(new GetPromptRequest(
"greeting",
Map.of("name", "Spring")
))
);
})
.flatMap(prompt -> {
return client.addRoot(new Root("file:///path", "description"))
.then(client.removeRoot("file:///path"));
})
.doFinally(signalType -> {
client.closeGracefully().subscribe();
})
.subscribe();
客户端传输
传输层处理 MCP 客户端和服务器之间的通信,为不同用例提供不同的实现。客户端传输管理消息序列化、连接建立和特定于协议的通信模式。
为基于进程内的通信创建传输
ServerParameters params = ServerParameters.builder("npx")
.args("-y", "@modelcontextprotocol/server-everything", "dir")
.build();
McpTransport transport = new StdioClientTransport(params);
为基于进程内的通信创建传输
ServerParameters params = ServerParameters.builder("npx")
.args("-y", "@modelcontextprotocol/server-everything", "dir")
.build();
McpTransport transport = new StdioClientTransport(params);
创建一个框架无关的(纯 Java API)SSE 客户端传输。包含在核心 mcp 模块中。
McpTransport transport = new HttpClientSseClientTransport("http://your-mcp-server");
创建基于 WebFlux 的 SSE 客户端传输。需要 mcp-webflux-sse-transport 依赖。
WebClient.Builder webClientBuilder = WebClient.builder()
.baseUrl("http://your-mcp-server");
McpTransport transport = new WebFluxSseClientTransport(webClientBuilder);
客户端能力
客户端可以配置各种能力:
var capabilities = ClientCapabilities.builder()
.roots(true) // 启用文件系统根目录支持,并提供列表变更通知
.sampling() // 启用 LLM 采样支持
.build();
根目录支持
根目录定义了服务器在文件系统中可以操作的边界:
// 动态添加根目录
client.addRoot(new Root("file:///path", "description"));
// 删除根目录
client.removeRoot("file:///path");
// 通知服务器根目录列表已更改
client.rootsListChangedNotification();
根目录功能允许服务器:
- 请求可访问的文件系统根目录列表
- 在根目录列表更改时接收通知
- 了解它们可以访问哪些目录和文件
采样支持
采样使服务器可以通过客户端请求 LLM 交互(“完成”或“生成”):
// 配置采样处理程序
Function<CreateMessageRequest, CreateMessageResult> samplingHandler = request -> {
// 与 LLM 接口的采样实现
return new CreateMessageResult(response);
};
// 创建支持采样的客户端
var client = McpClient.sync(transport)
.capabilities(ClientCapabilities.builder()
.sampling()
.build())
.sampling(samplingHandler)
.build();
此功能允许:
- 服务器利用 AI 功能而无需 API 密钥
- 客户端保持对模型访问和权限的控制
- 支持基于文本和图像的交互
- 可选地在提示中包含 MCP 服务器上下文
使用 MCP 客户端
工具执行
工具是客户端可以发现和执行的服务器端函数。MCP 客户端提供了列出可用工具和使用特定参数执行它们的方。每种工具都有一个唯一的名称,并接受参数映射。
// 列出可用工具及其名称
var tools = client.listTools();
tools.forEach(tool -> System.out.println(tool.getName()));
// 使用参数执行工具
var result = client.callTool("calculator", Map.of(
"operation", "add",
"a", 1,
"b", 2
));
// 列出可用工具及其名称
var tools = client.listTools();
tools.forEach(tool -> System.out.println(tool.getName()));
// 使用参数执行工具
var result = client.callTool("calculator", Map.of(
"operation", "add",
"a", 1,
"b", 2
));
// 异步列出可用工具
client.listTools()
.doOnNext(tools -> tools.forEach(tool ->
System.out.println(tool.getName())))
.subscribe();
// 异步执行工具
client.callTool("calculator", Map.of(
"operation", "add",
"a", 1,
"b", 2
))
.subscribe();
资源访问
资源代表客户端可以使用 URI 模板访问的服务器端数据源。MCP 客户端提供了发现可用资源和通过标准化接口检索其内容的方法。
// 列出可用资源及其名称
var resources = client.listResources();
resources.forEach(resource -> System.out.println(resource.getName()));
// 使用 URI 模板检索资源内容
var content = client.getResource("file", Map.of(
"path", "/path/to/file.txt"
));
// 列出可用资源及其名称
var resources = client.listResources();
resources.forEach(resource -> System.out.println(resource.getName()));
// 使用 URI 模板检索资源内容
var content = client.getResource("file", Map.of(
"path", "/path/to/file.txt"
));
// 异步列出可用资源
client.listResources()
.doOnNext(resources -> resources.forEach(resource ->
System.out.println(resource.getName())))
.subscribe();
// 异步检索资源内容
client.getResource("file", Map.of(
"path", "/path/to/file.txt"
))
.subscribe();
提示系统
提示系统允许与服务器端提示模板进行交互。这些模板可以被发现并使用自定义参数执行,允许基于预定义模式进行动态文本生成。
// 列出可用提示模板
var prompts = client.listPrompts();
prompts.forEach(prompt -> System.out.println(prompt.getName()));
// 使用参数执行提示模板
var response = client.executePrompt("echo", Map.of(
"text", "Hello, World!"
));
// 列出可用提示模板
var prompts = client.listPrompts();
prompts.forEach(prompt -> System.out.println(prompt.getName()));
// 使用参数执行提示模板
var response = client.executePrompt("echo", Map.of(
"text", "Hello, World!"
));
// 异步列出可用提示模板
client.listPrompts()
.doOnNext(prompts -> prompts.forEach(prompt ->
System.out.println(prompt.getName())))
.subscribe();
// 异步执行提示模板
client.executePrompt("echo", Map.of(
"text", "Hello, World!"
))
.subscribe();