在 .NET AI 聊天应用中升级到 Microsoft 代理框架
引言
随着人工智能技术的快速发展,简单的聊天机器人已经不能满足日益复杂的业务需求。Microsoft 推出的 Agent Framework 为 .NET 开发者提供了构建智能代理的强大工具,能够实现多步骤工作流、自主决策和复杂任务编排。本文将详细介绍如何将基于 .NET AI 模板的标准聊天应用升级到 Microsoft 代理框架,利用其强大的功能和灵活性来构建更智能的应用。
正文内容
1. Microsoft 代理框架概述
Microsoft Agent Framework 是微软推出的预览框架,专为在 .NET 中构建 AI 代理而设计。它超越了简单聊天机器人的能力,提供了以下核心功能:
- 通过多步骤工作流程进行推理和计划
- 使用工具和函数与 API、数据库和服务交互
- 在整个对话中维护上下文
- 基于指令和数据做出自主决策
- 在多代理场景中与其他代理协调
该框架建立在 .NET 开发者熟悉的模式之上,如依赖注入、中间件和遥测,并与 Microsoft.Extensions.AI 深度集成。
2. 先决条件
在开始升级前,需要准备以下环境:
- 已安装 .NET 9 SDK
- Visual Studio 或带有 C# Dev Kit 的 Visual Studio Code
- 可访问 Azure OpenAI 的 Azure 账户,或使用 GitHub 模型
- 已安装 .NET AI 应用模板
- 基本熟悉 .NET、Blazor 和 AI 概念
3. 创建基础 AI 聊天应用
3.1 安装模板
首先使用官方 .NET AI 模板创建基线聊天应用:
dotnet new install Microsoft.Extensions.AI.Templates
3.2 创建项目
可以通过 Visual Studio 或 CLI 创建项目:
Visual Studio 方式:
- 打开 Visual Studio 2022
- 选择"创建新项目"
- 搜索"AI Chat Web App"
- 配置项目名称和位置
- 选择 Azure OpenAI 作为 AI 提供程序
- 为矢量存储选择"本地磁盘"
- 选择 .NET Aspire 进行业务流程

CLI 方式:
使用 dotnet new 命令创建项目,配置选项与 Visual Studio 相同。
3.3 项目结构
模板生成的解决方案包含三个项目:
ChatApp20/ ├── ChatApp20.Web/ # 带有聊天 UI 的 Blazor Server 应用 ├── ChatApp20.AppHost/ # .NET Aspire 业务流程 └── ChatApp20.ServiceDefaults/ # 共享服务配置
主要工作将在 ChatApp20.Web 项目中进行。

4. 添加 Microsoft 代理框架
4.1 安装必要的 NuGet 包
首先将 Microsoft Agent Framework 包添加到 ChatApp20.Web.csproj:
<ItemGroup> <!-- 保留现有包 --> <PackageReference Include="Aspire.Azure.AI.OpenAI" Version="9.5.1-preview.1.25502.11" /> <PackageReference Include="Microsoft.Extensions.AI.OpenAI" Version="9.10.0-preview.1.25513.3" /> <!-- 添加 Microsoft 代理框架包 --> <PackageReference Include="Microsoft.Agents.AI" Version="1.0.0-preview.251009.1" /> <PackageReference Include="Microsoft.Agents.AI.Abstractions" Version="1.0.0-preview.251009.1" /> <PackageReference Include="Microsoft.Agents.AI.Hosting" Version="1.0.0-preview.251009.1" /> <PackageReference Include="Microsoft.Agents.AI.Hosting.OpenAI" Version="1.0.0-alpha.251009.1" /> <PackageReference Include="Microsoft.Agents.AI.OpenAI" Version="1.0.0-preview.251009.1" /> </ItemGroup>

