3步打造C# API安全密盾

引言:API 安全的重要性

在数字化浪潮中,应用程序编程接口(API)已成为不同软件系统之间通信和数据交互的关键桥梁。无论是企业内部的微服务架构,还是面向外部用户的在线服务,API 都承担着数据传输和业务逻辑执行的重任。据相关数据显示,如今大部分互联网流量(71%)都是 API 调用 ,这充分凸显了 API 在现代互联网架构中的核心地位。

但随着 API 的广泛应用,其安全问题也日益凸显。API 作为访问敏感数据的直接途径,一旦遭受攻击,后果不堪设想。数据泄露可能导致用户个人信息、商业机密等重要数据被窃取,给企业和用户带来巨大的损失;身份验证失败可能让攻击者绕过权限控制,非法访问受限资源;中间人攻击则可能篡改传输数据,破坏数据的完整性和真实性;注入攻击、DDoS 攻击等也会对系统的稳定性和可用性造成严重威胁。根据 Fastly 的一项调查,95% 的企业在过去 1 年中遇到过 API 安全问题 ,而 Marsh McLennan 的研究表明,与 API 相关的安全事件每年给全球企业造成的损失高达 750 亿美元 。这些惊人的数据警示着我们,API 安全已成为保障数据安全和业务流程正常运行的关键环节。

C# 作为一种强大的编程语言,在构建安全可靠的 API 方面具有显著优势。它依托于丰富的.NET 框架,为开发者提供了一套全面且易用的加密和安全相关的类库,如 System.Security.Cryptography 命名空间,涵盖了从对称加密、非对称加密到哈希函数等多种常用的安全机制,能够满足不同场景下的安全需求。接下来,我们就深入探讨如何利用 C# 的这些特性,分三步打造坚不可摧的 API 安全密盾。

第一步:加密前的密谋 —— 了解基础

(一)加密概念介绍

在深入探讨如何使用 C# 构建 API 安全防护体系之前,我们需要先熟悉几种常见的加密概念及其特性,它们是构建安全密盾的基石。

对称加密,就像一把钥匙开一把锁,加密和解密过程使用的是同一个密钥。这种加密方式的显著优势在于加密和解密的速度极快,能够高效地处理大量数据。例如,在一些对数据处理速度要求较高的场景中,如实时数据传输、大规模数据存储加密等,对称加密能够迅速完成加密和解密操作,确保数据的快速流转。但它也存在一个明显的短板,那就是密钥的管理难度较大。由于加密和解密使用相同的密钥,在数据传输之前,发送方和接收方必须通过安全的方式商定并妥善保存这个密钥。一旦密钥在传输过程中被窃取,整个加密体系就会面临被攻破的风险 。常见的对称加密算法有 AES(Advanced Encryption Standard)、DES(Data Encryption Standard)等,其中 AES 以其出色的安全性和性能,成为了目前应用最为广泛的对称加密算法之一 。

非对称加密则引入了一对密钥:公钥和私钥。公钥可以公开分发,任何人都可以使用它来加密消息;而私钥则由持有者妥善保管,只有私钥的拥有者才能用它来解密使用对应公钥加密的消息。这种加密方式的安全性极高,因为从公钥几乎无法推断出私钥,即使攻击者获取了公钥和密文,也难以破解出原始数据。它常用于数字签名、身份验证等场景,比如在电子合同签署过程中,使用非对称加密可以确保签名的不可伪造性和消息的完整性。然而,非对称加密的加密和解密过程涉及复杂的数学运算,速度相对较慢,在处理大量数据时效率较低 。典型的非对称加密算法有 RSA(Rivest-Shamir-Adleman)、ECC(Elliptic Curve Cryptography)等 。

