前言
如何运用开发语言 C#对 DeepSeek API 进行调用,使其能够如同 DeepSeek 官方网站那样涵盖分析、推理等流程,最终以类似于动态打字的效果将答案输出至界面。本文章包含了全部需要的运行代码,代码下载后,只需要修改deepseek API key就可以完美运行,效果呈现如下:
AI Agent 动态输出效果
deepseek API key
需要前往 DeepSeek 官方网站去申请 API Key,具体操作方式如下所示,该API Key是后续调用 DeepSeek API 进行开发和应用的重要凭证,务必妥善保管和使用。
图1 API申请入口
图2 API Key创建
Windows 窗体设计
首先,创建一个 C# 的 Windows 窗体应用程序。在这个过程中,采用 panel 控件来进行布局,以实现更加合理和美观的界面设计。在界面中,答案部分设置为一个 RichTextBox 控件,用于展示详细和丰富的答案内容。而在其下方的问题部分,同样也设置一个 RichTextBox 控件,方便用户输入和编辑相关的问题。此外,还需创建一个 Button 控件,并且对其 Text 属性进行修改,将其设定为“DeepSeek”。 设计界面如下:
界面设计
API类函数
对 API 函数予以封装,并将其存于一个类文件之内,同时,APIKey 置于类的 app.config 文件之中,完整代码如下:
using System;
using System.Text;
using System.Net.Http;
using System.Threading.Tasks;
using System.Text.Json;
using System.Windows.Forms;
using static System.Net.Mime.MediaTypeNames;
using System.Configuration;
using System.Runtime.InteropServices;
namespace DeepSeekService
{
public class DeepSeepClient
{
public static string ApiKey { get { return ConfigurationManager.AppSettings["DeepseekAPI_KEY"].ToString(); } }
private readonly HttpClient client;
private const string ApiUrl = "https://api.deepseek.com/v1/chat/completions";
private const string ModelName = "deepseek-chat";
public DeepSeepClient() {
client = new HttpClient();
client.DefaultRequestHeaders.Add("Authorization", $"Bearer {ApiKey}");
client.DefaultRequestHeaders.Add("Accept", "text/event-stream");
}
enum ProcessingPhase
{
Analyzing,
Researching,
Reasoning,
Finalizing
}
public async Task RunDeepSeek(RichTextBox rt, string prompt)
{
var requestData = new
{
model = ModelName,
messages = new[]
{
new { role = "system", content = "请按以下步骤回答:1.分析问题 2.联网资料 3.推理 4.答案. 提供的参考知识库:\n\\n经典作品常常与我们自己的生活息息相关。在《经典常谈》中,作者通过对比古今文化,指出许多经典所讨论的主题,例如爱情、死亡、自由与责任,依然是人类永恒的心结。这些主题在不同的历史阶段被赋予了全新的意义,使我们在面对生活中的困惑与挑战时,能够从中寻找到启发与指引。阅读经典,让我意识到自己的生活困扰其实早已被前人探讨过,不同的是我们面对这些问题的态度与答案。我们在阅读经典的同时,也在审视自己的价值观和人生观。我开始意识到,经典作品所引发的思考往往比作品本身更为重要。与其一味追求书中所传达的思想,不如在阅读中找到与自己内心的对话,发掘出属于自己的理解与感悟。\\n\\n最后,经典的吸引力在于它们的共鸣。正是这些作品,让我们在个体经验与历史智慧之间架起了一座桥梁。在这个信息碎片化的时代,经典作品无疑为我们的精神世界提供了一种持久的滋养。它们教会我们如何在复杂的社会中保持清醒,自省以及不断追求更高的理想。\\n\\n《经典常谈》带给我的新认识是,经典不仅是文化的承载体,更是人类思考与自我发现的工具。在未来的日子里,我将继续怀揣这份对经典的敬畏与热爱,让其在我的人生旅途中指引前行的方向。\";" },
new { role = "user", content = prompt }
},
stream = true
};
try
{
using (var request = new HttpRequestMessage(HttpMethod.Post, ApiUrl))
{
request.Content = new StringContent(
JsonSerializer.Serialize(requestData),
Encoding.UTF8,
"application/json");
using (var response = await client.SendAsync(
request,
HttpCompletionOption.ResponseHeadersRead))
{
response.EnsureSuccessStatusCode();
var currentPhase = ProcessingPhase.Analyzing;
var phaseBuffer = new StringBuilder();
var fullAnswer = new StringBuilder();
rt.AppendText(" Thinking...");
rt.AppendText(Environment.NewLine);
rt.Refresh();
using (var stream = await response.Content.ReadAsStreamAsync())
using (var reader = new System.IO.StreamReader(stream))
{
while (!reader.EndOfStream)
{
var line = await reader.ReadLineAsync();
if (string.IsNullOrWhiteSpace(line)) continue;
if (line.StartsWith("data: "))
{
var jsonData = line.Substring(6).Trim();
if (jsonData == "[DONE]") break;
using (var jsonDoc = JsonDocument.Parse(jsonData))
{
var choices = jsonDoc.RootElement
.GetProperty("choices");
foreach (var choice in choices.EnumerateArray())
{
var delta = choice.GetProperty("delta");
if (delta.TryGetProperty("content", out var content))
{
var text = content.GetString();
phaseBuffer.Append(text);
fullAnswer.Append(text);
#region "enabled if required" // 检测阶段标记
//var bufferStr = phaseBuffer.ToString();
//if (bufferStr.Contains("【分析】"))
//{
// UpdatePhase(ref currentPhase, ProcessingPhase.Analyzing, "问题分析",rt);
// phaseBuffer.Clear();
//}
//else if (bufferStr.Contains("【资料】"))
//{
// UpdatePhase(ref currentPhase, ProcessingPhase.Researching, "信息检索",rt);
// phaseBuffer.Clear();
//}
//else if (bufferStr.Contains("【推理】"))
//{
// UpdatePhase(ref currentPhase, ProcessingPhase.Reasoning, "逻辑推理",rt);
// phaseBuffer.Clear();
//}
#endregion
// output
AddTextToRichTextBox(rt, text);
}
}
}
}
}
}
Console.ForegroundColor = ConsoleColor.Green;
}
}
}
catch (Exception e)
{
Console.WriteLine($"\n exception handling: {e.Message}");
}
}
private static void AddTextToRichTextBox(RichTextBox rt, string text)
{
rt.SuspendLayout();
rt.AppendText(text);
rt.SelectionStart = rt.TextLength;
rt.ScrollToCaret();
rt.ResumeLayout();
}
static void UpdatePhase(ref ProcessingPhase currentPhase, ProcessingPhase newPhase, string phaseName,RichTextBox rt)
{
if (currentPhase == newPhase) return;
currentPhase = newPhase;
Console.ForegroundColor = ConsoleColor.Cyan;
Console.WriteLine($"\n\n Current Phase:{phaseName}");
rt.AppendText("Current Phase:{phaseName}");
rt.ScrollToCaret();
Console.ResetColor();
}
}
}
app.config配置
于 app.config 配置文件当中,对 API key 的配置情况如下:
Windows 窗体代码
创建一个 Windows 窗体,借助 Panel 来实施布局,而后创建两个 RichTextBox ,分别用以呈现输入的内容以及答案。创建一个button,命名为btDeepseek
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Forms;
using DeepSeekService;
namespace WindowsFormsApp1
{
public partial class Form1 : Form
{
public Form1()
{
InitializeComponent();
this.WindowState = FormWindowState.Maximized;
}
private async void btDeepseek_Click(object sender, EventArgs e)
{
if (!string.IsNullOrEmpty(rtbAnswer.Text.Trim()))
{
rtbAnswer.AppendText(Environment.NewLine);
rtbAnswer.AppendText(Environment.NewLine);
}
DeepSeepClient dsClient = new DeepSeepClient();
await dsClient.RunDeepSeek(rtbAnswer, rtbQuestion.Text);
}
}
}
总结
当运行以上代码之后,便成功地实现了对 deepseek api 的调用。在此过程中,能够根据具体的输入,动态地输出相应的结果内容,欢迎大家提出问题。