4.2 创建专用搜索功能服务
为了更好的关注点分离和可测试性,创建一个新的 SearchFunctions.cs 服务:
using System.ComponentModel; namespace ChatApp20.Web.Services; public class SearchFunctions { private readonly SemanticSearch _semanticSearch; public SearchFunctions(SemanticSearch semanticSearch) { _semanticSearch = semanticSearch; } [Description("使用短语或关键字搜索信息")] public async Task<IEnumerable<string>> SearchAsync( [Description("要搜索的短语")] string searchPhrase, [Description("如果可能,指定文件名仅搜索该文件")] string? filenameFilter = null) { var results = await _semanticSearch.SearchAsync(searchPhrase, filenameFilter, maxResults: 5); return results.Select(result => $"<result filename="{result.DocumentId}" page_number="{result.PageNumber}">{result.Text}</result>"); } }
4.3 在 Program.cs 中注册 AI 代理
使用 Agent Framework 的托管扩展配置 AI 代理:
// 注册使用代理框架的 AI 代理 builder.AddAIAgent("ChatAgent", (sp, key) => { var logger = sp.GetRequiredService<ILogger<Program>>(); var searchFunctions = sp.GetRequiredService<SearchFunctions>(); var chatClient = sp.GetRequiredService<IChatClient>(); // 创建并配置 AI 代理 var aiAgent = chatClient.CreateAIAgent( name: key, instructions: "你是一个有用的代理,提供简短而有趣的回答。", description: "帮助用户提供简短而有趣回答的 AI 代理。", tools: [AIFunctionFactory.Create(searchFunctions.SearchAsync)] ) .AsBuilder() .UseOpenTelemetry(configure: c => c.EnableSensitiveData = builder.Environment.IsDevelopment()) .Build(); return aiAgent; }); // 注册 SearchFunctions 以便通过 DI 注入到代理中 builder.Services.AddSingleton<SearchFunctions>();
4.4 更新聊天组件
更新 Chat.razor 以使用新的 AI 代理:
@inject IServiceProvider ServiceProvider @using Microsoft.Agents.AI @code { private AIAgent aiAgent = default!; protected override void OnInitialized() { // 解析 Program.cs 中注册为"ChatAgent"的键控 AI 代理 aiAgent = ServiceProvider.GetRequiredKeyedService<AIAgent>("ChatAgent"); } private async Task AddUserMessageAsync(ChatMessage userMessage) { // 用代理流式处理替换 ChatClient.GetStreamingResponseAsync await foreach (var update in aiAgent.RunStreamingAsync( messages: messages.Skip(statefulMessageCount), cancellationToken: currentResponseCancellation.Token)) { var responseUpdate = update.AsChatResponseUpdate(); messages.AddMessages(responseUpdate, filter: c => c is not TextContent); responseText.Text += update.Text; chatOptions.ConversationId = responseUpdate.ConversationId; ChatMessageItem.NotifyChanged(currentResponseMessage); } } }
5. 运行和测试增强应用
5.1 使用 .NET Aspire 运行
.NET Aspire 提供了服务发现、统一日志记录和遥测、健康检查等功能:
- 运行应用,Aspire 仪表板会自动在浏览器中打开
- 首次运行时,系统会提示配置 Azure OpenAI:
- 选择 Azure 订阅
- 选择或创建资源组
- 选择或预配 Azure OpenAI 资源
- 确保部署了聊天模型(如 gpt-4o-mini)和嵌入模型(如 text-embedding-3-small)

5.2 测试代理
应用运行后,在 Aspire 仪表板中点击 Web 端点(通常是 https://localhost:7001)进行测试:
基本对话:
用户: 你好!你好吗? 代理: 嘿!我很好——充满电,就像应急救生包一样。
带有语义搜索的工具调用:
用户: 应急救生包应该包括什么? 代理: 简短的救生包清单(有趣版) 急救用品——绷带、纱布、消毒剂。 <citation filename='Example_Emergency_Survival_Kit.pdf' page_number='1'>水和食物供应</citation>
特定文件查询:
用户: 告诉我关于 GPS 手表的功能 代理: GPS 手表包括... <citation filename='Example_GPS_Watch.pdf' page_number='2'>实时跟踪</citation>

6. 高级场景
6.1 向代理添加更多工具
可以轻松扩展代理功能,例如添加天气查询:
public class WeatherFunctions { [Description("获取某个位置的当前天气")] public async Task<string> GetWeatherAsync( [Description("城市和州/国家")] string location) { // 调用天气 API return $"{location}的天气: 晴天, 72°F"; } } // 在 Program.cs 中 builder.Services.AddSingleton<WeatherFunctions>(); builder.AddAIAgent("ChatAgent", (sp, key) => { var searchFunctions = sp.GetRequiredService<SearchFunctions>(); var weatherFunctions = sp.GetRequiredService<WeatherFunctions>(); var chatClient = sp.GetRequiredService<IChatClient>(); return chatClient.CreateAIAgent( name: key, instructions: "你可以搜索文档和查询天气...", tools: [ AIFunctionFactory.Create(searchFunctions.SearchAsync), AIFunctionFactory.Create(weatherFunctions.GetWeatherAsync) ] ).Build(); });
6.2 多代理场景
代理框架可以轻松协调多个专用代理:
// 注册研究代理 builder.AddAIAgent("ResearchAgent", (sp, key) => { var chatClient = sp.GetRequiredService<IChatClient>(); var searchFunctions = sp.GetRequiredService<SearchFunctions>(); return chatClient.CreateAIAgent( name: "ResearchAgent", instructions: "你是研究专家。从文档中查找和总结信息。", tools: [AIFunctionFactory.Create(searchFunctions.SearchAsync)] ).Build(); }); // 注册写作代理 builder.AddAIAgent("WritingAgent", (sp, key) => { var chatClient = sp.GetRequiredService<IChatClient>(); return chatClient.CreateAIAgent( name: "WritingAgent", instructions: "你是写作专家。获取信息并创建结构良好、引人入胜的内容。", tools: [] ).Build(); }); // 注册协调代理 builder.AddAIAgent("CoordinatorAgent", (sp, key) => { var chatClient = sp.GetRequiredService<IChatClient>(); var researchAgent = sp.GetRequiredKeyedService<AIAgent>("ResearchAgent"); var writingAgent = sp.GetRequiredKeyedService<AIAgent>("WritingAgent"); // 创建委托给其他代理的函数 async Task<string> ResearchAsync(string topic) { var messages = new[] { new ChatMessage(ChatRole.User, topic) }; var result = await researchAgent.RunAsync(messages); return result.Text ?? ""; } async Task<string> WriteAsync(string content) { var messages = new[] { new ChatMessage(ChatRole.User, $"基于以下内容写文章: {content}") }; var result = await writingAgent.RunAsync(messages); return result.Text ?? ""; } return chatClient.CreateAIAgent( name: "CoordinatorAgent", instructions: "协调研究和写作以创建全面的文章。", tools: [ AIFunctionFactory.Create(ResearchAsync), AIFunctionFactory.Create(WriteAsync) ] ).Build(); });
6.3 自定义代理中间件
可以添加自定义中间件进行日志记录、缓存或自定义行为:
builder.AddAIAgent("ChatAgent", (sp, key) => { var chatClient = sp.GetRequiredService<IChatClient>(); var searchFunctions = sp.GetRequiredService<SearchFunctions>(); var logger = sp.GetRequiredService<ILogger<Program>>(); return chatClient.CreateAIAgent( name: key, instructions: "...", tools: [AIFunctionFactory.Create(searchFunctions.SearchAsync)] ) .AsBuilder() .Use(async (messages, options, next, cancellationToken) => { // 自定义预处理 logger.LogInformation("代理正在处理 {MessageCount} 条消息", messages.Count()); // 调用管道中的下一个 var result = await next(messages, options, cancellationToken); // 自定义后处理 logger.LogInformation("代理生成了包含 {ContentCount} 个内容项的响应", result.Contents.Count); return result; }) .UseOpenTelemetry(configure: c => c.EnableSensitiveData = true) .Build(); });
7. 最佳实践
7.1 设计清晰的工具描述
代理工具调用的质量很大程度上取决于良好的描述:
[Description("在产品文档中搜索特定信息。" + "当用户询问功能、规格或产品使用方法时使用此工具。" + "返回带有文件名和页码的相关摘录以供引用。")] public async Task<IEnumerable<string>> SearchAsync( [Description("要搜索的特定短语、关键字或问题。" + "要具体并包含相关上下文。")] string searchPhrase, [Description("可选: 要搜索的精确文件名(如'ProductManual.pdf')。" + "留空则搜索所有文档。")] string? filenameFilter = null) { // 实现 }
7.2 测试代理行为
为代理工具创建单元测试,为代理工作流创建集成测试:
public class SearchFunctionsTests { [Fact] public async Task SearchAsync_WithValidQuery_ReturnsResults() { // 准备 var mockSemanticSearch = new Mock<SemanticSearch>(); mockSemanticSearch .Setup(s => s.SearchAsync("test", null, 5)) .ReturnsAsync(new List<IngestedChunk> { new IngestedChunk { DocumentId = "test.pdf", PageNumber = 1, Text = "测试内容" } }); var searchFunctions = new SearchFunctions(mockSemanticSearch.Object); // 执行 var results = await searchFunctions.SearchAsync("test"); // 断言 Assert.NotEmpty(results); Assert.Contains("测试内容", results.First()); } }
7.3 监控代理性能
使用 Application Insights 或 .NET Aspire 的仪表板监控:
- 每次代理交互的令牌使用量
- 工具调用模式(使用哪些工具,使用频率)
- 代理操作的响应时间
- 工具调用的错误率
- 通过反馈机制获得的用户满意度
8. 性能考虑
8.1 流式与非流式
代理框架支持流式和非流式响应:
使用流式的情况:
- 构建交互式聊天界面
- 用户期望实时反馈
- 处理长时间运行的查询
使用非流式的情况:
- 后台处理
- 批量操作
- 简单 API 端点
8.2 工具调用优化
尽量减少不必要的工具调用:
// 好: 具体指令 "仅当用户询问关于文档的具体问题时使用搜索工具。 如果可以从常识中回答,则不搜索。" // 不好: 模糊指令 "你可以访问搜索工具。"
9. 部署到 Azure
应用已准备好使用 .NET Aspire 的 Azure 预配进行部署:
# 登录 Azure az login # 创建 Azure 资源 cd ChatApp20.AppHost azd init azd up
这将:
- 预配 Azure OpenAI 资源
- 将 Web 应用部署到 Azure 容器应用
- 设置 Application Insights 进行监控
- 配置服务连接和身份验证
结论
通过本文的步骤,我们成功将标准的 AI 聊天应用升级为使用 Microsoft 代理框架的智能代理系统。这一升级带来了更清晰的架构关注点分离、更轻松的测试和内置的可观测性,同时仍然使用 .NET 开发者熟悉的模式。
Microsoft 代理框架的优势在于它不需要开发者学习全新的方式,而是建立在依赖注入、中间件和遥测等熟悉的 .NET 概念之上。无论是简单的工具调用还是复杂的多代理协调,该框架都提供了强大而灵活的功能来满足各种业务需求。
随着 AI 技术的不断发展,采用代理框架将使您的应用能够更好地适应未来需求,提供更智能、更自然的用户体验。