哈希(Hash)是一种单向的摘要算法,它能将任意长度的输入数据转换为固定长度的哈希值。哈希的主要特点是不可逆性,即无法从哈希值反向推导出原始数据。它常用于验证数据的完整性,比如在文件传输过程中,发送方计算文件的哈希值并一同发送,接收方在收到文件后重新计算哈希值,若两者一致,则说明文件在传输过程中未被篡改。但哈希并不适合直接用于加密通信,因为它无法提供保密性,任何人都可以计算出相同数据的哈希值 。常见的哈希算法有 SHA-256(Secure Hash Algorithm 256-bit)、MD5(Message-Digest Algorithm 5)等,不过由于 MD5 存在碰撞风险,安全性逐渐受到质疑,在安全要求较高的场景中已逐渐被 SHA-256 等更安全的算法所取代 。

(二)混合加密策略原理

在 API 加密的实际应用中,单一的加密方式往往难以满足复杂的安全需求,因此我们通常采用混合加密策略,将对称加密和非对称加密的优势结合起来。

具体来说,对称加密用于保护数据主体,因为它的加密速度快,能够高效地处理大量的数据传输,确保 API 的性能不受太大影响。例如,在 API 请求和响应中,对包含敏感信息的主体数据进行对称加密,使得即使数据在传输过程中被截获,攻击者也难以直接获取其中的内容。

而非对称加密则用于保护对称加密所使用的密钥。由于对称加密的密钥管理是一个难题,通过非对称加密,我们可以使用接收方的公钥对对称加密的密钥进行加密传输。这样,只有拥有对应私钥的接收方才能解密得到对称密钥,从而保证了对称密钥在传输过程中的安全性。

以一个简单的 API 数据传输场景为例,客户端生成一个随机的对称密钥,使用该密钥对要发送给服务器的数据进行对称加密,得到密文数据。然后,客户端使用服务器的公钥对这个对称密钥进行非对称加密,得到加密后的密钥。在传输时,将密文数据和加密后的密钥一同发送给服务器。服务器收到后,首先使用自己的私钥解密得到对称密钥,再用这个对称密钥解密密文数据,从而获取到原始的明文数据。

这种混合加密策略既利用了对称加密的高效性,又借助了非对称加密的高安全性,实现了安全与性能的平衡,是保障 API 数据安全传输的常用且有效的手段 。

第二步:C# 加密实战 —— 走起!

(一)准备阶段:引入帮手

在 C# 项目中,我们首先要引入加密相关的命名空间,这些命名空间就像是我们打造安全密盾的得力工具。主要涉及System.Security.Cryptography和System.Text。System.Security.Cryptography命名空间是加密的核心库,它提供了丰富的加密算法和工具类,涵盖了对称加密、非对称加密、哈希算法等各种加密机制,是实现数据加密和解密的关键所在 。而System.Text命名空间则用于字符串处理,在加密过程中,我们经常需要将字符串转换为字节数组进行处理,或者将加密后的字节数组再转换回字符串形式,System.Text命名空间中的类和方法为这些操作提供了便利 。

引入这两个命名空间的代码如下:

using System.Security.Cryptography; // 加密核心库
using System.Text; // 字符串处理

通过引入这两个命名空间,我们的项目就具备了使用各种加密算法和进行字符串处理的能力,为后续的加密实战奠定了基础。

(二)对称加密:AES 的秘密

在了解了加密的基本概念和准备好必要的工具后,我们开始使用 AES 进行对称加密的实战。AES(Advanced Encryption Standard)是一种被广泛应用的对称加密算法,以其安全性高、速度快等优点,成为了保护数据主体的首选算法之一 。

  1. 代码实现

以下是使用 AES 进行对称加密的 C# 代码:

public static byte[] EncryptStringToBytes_Aes(string plainText, byte[] key, byte[] iv)
{
    using (Aes aesAlg = Aes.Create())
    {
        aesAlg.Key = key;
        aesAlg.IV = iv;
        ICryptoTransform encryptor = aesAlg.CreateEncryptor(aesAlg.Key, aesAlg.IV);
        using (MemoryStream msEncrypt = new MemoryStream())
        {
            using (CryptoStream csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write))
            {
                using (StreamWriter swEncrypt = new StreamWriter(csEncrypt))
                {
                    swEncrypt.Write(plainText);
                }
                return msEncrypt.ToArray();
            }
        }
    }
}
  1. 原理注释

