使用C#创建一个MCP客户端

前言

网上使用Python创建一个MCP客户端的教程已经有很多了,而使用C#创建一个MCP客户端的教程还很少。

为什么要创建一个MCP客户端呢?

创建了一个MCP客户端之后,你就可以使用别人写好的一些MCP服务了。

效果展示

为了方便大家复现,我没有使用WPF/Avalonia之类的做界面。只是一个简单的控制台程序,可以很容易看懂。

image-20250314173410130

接入了fetch_mcp可以实现获取网页内容了,使用的模型只要具有tool use能力的应该都可以。

我使用的是Qwen/Qwen2.5-72B-Instruct。

开始实践

主要使用的包如下所示:

image-20250314173634157

首先获取MCP服务器:

 private static async Task<IMcpClient> GetMcpClientAsync()
 {
     DotEnv.Load();
     var envVars = DotEnv.Read();
     McpClientOptions options = new()
     {
         ClientInfo = new() { Name = "SimpleToolsConsole", Version = "1.0.0" }
     };

     var config = new McpServerConfig
     {
         Id = "test",
         Name = "Test",
         TransportType = TransportTypes.StdIo,
         TransportOptions = new Dictionary<string, string>
         {
             ["command"] = envVars["MCPCommand"],
             ["arguments"] = envVars["MCPArguments"],
         }
     };

     var factory = new McpClientFactory(
         new[] { config },
         options,
         NullLoggerFactory.Instance
     );

     return await factory.GetClientAsync("test");
 }

写死的话就是这样写:

 private static async Task<IMcpClient> GetMcpClientAsync()
 {
     DotEnv.Load();
     var envVars = DotEnv.Read();
     McpClientOptions options = new()
     {
         ClientInfo = new() { Name = "SimpleToolsConsole", Version = "1.0.0" }
     };

     var config = new McpServerConfig
     {
         Id = "test",
         Name = "Test",
         TransportType = TransportTypes.StdIo,
         TransportOptions = new Dictionary<string, string>
         {
             ["command"] = node,
             ["arguments"] = D:/Learning/AI-related/fetch-mcp/dist/index.js,
         }
     };

     var factory = new McpClientFactory(
         new[] { config },
         options,
         NullLoggerFactory.Instance
     );

     return await factory.GetClientAsync("test");
 }

重点在:

 TransportOptions = new Dictionary<string, string>
         {
             ["command"] = node,
             ["arguments"] = D:/Learning/AI-related/fetch-mcp/dist/index.js,
         }

用于连接你想连接的MCP服务器。

如果能正确显示你连接mcp服务器提供的工具,说明连接成功。

  var listToolsResult = await client.ListToolsAsync();
  var mappedTools = listToolsResult.Tools.Select(t => t.ToAITool(client)).ToList();
  Console.WriteLine("Tools available:");
  foreach (var tool in mappedTools)
  {
      Console.WriteLine("  " + tool);
  }

image-20250314174210161

开启一个聊天循环:

    Console.WriteLine("\nMCP Client Started!");
    Console.WriteLine("Type your queries or 'quit' to exit.");

    ChatDemo chatDemo = new ChatDemo();

    while (true)
    {
        try
        {
            Console.ForegroundColor = ConsoleColor.DarkYellow;
            Console.Write("\nQuery: ");
            string query = Console.ReadLine()?.Trim() ?? string.Empty;

            if (query.ToLower() == "quit")
                break;
            if (query.ToLower() == "clear")
            {
                Console.Clear();
                chatDemo.Messages.Clear();                    
            }
            else 
            {
                string response = await chatDemo.ProcessQueryAsync(query, mappedTools);
                Console.ForegroundColor = ConsoleColor.DarkYellow;
                Console.WriteLine($"AI回答:{response}");
                Console.ForegroundColor = ConsoleColor.White;
            }                      
        }
        catch (Exception ex)
        {
            Console.WriteLine($"\nError: {ex.Message}");
        }
    }
}

处理每次询问:

 public async Task<string> ProcessQueryAsync(string query, List<AITool> tools)
 {
     if(Messages.Count == 0)
     {
         Messages =
         [
          // Add a system message
         new(ChatRole.System, "You are a helpful assistant, helping us test MCP server functionality."),
         ];
     }
     
     // Add a user message
     Messages.Add(new(ChatRole.User, query));

     var response = await ChatClient.GetResponseAsync(
            Messages,
            new() { Tools = tools });
     Messages.AddMessages(response);
     var toolUseMessage = response.Messages.Where(m => m.Role == ChatRole.Tool);

     if (toolUseMessage.Count() > 0)
     {
         var functionMessage = response.Messages.Where(m => m.Text == "").First();             
         var functionCall = (FunctionCallContent)functionMessage.Contents[1];
         Console.ForegroundColor = ConsoleColor.Green;
         string arguments = "";
         foreach (var arg in functionCall.Arguments)
         {
             arguments += $"{arg.Key}:{arg.Value};";
         }
         Console.WriteLine($"调用函数名:{functionCall.Name};参数信息:{arguments}");
         foreach (var message in toolUseMessage)
         {
             var functionResultContent = (FunctionResultContent)message.Contents[0];
             Console.WriteLine($"调用工具结果:{functionResultContent.Result}");
         }

         Console.ForegroundColor = ConsoleColor.White;
     }
     else
     {
         Console.ForegroundColor = ConsoleColor.Green;
         Console.WriteLine("本次没有调用工具");
         Console.ForegroundColor = ConsoleColor.White;
     }

     return response.Text;
 }

