Cocos2dx c++到安卓Java/kt实现Rsa、Aes、Base64加密解密

2024-01-04 10:30

本文主要是介绍Cocos2dx c++到安卓Java/kt实现Rsa、Aes、Base64加密解密,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1977年,三位数学家Rivest、Shamir 和 Adleman 设计了一种算法,可以实现非对称加密。这种算法用他们三个人的名字命名,叫做RSA算法。从那时直到现在,RSA算法一直是最广为使用的"非对称加密算法"。毫不夸张地说,只要有计算机网络的地方,就有RSA算法。这种算法非常可靠,密钥越长,它就越难破解。根据已经披露的文献,目前被破解的最长RSA密钥是232个十进制位,也就是768个二进制位,因此可以认为,1024位的RSA密钥基本安全,2048位的密钥极其安全,当然量子计算机除外。

以下下方法实现了非对称加密算法Rsa c++的加解密和java的加解密,可以在jni过程中提高安全性,代码中的公钥和私钥建议从文件获取,不以明文的方式放在本地,方法不予赘述

//
//  RsaUtil.cpp
//
//  C++加密和解密过程
//
//  Created by Versprechen on 2023/9/14.
//#include <iostream>
#include "RsaUtil.hpp"#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID
#include "openssl/rsa.h"
#include "openssl/pem.h"
#include "openssl/err.h"
#include "openssl/evp.h"
#endif#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include "JniTools.h"
#include "Base64Util.hpp"RsaUtil::RsaUtil()
{
}RsaUtil::~RsaUtil()
{
}
RsaUtil *RsaUtil::m_instance = nullptr;
RsaUtil *RsaUtil::getInstance()
{if (nullptr == m_instance){m_instance = new RsaUtil;}return m_instance;
}void RsaUtil::release()
{if (m_instance){}CC_SAFE_DELETE(m_instance);
}void RsaUtil::initKey()
{m_privateKey = Base64Util::getInstance()->base64_decode(m_privateKey);m_publicKey = Base64Util::getInstance()->base64_decode(m_publicKey);
}/**  非对称加密算法Rsa工具类的使用:*  双方加密解密遵循 PKCS#8 公钥和私钥格式*  1、createKey:创建公钥和私钥*     加密前需要保证公钥格式以"-----BEGIN RSA PUBLIC KEY-----"开头,以"-----END RSA PUBLIC KEY-----"结尾*     私钥以"-----BEGIN RSA PRIVATE KEY-----"开头*  2、RSAEncrypt加密,加密之后会使用base64_encode加密,防止传输错误*  3、RSADecrypt解密,解密流程是先把传回来的字符串base64_decode解密,再走加密算法**  注意:使用Jni交互的数据需要先使用base64加密/解密*/// 创建rsa的私钥和公钥,覆盖成员变量
void RsaUtil::createKeys()
{
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROIDRSA *rsa = RSA_new();BIGNUM *e = BN_new();BN_set_word(e, RSA_F4);RSA_generate_key_ex(rsa, 2048, e, NULL);BIO *publicBIO = BIO_new(BIO_s_mem());PEM_write_bio_RSA_PUBKEY(publicBIO, rsa);BIO *privateBIO = BIO_new(BIO_s_mem());EVP_PKEY *pkey = EVP_PKEY_new();EVP_PKEY_assign_RSA(pkey, rsa); // rsa will be freed when pkey is freedPEM_write_bio_PKCS8PrivateKey(privateBIO, pkey, NULL, NULL, 0, NULL, NULL);char *publicKey;long publicKeyLength = BIO_get_mem_data(publicBIO, &publicKey);char *privateKey;long privateKeyLength = BIO_get_mem_data(privateBIO, &privateKey);m_publicKey = std::string(publicKey, publicKeyLength);m_privateKey = std::string(privateKey, privateKeyLength);m_publicKey = removeKeyHeaderAndFooter(m_publicKey);m_privateKey = removeKeyHeaderAndFooter(m_privateKey);NXLOG("公钥:%s", m_publicKey.c_str());NXLOG("私钥:%s", m_privateKey.c_str());BN_free(e);BIO_free(publicBIO);BIO_free(privateBIO);EVP_PKEY_free(pkey);
#endif
}// 由于java需要去掉头尾标记符,生成之后需要去掉给他们
std::string RsaUtil::removeKeyHeaderAndFooter(const std::string &key)
{std::string result = key;std::string header1 = "-----BEGIN RSA PUBLIC KEY-----";std::string header2 = "-----BEGIN RSA PRIVATE KEY-----";std::string footer = "-----END RSA PUBLIC KEY-----";size_t pos;while ((pos = result.find(header1)) != std::string::npos)result.erase(pos, header1.length());while ((pos = result.find(header2)) != std::string::npos)result.erase(pos, header2.length());while ((pos = result.find(footer)) != std::string::npos)result.erase(pos, footer.length());return result;
}// 加密函数
std::string RsaUtil::RSAEncrypt(std::string plainText)
{
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROIDBIO *keybio = BIO_new_mem_buf(m_publicKey.c_str(), -1);RSA *rsa = PEM_read_bio_RSA_PUBKEY(keybio, NULL, NULL, NULL); // 修改这里std::string encryptedData(RSA_size(rsa), '\0');int encryptedDataLength = RSA_public_encrypt(plainText.size(), reinterpret_cast<const unsigned char *>(plainText.c_str()), reinterpret_cast<unsigned char *>(&encryptedData[0]), rsa, RSA_PKCS1_PADDING);BIO_free(keybio);RSA_free(rsa);auto output = encryptedData.substr(0, encryptedDataLength);return Base64Util::getInstance()->base64_encode(reinterpret_cast<const unsigned char *>(output.c_str()), output.length());
#endif
}// 解密
std::string RsaUtil::RSADecrypt(std::string cipherText)
{
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROIDstring decode = Base64Util::getInstance()->base64_decode(cipherText);BIO *keybio = BIO_new_mem_buf(m_privateKey.c_str(), -1);RSA *rsa = PEM_read_bio_RSAPrivateKey(keybio, NULL, NULL, NULL);std::string data(RSA_size(rsa), '\0');int dataLength = RSA_private_decrypt(decode.size(), reinterpret_cast<const unsigned char *>(decode.c_str()), reinterpret_cast<unsigned char *>(&data[0]), rsa, RSA_PKCS1_PADDING);BIO_free(keybio);RSA_free(rsa);return data.substr(0, dataLength);
#endif
}