这段代码就像是一场精心编排的加密魔术,将明文字符串plainText加密成字节数组返回。具体步骤如下:

  • 首先,通过Aes.Create()创建一个 AES 加密算法的实例aesAlg 。这个实例是我们进行加密操作的核心工具,它提供了一系列的属性和方法来配置和执行加密过程 。

  • 然后,设置aesAlg的Key和IV属性。Key是加密和解密使用的密钥,必须妥善保管,因为一旦密钥泄露,加密的数据就可能被轻易破解 。IV(Initialization Vector,初始化向量)是一个随机值,用于增加加密的安全性,它和密钥一起确保相同的明文在不同的加密操作中生成不同的密文,防止攻击者通过分析密文来破解加密算法 。

  • 接着,调用aesAlg.CreateEncryptor(aesAlg.Key, aesAlg.IV)创建一个加密器encryptor 。这个加密器负责执行实际的加密操作,它根据设置的密钥和初始化向量,将明文数据按照 AES 算法的规则进行加密变换 。

  • 之后,创建一个MemoryStream对象msEncrypt,它用于在内存中存储加密后的字节数据 。同时,创建一个CryptoStream对象csEncrypt,并将其与msEncrypt和encryptor关联起来 。CryptoStream就像是一个桥梁,它将加密器和内存流连接起来,使得加密后的字节数据能够正确地写入到内存流中 。

  • 再创建一个StreamWriter对象swEncrypt,它用于将明文字符串写入到CryptoStream中 。在写入过程中,CryptoStream会自动调用加密器对写入的数据进行加密,并将加密后的字节数据存储到MemoryStream中 。

  • 最后,通过msEncrypt.ToArray()将MemoryStream中的加密字节数据转换为字节数组并返回 。这个字节数组就是加密后的密文,它可以被安全地传输或存储 。

在实际应用中,务必妥善保管好密钥key和初始化向量iv,可以将它们存储在安全的配置文件中,或者使用硬件安全模块(HSM)等设备来存储和管理,以确保加密的安全性 。

(三)非对称加密:RSA 的守护

虽然 AES 对称加密可以高效地保护数据主体,但密钥的传输安全是一个关键问题。为了解决这个问题,我们使用 RSA 非对称加密来保护 AES 密钥的传输。RSA(Rivest-Shamir-Adleman)是一种经典的非对称加密算法,它基于大整数分解的数学难题,提供了极高的安全性 。

  1. 代码实现

以下是用 RSA 加密 AES 密钥的 C# 代码:

public static byte[] EncryptUsingRSA(byte[] dataToEncrypt, RSAParameters publicKey)
{
    using (RSACryptoServiceProvider rsa = new RSACryptoServiceProvider())
    {
        rsa.ImportParameters(publicKey);
        return rsa.Encrypt(dataToEncrypt, false); // false 表示使用公钥加密
    }
}
  1. 原理注释

这段代码的作用是使用 RSA 的公钥对需要加密的数据dataToEncrypt(通常是 AES 密钥)进行加密 。具体实现步骤如下:

  • 首先,创建一个RSACryptoServiceProvider对象rsa,它是 RSA 加密算法在 C# 中的实现类,提供了一系列用于 RSA 加密、解密、签名和验证的方法 。

  • 然后,通过rsa.ImportParameters(publicKey)导入 RSA 的公钥参数publicKey 。公钥参数包含了用于加密的关键信息,通过导入公钥参数,rsa对象就可以使用这个公钥进行加密操作 。

  • 最后,调用rsa.Encrypt(dataToEncrypt, false)方法对数据dataToEncrypt进行加密 。其中,第二个参数false表示使用公钥进行加密 。加密后的字节数组将被返回,这个加密后的字节数组就是经过 RSA 公钥加密后的 AES 密钥,只有拥有对应私钥的接收方才能解密得到原始的 AES 密钥 。