代码已经放到GitHub,地址:https://github.com/Ming-jiayou/mcp_demo。

将.env-example修改为.env应该就可以运行,如果报错,设置成嵌入的资源即可。

.env配置示例:

API_KEY=sk-xxx
BaseURL=https://api.siliconflow.cn/v1
ModelID=Qwen/Qwen2.5-72B-Instruct
MCPCommand=node
MCPArguments=D:/Learning/AI-related/fetch-mcp/dist/index.js

最后

对C#使用MCP感兴趣的朋友可以关注这个项目:https://github.com/PederHP/mcpdotnet。

有问题欢迎一起交流学习。

<think>好的,我现在需要帮助用户解决如何在数据库中为cursor实现或多配置MCP的问题。首先,我需要明确用户的需求。用户提到的“MCP”可能是指“MyCursor Provider”或者某种自定义的内容提供者,参考他们提供的代码片段,特别是引用[1]和[2],里面有一个MyCursor类继承自BetterCursorWrapper,并且在构造函数中传入了ContentProvider(mcp)。所以,用户可能想要了解如何创建这样的游标,并关联到特定的内容提供者。 接下来,我需要确定用户的问题背景。他们可能在开发Android应用,涉及到数据库操作,特别是使用ContentProvider和自定义Cursor。用户可能希望了解如何正确配置和管理这些游标,以便在数据变化时能够重新加载数据,如requery方法所示。 然后,我需要回忆Android中Cursor和ContentProvider的工作原理。Cursor用于遍历查询结果,而ContentProvider管理数据访问。用户提供的MyCursor类在requery时调用了MyContactsProvider.loadNewData(mcp),这可能用于刷新数据。因此,实现MCP的关键在于如何将自定义Cursor与ContentProvider结合,确保数据动态更新。 接下来,我需要组织回答结构。首先,解释MCP的可能含义,然后分步骤说明如何实现,包括创建自定义Cursor、集成内容提供者、配置数据源等。同时,需要引用用户提供的代码示例,说明各个部分的作用,比如构造函数如何接收ContentProvider实例,requery方法如何更新数据。 还要考虑用户可能的后续问题,比如动态数据加载、多配置管理、生命周期处理等,所以在回答中可以加入这些扩展点。此外,确保回答符合之前的系统指令,使用正确的中文,行内公式用$...$,但这里可能用不上数学公式,所以可能不需要。同时,生成相关问题,帮助用户进一步探索。 最后,检查回答是否符合所有要求:结构清晰,步骤明确,引用用户提供的代码作为示例,并添加引用标识。确保没有使用Markdown,语言口语化,自然分段,使用缩进表示代码块。</think>在Android开发中,为Cursor实现MCP(MyCursor Provider)通常涉及自定义Cursor类与内容提供者(ContentProvider)的集成。以下是关键实现步骤: 一、创建MCP的核心组件 1. 自定义Cursor类需继承CursorWrapper或其子类 ```java public class MyCursor extends BetterCursorWrapper { private ContentProvider mCp; // 内容提供者引用 public MyCursor(MatrixCursor mc, ContentProvider inCp) { super(mc); // 初始化基础游标[^1] mCp = inCp; // 注入内容提供者实例[^2] } } ``` 二、实现动态数据加载 1. 重写requery()方法实现数据刷新 ```java @Override public boolean requery() { MatrixCursor mc = MyContactsProvider.loadNewData(mCp); // 通过MCP获取新数据[^1] this.setInternalCursor(mc); // 更新底层游标数据 return super.requery(); } ``` 三、多配置实现方案 1. 通过构造器注入不同配置 ```java // 创建支持多个数据库配置的Cursor public MyCursor(MatrixCursor mc, ContentProvider cp, DatabaseConfig config) { super(mc); applyConfiguration(config); // 应用数据库连接配置 bindContentProvider(cp); // 绑定内容提供者 } ``` 四、关键配置管理 1. 数据源配置建议: - 使用$ContentResolver$处理URI映射 - 通过$UriMatcher$实现多表支持 - 配置参数应包含: ```java class McpConfig { String databaseName; int version; SQLiteOpenHelper helper; } ``` 五、生命周期管理要点 1. 必须实现$close()$方法释放资源: ```java @Override public void close() { releaseDatabaseConnection(); // 释放数据库连接 super.close(); // 执行父类关闭操作[^2] } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值