​​​​​​​//
//  RsaUtil.hpp
//
//  Created by Versprechen on 2023/9/14.
//#ifndef RsaUtil_hpp
#define RsaUtil_hpp#include "cocos2d.h"
#include "GameState.h"
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID
#include "openssl/rsa.h"
#include "openssl/pem.h"
#include "openssl/err.h"
#include "openssl/evp.h"
#endif
USING_NS_CC;class RsaUtil {
public:RsaUtil();~RsaUtil();static RsaUtil *getInstance();public:static RsaUtil *m_instance;static void release();void createKeys();std::string removeKeyHeaderAndFooter(const std::string &key);std::string RSAEncrypt(std::string plainText);std::string RSADecrypt(std::string cipherText);void initKey();//使用Rsa加密解密的公钥和私钥std::string m_privateKey = "你经过Base64之后的公钥";std::string m_publicKey = "你经过Base64之后的私钥";
};
#endif /* RsaUtil_hpp */

=====java/kt版本加解密====

package org.cocos2dx.cpp.adFrameimport java.security.KeyFactory
import java.security.KeyPair
import java.security.KeyPairGenerator
import java.security.NoSuchAlgorithmException
import java.security.interfaces.RSAPrivateKey
import java.security.interfaces.RSAPublicKey
import java.security.spec.PKCS8EncodedKeySpec
import java.security.spec.X509EncodedKeySpec
import javax.crypto.Cipherobject RSAUtils {private const val TRANSFORMATION = "RSA/ECB/PKCS1Padding"/** 生成密钥对,即公钥和私钥。key长度是512-2048,一般为1024 本文使用2048长度的key */@Throws(NoSuchAlgorithmException::class)fun generateRSAKeyPair(keyLength: Int): KeyPair? {val kpg = KeyPairGenerator.getInstance("RSA")kpg.initialize(keyLength)return kpg.genKeyPair()}/** 获取公钥 */fun getPublicKey(keyPair: KeyPair): ByteArray? {val rsaPublicKey = keyPair.public as RSAPublicKeyreturn rsaPublicKey.encoded}/** 获取私钥 */fun getPrivateKey(keyPair: KeyPair): ByteArray? {val rsaPrivateKey = keyPair.private as RSAPrivateKeyreturn rsaPrivateKey.encoded}/*** 使用公钥加密* @param content 需要加密的密文(String类型)* @param publicKeyString 公钥(String类型)* @return base64数据* @throws Exception*/@Throws(Exception::class)fun encryptByPublicKey(content: String): String? {val publicKeyString = "你经过base64之后的公钥建议从文件获取";val publicKey: ByteArray = Base64Utils.decode2byte(publicKeyString.replace("\n", "").toByteArray())// 得到公钥对象val keySpec = X509EncodedKeySpec(publicKey)val keyFactory = KeyFactory.getInstance("RSA")val pubKey = keyFactory.generatePublic(keySpec)// 加密数据val cp = Cipher.getInstance(TRANSFORMATION)cp.init(Cipher.ENCRYPT_MODE, pubKey)val byteArray = cp.doFinal(content.toByteArray())return if (byteArray != null) Base64Utils.encode2Str(byteArray) else null}/*** 使用私钥解密* @param content content 需要加密的密文* @param privateKeyString  私钥* @return String数据所以需要fastjson转object* @throws Exception*/@Throws(java.lang.Exception::class)fun decryptByPrivateKey(content: String?): String? {val privateKeyString = "你经过base64之后的私钥建议从文件获取";val privateKey = Base64Utils.decode2byte(privateKeyString.toByteArray())val encrypt: ByteArray = Base64Utils.decode2byte(content)// 得到私钥对象val keySpec = PKCS8EncodedKeySpec(privateKey)val kf = KeyFactory.getInstance("RSA")val keyPrivate = kf.generatePrivate(keySpec)// 解密数据val cp = Cipher.getInstance(TRANSFORMATION)cp.init(Cipher.DECRYPT_MODE, keyPrivate)val byteArray = cp.doFinal(encrypt)return if (byteArray != null) String(byteArray) else null}
}

=====================================================================

AES的全称是Advanced Encryption Standard,意思是高级加密标准。它的出现主要是为了取代DES加密算法的,因为我们都知道DES算法的密钥长度是56Bit,因此算法的理论安全强度是2的56次方。但二十世纪中后期正是计算机飞速发展的阶段,元器件制造工艺的进步使得计算机的处理能力越来越强,虽然出现了3DES的加密方法,但由于它的加密时间是DES算法的3倍多,64Bit的分组大小相对较小,所以还是不能满足人们对安全性的要求。于是1997年1月2号,美国国家标准技术研究所宣布希望征集高级加密标准,用以取代DES。AES也得到了全世界很多密码工作者的响应,先后有很多人提交了自己设计的算法。最终有5个候选算法进入最后一轮:Rijndael,Serpent,Twofish,RC6和MARS。最终经过安全性分析、软硬件性能评估等严格的步骤,Rijndael算法获胜。

下面是对应的对称的Aes加密算法,流程较为简单,但是加密解密效率较高

#include "AesUtil.hpp"
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID
#include "openssl/aes.h"
#include "openssl/rsa.h"
#include "openssl/pem.h"
#include "openssl/err.h"
#include "openssl/evp.h"
#include "Base64Util.hpp"
#endif#define AES_KEY_LENGTH 16  // aes-128AesUtil::AesUtil()
{
}AesUtil::~AesUtil()
{
}
AesUtil *AesUtil::m_instance = nullptr;
AesUtil *AesUtil::getInstance()
{if (nullptr == m_instance){m_instance = new AesUtil;}return m_instance;
}void AesUtil::release()
{if (m_instance){}CC_SAFE_DELETE(m_instance);
}void AesUtil::printhexDump(const char *buffer, size_t len)
{if (buffer == NULL || len <= 0){return;}CCLOG("0x%x: (len=%d)[", buffer, len);for (size_t i = 0; i < len; i++){CCLOG("%.2X ", (unsigned char)buffer[i]);}CCLOG("]\n");
}
string AesUtil::aes_decrypt(string encData)
{
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROIDstring decode = Base64Util::getInstance()->base64_decode(encData);string code = FileUtils::getInstance()->getStringFromFile("com.gamezone.adventurequest.amr");if (code.length() != 16) {// Handle error: code string length is not 32}unsigned char key[17];memcpy(key, code.c_str(), 16);key[16] = '\0';unsigned char decData[1024];int decLen = 0;int outlen = 0;EVP_CIPHER_CTX *ctx2;ctx2 = EVP_CIPHER_CTX_new();EVP_CipherInit_ex(ctx2, EVP_aes_128_ecb(), NULL, key, NULL, AES_DECRYPT);EVP_CipherUpdate(ctx2, decData, &outlen, (const unsigned char*)(decode.c_str()), decode.length());decLen = outlen;if(EVP_CipherFinal_ex(ctx2, decData + outlen, &outlen)){decLen += outlen;}EVP_CIPHER_CTX_free(ctx2);decData[decLen] = '\0';CCLOG("decrypt: %s\n", decData);return StringUtils::format("%s",decData);
#endif
}string AesUtil::aes_encrypt(string rawData)
{
#if CC_TARGET_PLATFORM == CC_PLATFORM_ANDROIDstring code = FileUtils::getInstance()->getStringFromFile("");//获取你的秘钥if (code.length() != 16) {// Handle error: code string length is not 32}unsigned char key[17];memcpy(key, code.c_str(), 16);key[16] = '\0';int rawDataLen = strlen(rawData.c_str());unsigned char encData[1024];int encLen = 0;int outLen = 0;EVP_CIPHER_CTX *ctx;ctx = EVP_CIPHER_CTX_new();EVP_CipherInit_ex(ctx, EVP_aes_128_ecb(), NULL, key, NULL, AES_ENCRYPT);EVP_CipherUpdate(ctx, encData, &outLen, (const unsigned char*)(rawData.c_str()), rawDataLen);encLen = outLen;if(EVP_CipherFinal_ex(ctx, encData + outLen, &outLen)){encLen += outLen;}EVP_CIPHER_CTX_free(ctx);printhexDump(reinterpret_cast<const char *>(encData), encLen);return Base64Util::getInstance()->base64_encode(encData,encLen);
#endif
}
//
//  AesUtil.hpp
//
//  Created by Versprechen on 2023/9/19.
//#ifndef AesUtil_hpp
#define AesUtil_hpp#include <stdio.h>
#include <string>
#include "cocos2d.h"using namespace std;
using namespace cocos2d;class AesUtil {
public:AesUtil();~AesUtil();static AesUtil *getInstance();public:static AesUtil *m_instance;static void release();string aes_decrypt(string encData);string aes_encrypt(string rawData);void printhexDump(const char *buffer, size_t len);};
#endif /* AesUtil_hpp */

=====java/kt版本加解密====

package org.cocos2dx.cpp.adFrameimport android.content.Context
import java.io.BufferedReader
import java.io.InputStreamReader
import javax.crypto.Cipher
import javax.crypto.spec.SecretKeySpecobject AESUtils {lateinit var aesKey: Stringfun initAesKey(context: Context) {val stringBuilder = StringBuilder()try {val assetManager = context.assetsval bufferedReader = BufferedReader(InputStreamReader(//获取你的秘钥))var line: String?while (bufferedReader.readLine().also { line = it } != null) {stringBuilder.append(line)}} catch (e: Exception) {e.printStackTrace()}aesKey = stringBuilder.toString()}fun aseDecrypt(str: String): String {val key = SecretKeySpec(aesKey.toByteArray(), "AES")val cipher = Cipher.getInstance("AES/ECB/PKCS5Padding")cipher.init(Cipher.DECRYPT_MODE, key)return String(cipher.doFinal(Base64Utils.decode2byte(str.toByteArray())))}fun aesEncrypt(str: String): String {val key = SecretKeySpec(aesKey.toByteArray(), "AES")val cipher = Cipher.getInstance("AES/ECB/PKCS5Padding")cipher.init(Cipher.ENCRYPT_MODE, key)return Base64Utils.encode2Str(cipher.doFinal(str.toByteArray()))}
}

AES加密算法正常自己c++加密解密可以统一方法,但是和java、js互相加解密需要遵循以下原则,否则会解密失败,以下截图来自csdn:CryptoJS C/C++ openssl Java AES_256_ecb 加密解密互通_java aes-256-ecb 加密-CSDN博客 

在加密过程中,输出的可能是十六进制,通常先做base64再传输,以下是c++版本的Base64算法

//
//  Base64Util.cpp
//  game-blockpuzzle-mobile
//
//  Created by Versprechen on 2023/9/19.
//#include "Base64Util.hpp"
Base64Util::Base64Util()
{
}Base64Util::~Base64Util()
{
}
Base64Util *Base64Util::m_instance = nullptr;
Base64Util *Base64Util::getInstance()
{if (nullptr == m_instance){m_instance = new Base64Util;}return m_instance;
}void Base64Util::release()
{if (m_instance){}CC_SAFE_DELETE(m_instance);
}
static const std::string base64_chars ="ABCDEFGHIJKLMNOPQRSTUVWXYZ""abcdefghijklmnopqrstuvwxyz""0123456789+/";
std::string Base64Util::base64_encode(unsigned char const *bytes_to_encode, unsigned int in_len)
{std::string ret;int i = 0;int j = 0;unsigned char char_array_3[3];unsigned char char_array_4[4];while (in_len--){char_array_3[i++] = *(bytes_to_encode++);if (i == 3){char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);char_array_4[3] = char_array_3[2] & 0x3f;for (i = 0; (i < 4); i++)ret += base64_chars[char_array_4[i]];i = 0;}}if (i){for (j = i; j < 3; j++)char_array_3[j] = '\0';char_array_4[0] = (char_array_3[0] & 0xfc) >> 2;char_array_4[1] = ((char_array_3[0] & 0x03) << 4) + ((char_array_3[1] & 0xf0) >> 4);char_array_4[2] = ((char_array_3[1] & 0x0f) << 2) + ((char_array_3[2] & 0xc0) >> 6);char_array_4[3] = char_array_3[2] & 0x3f;for (j = 0; (j < i + 1); j++)ret += base64_chars[char_array_4[j]];while ((i++ < 3))ret += '=';}return ret;
}std::string Base64Util::base64_decode(std::string const &encoded_string)
{int in_len = encoded_string.size();int i = 0;int j = 0;int in_ = 0;unsigned char char_array_4[4], char_array_3[3];std::string ret;while (in_len-- && (encoded_string[in_] != '=') && is_base64(encoded_string[in_])){char_array_4[i++] = encoded_string[in_];in_++;if (i == 4){for (i = 0; i < 4; i++)char_array_4[i] = base64_chars.find(char_array_4[i]);char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];for (i = 0; (i < 3); i++)ret += char_array_3[i];i = 0;}}if (i){for (j = i; j < 4; j++)char_array_4[j] = 0;for (j = 0; j < 4; j++)char_array_4[j] = base64_chars.find(char_array_4[j]);char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];for (j = 0; (j < i - 1); j++)ret += char_array_3[j];}return ret;
}inline bool Base64Util::is_base64(unsigned char c)
{return (isalnum(c) || (c == '+') || (c == '/'));
}
//
//  Base64Util.hpp
//  game-blockpuzzle-mobile
//
//  Created by Versprechen on 2023/9/19.
//#ifndef Base64Util_hpp
#define Base64Util_hpp#include <stdio.h>
#include <string>
#include "cocos2d.h"using namespace std;class Base64Util {
public:Base64Util();~Base64Util();static Base64Util *getInstance();public:static Base64Util *m_instance;static void release();std::string base64_encode(unsigned char const *bytes_to_encode, unsigned int in_len);std::string base64_decode(std::string const &encoded_string);inline bool is_base64(unsigned char c);};
#endif /* Base64Util_hpp */

这篇关于Cocos2dx c++到安卓Java/kt实现Rsa、Aes、Base64加密解密的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/569032

相关文章

Oracle查询优化之高效实现仅查询前10条记录的方法与实践

《Oracle查询优化之高效实现仅查询前10条记录的方法与实践》:本文主要介绍Oracle查询优化之高效实现仅查询前10条记录的相关资料,包括使用ROWNUM、ROW_NUMBER()函数、FET... 目录1. 使用 ROWNUM 查询2. 使用 ROW_NUMBER() 函数3. 使用 FETCH FI

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用

Java实现Excel与HTML互转

《Java实现Excel与HTML互转》Excel是一种电子表格格式,而HTM则是一种用于创建网页的标记语言,虽然两者在用途上存在差异,但有时我们需要将数据从一种格式转换为另一种格式,下面我们就来看看... Excel是一种电子表格格式,广泛用于数据处理和分析,而HTM则是一种用于创建网页的标记语言。虽然两

java图像识别工具类(ImageRecognitionUtils)使用实例详解

《java图像识别工具类(ImageRecognitionUtils)使用实例详解》:本文主要介绍如何在Java中使用OpenCV进行图像识别,包括图像加载、预处理、分类、人脸检测和特征提取等步骤... 目录前言1. 图像识别的背景与作用2. 设计目标3. 项目依赖4. 设计与实现 ImageRecogni

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Java访问修饰符public、private、protected及默认访问权限详解

《Java访问修饰符public、private、protected及默认访问权限详解》:本文主要介绍Java访问修饰符public、private、protected及默认访问权限的相关资料,每... 目录前言1. public 访问修饰符特点:示例:适用场景:2. private 访问修饰符特点:示例:

详解Java如何向http/https接口发出请求

《详解Java如何向http/https接口发出请求》这篇文章主要为大家详细介绍了Java如何实现向http/https接口发出请求,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 用Java发送web请求所用到的包都在java.net下,在具体使用时可以用如下代码,你可以把它封装成一

使用Python实现在Word中添加或删除超链接

《使用Python实现在Word中添加或删除超链接》在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能,本文将为大家介绍一下Python如何实现在Word中添加或... 在Word文档中,超链接是一种将文本或图像连接到其他文档、网页或同一文档中不同部分的功能。通过添加超

windos server2022里的DFS配置的实现

《windosserver2022里的DFS配置的实现》DFS是WindowsServer操作系统提供的一种功能,用于在多台服务器上集中管理共享文件夹和文件的分布式存储解决方案,本文就来介绍一下wi... 目录什么是DFS?优势:应用场景:DFS配置步骤什么是DFS?DFS指的是分布式文件系统(Distr

NFS实现多服务器文件的共享的方法步骤

《NFS实现多服务器文件的共享的方法步骤》NFS允许网络中的计算机之间共享资源,客户端可以透明地读写远端NFS服务器上的文件,本文就来介绍一下NFS实现多服务器文件的共享的方法步骤,感兴趣的可以了解一... 目录一、简介二、部署1、准备1、服务端和客户端:安装nfs-utils2、服务端:创建共享目录3、服