在这个过程中,RSA 就像一位忠诚的骑士,手持公钥之剑,守护着我们的 AES 密钥安全过河(在网络中传输) 。公钥可以公开分发,任何人都可以使用它来加密数据,但只有持有对应私钥的接收方才能解密 。因此,私钥的安全性至关重要,必须严格保密,防止私钥泄露导致加密数据被破解 。

第三步:实战演练 —— 整合加密步骤

(一)生成 RSA 密钥对

在实际应用中,首先需要生成 RSA 密钥对,这是实现非对称加密的基础。在 C# 中,使用RSA.Create()方法可以轻松生成 RSA 密钥对,其中包含公钥和私钥 。公钥用于加密数据,私钥用于解密数据,两者相互关联且缺一不可 。以下是生成 RSA 密钥对的代码示例:

using (RSA rsa = RSA.Create())
{
    // 导出公钥参数
    RSAParameters publicKey = rsa.ExportParameters(false);
    // 导出私钥参数
    RSAParameters privateKey = rsa.ExportParameters(true);
    // 这里可以将公钥和私钥保存到安全的地方,例如配置文件或密钥管理系统
    // 为了演示方便,我们直接打印公钥和私钥的字节数组
    Console.WriteLine("公钥: " + BitConverter.ToString(publicKey.Exponent).Replace("-", "") + BitConverter.ToString(publicKey.Modulus).Replace("-", ""));
    Console.WriteLine("私钥: " + BitConverter.ToString(privateKey.D).Replace("-", "") + BitConverter.ToString(privateKey.Modulus).Replace("-", ""));
}

在这段代码中,RSA.Create()创建了一个RSA对象实例,它是 RSA 加密算法的具体实现 。通过rsa.ExportParameters(false)方法导出公钥参数,其中false表示不包含私钥信息 ;通过rsa.ExportParameters(true)方法导出私钥参数,true表示包含私钥信息 。在实际应用中,私钥必须严格保密,建议将其存储在安全的硬件设备(如硬件安全模块 HSM)或经过加密的存储介质中 ,以防止私钥泄露导致加密体系被攻破 。

(二)使用 AES 加密数据

选择一段明文数据,然后利用前面实现的 AES 加密方法对其进行加密 。假设我们有一段表示用户敏感信息的字符串,需要将其加密后传输 。以下是使用 AES 加密数据的示例代码:

// 假设这是我们要加密的明文数据
string plainText = "用户的敏感信息,如身份证号、银行卡号等";
// 生成AES加密所需的密钥和初始化向量
byte[] key = new byte[32]; // 32字节的密钥,适用于AES-256
byte[] iv = new byte[16]; // 16字节的初始化向量
using (RNGCryptoServiceProvider rng = new RNGCryptoServiceProvider())
{
    rng.GetBytes(key);
    rng.GetBytes(iv);
}
// 调用AES加密方法
byte[] encryptedBytes = EncryptStringToBytes_Aes(plainText, key, iv);
// 将加密后的字节数组转换为Base64字符串,以便于传输或存储
string encryptedText = Convert.ToBase64String(encryptedBytes);
Console.WriteLine("加密后的密文: " + encryptedText);

在这段代码中,首先定义了要加密的明文字符串plainText 。然后通过RNGCryptoServiceProvider生成随机的 AES 密钥key和初始化向量iv ,确保每次加密使用的密钥和向量都是不同的,增加加密的安全性 。接着调用之前实现的EncryptStringToBytes_Aes方法对明文进行加密,得到加密后的字节数组encryptedBytes 。最后,为了便于传输和存储,将加密后的字节数组转换为 Base64 编码的字符串encryptedText 。

(三)用 RSA 加密 AES 密钥

为了保证 AES 密钥在传输过程中的安全,我们使用 RSA 的公钥对 AES 密钥进行加密 。假设已经生成了 RSA 密钥对,并且获取到了公钥参数 。以下是用 RSA 加密 AES 密钥的代码:

// 假设已经生成了RSA密钥对,并获取到公钥参数
RSAParameters publicKey = GetPublicKey(); // 假设这是获取公钥的方法
// 使用RSA加密AES密钥
byte[] encryptedAesKey = EncryptUsingRSA(key, publicKey);
// 将加密后的AES密钥转换为Base64字符串,以便于传输
string encryptedAesKeyText = Convert.ToBase64String(encryptedAesKey);
Console.WriteLine("加密后的AES密钥: " + encryptedAesKeyText);

在这段代码中,首先通过GetPublicKey()方法获取 RSA 公钥参数(这里假设已经有一个方法来获取公钥) 。然后调用EncryptUsingRSA方法,使用 RSA 公钥对 AES 密钥key进行加密,得到加密后的 AES 密钥字节数组encryptedAesKey 。最后,同样将加密后的 AES 密钥转换为 Base64 编码的字符串encryptedAesKeyText ,方便在网络中传输 。

(四)发送加密后的数据

在完成数据的加密和 AES 密钥的加密后,我们将加密后的数据和加密后的 AES 密钥一起发送给接收方 。接收方在收到数据后,需要按照相反的步骤进行解密 。

  1. 发送方操作

将 AES 加密后的密文encryptedText和 RSA 加密后的 AES 密钥encryptedAesKeyText一起发送给接收方 。可以通过网络请求(如 HTTP 请求)将这些数据发送到目标服务器 。例如,使用HttpClient发送 POST 请求:

using (HttpClient client = new HttpClient())
{
    var content = new MultipartFormDataContent();
    content.Add(new StringContent(encryptedText), "encryptedData");
    content.Add(new StringContent(encryptedAesKeyText), "encryptedAesKey");
    HttpResponseMessage response = client.PostAsync("https://example.com/api/receive", content).Result;
    if (response.IsSuccessStatusCode)
    {
        Console.WriteLine("数据发送成功");
    }
    else
    {
        Console.WriteLine("数据发送失败: " + response.StatusCode);
    }
}

在这段代码中,创建了一个HttpClient对象用于发送 HTTP 请求 。使用MultipartFormDataContent将加密后的密文和加密后的 AES 密钥作为表单数据添加到请求中 ,分别命名为encryptedData和encryptedAesKey 。然后通过client.PostAsync方法发送 POST 请求到指定的 API 地址https://example.com/api/receive ,并根据响应状态码判断数据是否发送成功 。

  1. 接收方操作

接收方在收到请求后,首先使用自己的 RSA 私钥解密得到 AES 密钥 ,然后再用 AES 密钥解密密文数据 。以下是接收方的解密代码示例:

// 假设已经获取到RSA私钥参数
RSAParameters privateKey = GetPrivateKey(); // 假设这是获取私钥的方法
// 从请求中获取加密后的AES密钥和密文数据
string receivedEncryptedAesKey = Request.Form["encryptedAesKey"];
string receivedEncryptedText = Request.Form["encryptedData"];
// 将Base64编码的字符串转换回字节数组
byte[] receivedEncryptedAesKeyBytes = Convert.FromBase64String(receivedEncryptedAesKey);
byte[] receivedEncryptedTextBytes = Convert.FromBase64String(receivedEncryptedText);
// 使用RSA私钥解密AES密钥
byte[] decryptedAesKey = DecryptUsingRSA(receivedEncryptedAesKeyBytes, privateKey);
// 使用AES密钥解密密文数据
string decryptedText = DecryptStringFromBytes_Aes(receivedEncryptedTextBytes, decryptedAesKey, GetIV()); // 假设这是获取IV的方法
Console.WriteLine("解密后的明文: " + decryptedText);

在这段代码中,首先通过GetPrivateKey()方法获取 RSA 私钥参数(假设已经有获取私钥的方法) 。然后从 HTTP 请求的表单数据中获取加密后的 AES 密钥receivedEncryptedAesKey和密文数据receivedEncryptedText 。将这两个 Base64 编码的字符串转换回字节数组 。接着调用DecryptUsingRSA方法(假设已经实现该方法用于 RSA 解密),使用 RSA 私钥对加密后的 AES 密钥进行解密,得到原始的 AES 密钥decryptedAesKey 。最后,调用DecryptStringFromBytes_Aes方法(假设已经实现该方法用于 AES 解密),使用解密得到的 AES 密钥和获取到的初始化向量(这里假设通过GetIV()方法获取)解密密文数据,得到原始的明文decryptedText 。通过这样的流程,实现了数据的安全传输和解密 。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/965449.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

langchain教程-7.Embedding/文本向量化

前言 该系列教程的代码: https://github.com/shar-pen/Langchain-MiniTutorial 我主要参考 langchain 官方教程, 有选择性的记录了一下学习内容 这是教程清单 1.初试langchain2.prompt3.OutputParser/输出解析4.model/vllm模型部署和langchain调用5.DocumentLoader/多种文档…

Mac下使用brew安装go 以及遇到的问题

首先按照网上找到的命令进行安装 brew install go 打开终端输入go version,查看安装的go版本 go version 配置环境变量 查看go的环境变量配置: go env 事实上安装好后的go已经可以使用了。 在home/go下新建src/hello目录,在该目录中新建…

Ubuntu部署Deepseek-R1模型(8b)

安装ubuntu系统 本机电脑系统ubuntu-20.04 #升级软件 sudo apt-get update#安装curl sudo apt-get install curl通过以上两条指令,完成了curl命令的安装。 安装ollama 打开Ollama官网 选择Linux, 给出如上图方框所示的一条指令 curl -fsSL https:…

【ROS视频推流】使用web_video_server完成视频推流

🚀 本文简要介绍一下使用web_video_server功能包完成实时视频推流的方法。 假设有A,B两个设备,它们之间可以ping通。我们需要将A设备上的实时摄像头图像推流并在B设备的浏览器上显示。 🌔01准备工作 # A设备 # 下载视频推流功能包 #&#xff…

[LVGL] 在VC_MFC中移植LVGL

前言: 0. 在MFC中开发LVGL的优点是可以用多个Window界面做辅助扩展 1.本文基于VC2022-MFC单文档框架移植lvgl8 2. gitee上下载lvgl8.3 源码,并将其文件夹改名为lvgl lvgl: LVGL 是一个开源图形库,提供您创建具有易于使用的图形元素、漂亮…

Java----线程池

什么是线程池呢,先举一个情景: 一个火锅店开业了,早上人比较少,大家进店后不需要预约,直接付款在店里的桌子上吃饭,慢慢的人多了,店里的桌子不够用了,没座位的人可以先预约&#xf…

安卓开发,底部导航栏

1、创建导航栏图标 使用系统自带的矢量图库文件,鼠标右键点击res->New->Vector Asset 修改 Name , Clip art 和 Color 再创建一个 同样的方法再创建四个按钮 2、添加百分比布局依赖 app\build.gradle.kts 中添加百分比布局依赖,并点击Sync Now …

每日Attention学习22——Inverted Residual RWKV

模块出处 [arXiv 25] [link] [code] RWKV-UNet: Improving UNet with Long-Range Cooperation for Effective Medical Image Segmentation 模块名称 Inverted Residual RWKV (IR-RWKV) 模块作用 用于vision的RWKV结构 模块结构 模块代码 注:cpp扩展请参考作者原…

Git--使用教程

Git的框架讲解 Git 是一个分布式版本控制系统,其架构设计旨在高效地管理代码版本,支持分布式协作,并确保数据的完整性和安全性。 Git 的核心组件: 工作区(Working Directory): 工作区是你在本…

智慧停车系统:不同规模停车场的应用差异与YunCitys解决方案

在智慧停车领域,不同规模停车场因自身特点,对智慧停车系统的需求和应用效果存在显著差异。云创智城凭借丰富的经验和先进的技术,为各类规模停车场打造了贴合需求的智慧停车系统,下面为您详细剖析。 小型停车场:精准高…

snort的学习记录

一、what is snort?什么是snort? Snort 是一款开源的 网络入侵检测系统(NIDS) 和 网络入侵防御系统(NIPS),能够实时监控网络流量,检测恶意行为(如端口扫描、SQL注入、DDoS攻击等&a…

PHP-trim

[题目信息]: 题目名称题目难度PHP-trim1 [题目考点]: trim() 函数移除字符串两侧的空白字符或其他预定义字符。[Flag格式]: SangFor{dl9hFiITmhQNAJysCgigAskyCZ6kQaDc}[环境部署]: docker-compose.yml文件或者docker tar原始文件。 ht…

maven如何不把依赖的jar打包到同一个jar?

spring boot项目打jar包部署: 经过以下步骤, 最终会形成maven依赖的多个jar(包括lib下添加的)、 我们编写的程序代码打成一个jar,将程序jar与 依赖jar分开,便于管理: success: 最终…

【ArcGIS Pro 简介1】

ArcGIS Pro 是由 Esri (Environmental Systems Research Institute)公司开发的下一代桌面地理信息系统(GIS)软件,是传统 ArcMap 的现代化替代产品。它结合了强大的空间分析能力、直观的用户界面和先进的三维可视化技术…

DeepSeek 部署过程中的问题

文章目录 DeepSeek 部署过程中的问题一、部署扩展:docker 部署 DS1.1 部署1.2 可视化 二、问题三、GPU 设置3.1 ollama GPU 的支持情况3.2 更新 GPU 驱动3.3 安装 cuda3.4 下载 cuDNN3.5 配置环境变量 四、测试 DeepSeek 部署过程中的问题 Windows 中 利用 ollama 来…

Elasticsearch基本使用详解

文章目录 Elasticsearch基本使用详解一、引言二、环境搭建1、安装 Elasticsearch2、安装 Kibana(可选) 三、索引操作1、创建索引2、查看索引3、删除索引 四、数据操作1、插入数据2、查询数据(1)简单查询(2)…

点(线)集最小包围外轮廓效果赏析

“ 图像、点集、线集合最小外轮廓计算应用较为广泛,如抠图、神奇选择、LOD、碰撞检查等领域,提高场景效率” 1.前言 作者基于递归迭代求解实现点集的最小外轮廓计算,在CGLib库中实现,已集成于CGViewer,可联系作者试用&…

IOPS与吞吐量、读写块大小及延迟之间的关系

IOPS(每秒输入/输出操作次数)、吞吐量、读写块大小及延迟是衡量存储系统性能的四个关键指标,它们之间存在密切的关系。以下从多个方面详细说明这些指标之间的关系: 1. IOPS与吞吐量的关系 公式关系:吞吐量&#xff0…

FPGA与ASIC:到底选哪个好?

不少人想转行FPGA,但在ASIC和FPGA之间犹豫不决。要做出选择,首先需要清楚两者的区别和各自特点。 FPGA (Field Programmable Gate Array) 是一种现场可编程门阵列芯片,本质上它是一种半定制的芯片,可以根据需要重新编程&#xff…

LLMs之data:synthetic-data-generator的简介、安装和使用方法、案例应用之详细攻略

LLMs之data:synthetic-data-generator的简介、安装和使用方法、案例应用之详细攻略 目录 synthetic-data-generator的简介 1、核心功能和优势 2、特点 synthetic-data-generator的安装和使用方法 1、安装 pip安装 安装依赖项 运行应用 2、使用方法 快速入…