以太坊代币标准解读及相关Dapp的搭建

news2025/1/24 2:18:49

文章目录

  • 什么是以太坊代币标准
    • 1、什么是以太坊
    • 2、以太坊代币标准
  • 同质化代币 Dapp 搭建
    • 1、MetaMask 的安装
    • 2、Ganache 的安装
    • 3、实现 ERC-20 代币协议
    • 4、前端页面的编写
    • 5、部署流程及操作演示

什么是以太坊代币标准

1、什么是以太坊

以太坊(Ethereum)是一个去中心化的开源的有智能合约功能的公共区块链平台。以太币(ETH 或 Ξ)是以太坊的原生加密货币。截至2021年12月,以太币是市值第二高的加密货币,仅次于比特币。以太坊是使用最多的区块链。
以太坊的概念首次在2013至2014年间由程序员维塔利克·布特林受比特币启发后提出,大意为“下一代加密货币与去中心化应用平台”,在2014年通过ICO众筹得以开始发展。以太坊亦被称为“第二代的区块链平台”,仅次于比特币。(维基百科)

以太坊官网
在这里插入图片描述
以太坊作为区块链技术的一个开源平台,允许任何人在区块链上构建和使用去中心化的应用程序和加密货币。

以太坊的主要特点包括:

  • 智能合约 - 以太坊提供了智能合约的功能,可以在区块链上实现自动化的数字合约。这使得在区块链上进行可信交易成为可能。
  • 去中心化应用(Dapp) - 在以太坊上可以构建各种去中心化应用,以避免单一实体控制的风险。这些DApp可以实现很多创新的功能。
  • 加密货币 - 以太坊有自己的加密货币以太币,可以在以太坊区块链上进行交易。以太币是仅次于比特币的第二大数字货币。
  • 挖矿 - 以太坊也采用了工作证明机制,允许通过挖矿来维护网络安全和达成共识。挖矿可以获得以太币奖励。
  • 联盟链 - 以太坊支持创建私有或联盟链,在许可的网络内执行交易。这扩大了以太坊的应用范围。
    在这里插入图片描述

2、以太坊代币标准

以太坊代币标准是指在以太坊上发行代币需要遵循的一系列技术规范,以方便以太坊开发者在以太坊区块链上发行和交易加密数字资产,常见的代币标准有 ERC-20、ERC-721、ERC-777、ERC-1155、ERC-4626等。代币标准的实施有助于在项目的不同实现中保持统一的兼容性(例如以太坊客户和钱包),并确保智能合约和 dapps 仍保持兼容。(参考以太坊文档)

在这里插入图片描述
(图片来自以太坊官方网站)

同质化代币 Dapp 搭建

本文介绍基于 MetaMask + Ganache + Web3.js同质化代币 Dapp 搭建。

1、MetaMask 的安装

MetaMask 官网

MetaMask(小狐狸钱包),是用于与以太坊区块链进行交互的软件加密货币钱包。它可以通过浏览器扩展程序或移动应用程序让用户访问其以太坊钱包,与去中心化应用进行交互。
MetaMask 由 ConsenSys Software Inc. 开发运营,主要专注于以太坊为基础的工具及基础设施。(维基百科)

进入 MetaMask 官网直接点击下载 MetaMask 的浏览器扩展,第一次使用需要自己注册账号(注意保存好私钥)。
在这里插入图片描述
在这里插入图片描述

2、Ganache 的安装

Ganache 官网

Ganache 是一个能够在本地快速部署以太坊私有链的模拟应用。能够帮助开发者快速、简单、安全地测试和调试他们的去中心化应用(DApp),并在发布到生产环境之前预览其性能和功能。通过 Ganache,开发者可以快速查看所有账户的当前状态,包括他们的地址、私钥、交易和余额。

同样的,可以在 Ganache 官网点击下载 windows 版本的 Ganache ,操作简单,开箱即用。
在这里插入图片描述
在这里插入图片描述

3、实现 ERC-20 代币协议

ERC-20 代币标准

ERC-20(以太坊意见征求 20)由 Fabian Vogelsteller 提出于 2015 年 11 月。这是一个能实现智能合约中代币的应用程序接口标准。

ERC-20 的功能示例包括:

  • 将代币从一个帐户转到另一个帐户
  • 获取帐户的当前代币余额
  • 获取网络上可用代币的总供应量
  • 批准一个帐户中一定的代币金额由第三方帐户使用

如果智能合约实施了下列方法和事件,它可以被称为 ERC-20 代币合约,一旦部署,将负责跟踪以太坊上创建的代币。(以太坊官方文档)

在这里我们用 solidity 实现 ERC-20 代币协议:

// SPDX-License-Identifier: MIT
pragma solidity ^0.8.4;

//  ERC-20 代币合约接口 
interface IERC20 {
    // 转账事件 
    event Transfer(address indexed _from, address indexed _to, uint256 _value);
    // 授权事件 
    event Approval(address indexed _owner, address indexed _spender, uint256 _value);

    // 代币名称
    function name() external view returns (string memory);
    // 代币符号 
    function symbol() external view returns (string memory);
    // 小数位数
    function decimals() external view returns (uint8);
    // 总供给量 
    function totalSupply() external view returns (uint256);
    // 余额查询
    function balanceOf(address _owner) external view returns (uint256 balance);
    // 转账
    function transfer(address _to, uint256 _value) external returns (bool success);
    // 从 from 账户转账
    function transferFrom(address _from, address _to, uint256 _value) external returns (bool success);
    // 授权转账
    function approve(address _spender, uint256 _value) external returns (bool success);
    // 查询授权余额 
    function allowance(address _owner, address _spender) external view returns (uint256 remaining);
}
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.4;

import "./IERC-20.sol";

// ERC-20 代币实现
contract ERC20Token is IERC20 { 
    event Forge(address indexed account, uint256 amount);       // 锻造代币事件
    event Destroy(address indexed account, uint amount);        // 销毁代币事件 
    
    mapping(address => uint256) public balanceOf; 
    mapping(address => mapping(address => uint256))  public  allowance;

    uint256 public totalSupply;     
    string public name;             
    string public symbol;           
    uint8 public decimals;         

    // 构造函数 
    constructor(string memory name_, string memory symbol_, uint8 decimals_) {
        name = name_;                  
        symbol = symbol_; 
        decimals = decimals_; 
    }

    // 锻造代币 
    function forgeToken(address account, uint amount) external {
        require(amount > 0, "ERC-20: amount less than or equal to 0.");
        require(account != address(0), "ERC-20: forge to zero address.");

        balanceOf[account] += amount; 
        totalSupply += amount; 

        emit Forge(account, amount);
    }
                
    // 销毁代币 
    function destroyToken(address account, uint amount) external {
        require(amount > 0, "ERC-20: amount less than or equal to 0.");
        require(account != address(0), "ERC-20: destroy to zero address.");
        uint256 accountBalance = balanceOf[account]; 
        require(accountBalance >= amount, "ERC-20: destroy amount exceeds balance.");

        balanceOf[account] = accountBalance - amount;
        totalSupply -= amount; 

        emit Destroy(account, amount);
    }

    // 转账
    function transfer(address _to, uint256 _value) external returns (bool success){
        _transfer(msg.sender, _to, _value);
        return true; 
    }

    // 从 _from 账户转账 _value 到 _to 账户
    function transferFrom(address _from, address _to, uint256 _value) external returns (bool success) {
        uint256 currentAllowance = allowance[_from][msg.sender];                        // 查询授权额度
        require(currentAllowance >= _value, "ERC-20: transfer amount exceeds allowances.");           // 是否超过授权额度 

        _transfer(_from, _to, _value);
        _approve(_from, msg.sender, currentAllowance - _value);

        return true;
    }

    function _transfer(address _from, address _to, uint256 _value) internal  {
        require(_value > 0, "ERC-20: value less than or equal to 0.");
        require(_from != address(0), "ERC-20: transfer from the zero address.");
        require(_to != address(0), "ERC-20: transfer to zero address.");
        uint256 senderBalance = balanceOf[_from];
        require(senderBalance >= _value, "ERC-20: sender amount exceeds balance.");

        balanceOf[_from] = senderBalance - _value; 
        balanceOf[_to] += _value;

        emit Transfer(_from, _to, _value); 
    }

    // 进行授权 
    function approve(address _spender, uint256 _value) external returns (bool success){
        _approve(msg.sender, _spender, _value); 
        return true;
    }

    function _approve(address _from, address _to, uint256 _value) internal {
        require(_value >= 0, "ERC-20: value less than or equal to 0.");
        require(_from != address(0), "ERC-20: approve from the zero address.");    
        require(_to != address(0), "ERC-20: approve to the zero address.");

        allowance[_from][_to] = _value; 

        emit Approval(_from, _to, _value);
    }
}

4、前端页面的编写

使用 bootstrap3.3.7 框架编写一个简单的前端界面
在这里插入图片描述

index.html 源码

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>同质化代币Dapp</title>
    <link href='https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css' rel='stylesheet' type='text/css'>
    <style>
        .jumbotron{
            background-color: #EDF0F5;
        }
    </style>
</head>
<body>
    <div class="container">
        <div class="row clearfix">
            <div class="col-md-12 column">
                <h1 style="text-align: center;">💱同质化代币 Dapp 搭建</h1>
                <div class="jumbotron" >
                    <div class="row clearfix">
                        <div class="col-md-3 column">
                            <h3 id="tokenName"></h3>
                        </div>
                        <div class="col-md-3 column">
                            <h3 id="tokenSymbol"></h3>
                        </div>
                        <div class="col-md-3 column">
                            <h3 id="tokenDecimals"></h3>
                        </div>
                        <div class="col-md-3 column">
                            <h3 id="totalSupply"></h3>
                        </div>
                    </div>
                </div>
                <div class="row clearfix">
                    <div class="col-md-4 column">
                        <span style="font-size: 20px;"><b>授权</b></span>
                        <form class="form-horizontal" role="form">
                            <div class="form-group">
                                <label for="approveSpender" class="col-sm-3 control-label">_spender:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="approveSpender" placeholder="address"/>
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="approveValue" class="col-sm-3 control-label">_value:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="approveValue" placeholder="uint256"/>
                                </div>
                            </div>
                            <div class="form-group">
                                <div class="col-sm-offset-3 col-sm-9">
                                    <a class="btn btn-primary" onclick="homoTokens('approve')">确定</a>
                                    <font id="approveResult" style="float: right; font-size:medium;"></font>
                                </div>
                            </div>
                        </form>
                    </div>
                    <div class="col-md-4 column">
                        <span style="font-size: 20px;"><b>允许</b></span>
                        <form class="form-horizontal" role="form">
                            <div class="form-group">
                                <label for="allowanceOwner" class="col-sm-3 control-label">_owner:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="allowanceOwner" placeholder="address" />
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="allowanceSpender" class="col-sm-3 control-label">_spender:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="allowanceSpender" placeholder="address" />
                                </div>
                            </div>
                            <div class="form-group">
                                <div class="col-sm-offset-3 col-sm-9">
                                    <a class="btn btn-success" onclick="homoTokens('allownace')">查询</a>
                                    <font id="allownaceResult" style="float: right; font-size:medium;"></font>
                                </div>
                            </div>
                        </form>
                    </div>
                    <div class="col-md-4 column">
                        <span style="font-size: 20px;"><b>转账</b></span>
                        <form class="form-horizontal" role="form">
                            <div class="form-group">
                                <label for="transferTo" class="col-sm-3 control-label">_to:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="transferTo" placeholder="address" />
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="transferValue" class="col-sm-3 control-label">_value:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="transferValue" placeholder="uint256" />
                                </div>
                            </div>
                            <div class="form-group">
                                <div class="col-sm-offset-3 col-sm-9">
                                    <a class="btn btn-primary" onclick="homoTokens('transfer')">确定</a>
                                    <font id="transferResult" style="float: right; font-size:medium;"></font>
                                </div>
                            </div>
                        </form>
                    </div>
                </div> 
                <div class="row clearfix">
                    <div class="col-md-4 column">
                        <span style="font-size: 20px;"><b>从...转账</b></span>
                        <form class="form-horizontal" role="form">
                            <div class="form-group">
                                <label for="transferFromFrom" class="col-sm-3 control-label">_from:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="transferFromFrom" placeholder="address" />
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="transferFromTo" class="col-sm-3 control-label">_to:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="transferFromTo" placeholder="address" />
                                </div>
                            </div>
                            <div class="form-group">
                                <label for="transferFromValue" class="col-sm-3 control-label">_value:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="transferFromValue" placeholder="uint256" />
                                </div>
                            </div>
                            <div class="form-group">
                                <div class="col-sm-offset-3 col-sm-9">
                                    <a class="btn btn-primary" onclick="homoTokens('transferFrom')">确定</a>
                                    <font id="transferFromResult" style="float: right; font-size:medium;"></font>
                                </div>
                            </div>
                        </form>
                    </div>
                    <div class="col-md-4 column">
                        <span style="font-size: 20px;"><b>余额</b></span>
                        <form class="form-horizontal" role="form">
                            <div class="form-group">
                                <label for="balanceOfOwner" class="col-sm-3 control-label">_owner:</label>
                                <div class="col-sm-9">
                                    <input type="text" class="form-control" id="balanceOfOwner" placeholder="address" />
                                </div>
                            </div>
                            <div class="form-group">
                                <div class="col-sm-offset-3 col-sm-9">
                                    <a  class="btn btn-success" onclick="homoTokens('balanceOf')">查询</a>
                                    <font id="balanceOfResult" style="float: right; font-size:medium;"></font>
                                </div>
                            </div>
                        </form>
                    </div>
                </div> 
                <hr/>
            </div>
        </div>
    </div>
</body>
<script src="https://cdn.jsdelivr.net/npm/web3@0.20.1/dist/web3.js"></script>
<script src="./index.js"></script>
</html>

index.js 源码

const web3 = new Web3(new Web3.providers.HttpProvider("HTTP://127.0.0.1:7545"));   
const abi = JSON.parse('[{"inputs":[{"internalType":"string","name":"name_","type":"string"},{"internalType":"string","name":"symbol_","type":"string"},{"internalType":"uint8","name":"decimals_","type":"uint8"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_owner","type":"address"},{"indexed":true,"internalType":"address","name":"_spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"_value","type":"uint256"}],"name":"Approval","type":"event"},{"inputs":[{"internalType":"address","name":"_spender","type":"address"},{"internalType":"uint256","name":"_value","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Destroy","type":"event"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"destroyToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Forge","type":"event"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"forgeToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_value","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"_from","type":"address"},{"indexed":true,"internalType":"address","name":"_to","type":"address"},{"indexed":false,"internalType":"uint256","name":"_value","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"_from","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_value","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"success","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]')
const token20Contract = web3.eth.contract(abi);
const contractInstance = token20Contract.at('0xe95D98B5975B4D54Fe6d6F555A55009F1AEec13B');        //部署的合约地址
const fromAccount = web3.eth.accounts[0]; 
console.log(web3.eth.accounts); 

// 页面加载完执行 
window.onload = function () {
    const name = contractInstance.name.call().toString();
    const symbol = contractInstance.symbol.call().toString();
    const decimals = contractInstance.decimals.call().toString();
    const totalSupply = contractInstance.totalSupply.call().toString();

    setElementText('tokenName', "名称: " + name);
    setElementText('tokenSymbol', "符号: " + symbol);
    setElementText('tokenDecimals', "小数位: " + decimals);
    setElementText('totalSupply', "总供给量: " + totalSupply);
};

// 设置元素文本 
function setElementText(eleId, text) {
  const element = document.getElementById(eleId);
  element.innerHTML = text;
}

// 获取元素值 
function getElementValues(eleIds) {
    const result = []; 
    for (const id of eleIds) {
        const inputValue = document.getElementById(id).value;
        if (!inputValue) {
            alert("缺少必要的参数。");
            return;
        } 
        result.push(inputValue);
    }
    return result; 
}
    
// 同质化代币 
function homoTokens(id) {
    if (id == "approve") {
        const argument = getElementValues(['approveSpender', 'approveValue']);
        if (argument) {
            // 返回交易的哈希值
            const transactionHash = contractInstance.approve(argument[0], argument[1], { from: fromAccount });
            if (transactionHash) setElementText("approveResult", 'true');
            else setElementText("approveResult", 'error');
        } 
    } else if (id == "allownace") {
        const argument = getElementValues(['allowanceOwner', 'allowanceSpender']);
        if (argument) {
            const result = contractInstance.allowance.call(argument[0], argument[1]).toString();
            setElementText("allownaceResult", result);
        }
    } else if (id == "transfer") {
        const argument = getElementValues(['transferTo', 'transferValue']);
        if (argument) {
            const transactionHash = contractInstance.transfer(argument[0], argument[1], { from: fromAccount });
            if (transactionHash) setElementText("transferResult", 'true');
            else setElementText("transferResult", 'error');
        } 
    } else if (id == "transferFrom") {
        const argument = getElementValues(['transferFromFrom', 'transferFromTo', 'transferFromValue']);
        if (argument) {
            const transactionHash = contractInstance.transferFrom(argument[0], argument[1],
                    argument[2], { from: web3.eth.accounts[1] });
            if (transactionHash) setElementText("transferFromResult", 'true');
            else setElementText("transferFromResult", 'error');
        } 
    } else {
        const argument = getElementValues(['balanceOfOwner']);
        if (argument) {
            const result = contractInstance.balanceOf.call(argument[0]).toString(); 
            setElementText("balanceOfResult", result);
        }
    }
}

5、部署流程及操作演示

首先需要在 MetaMask 中导入 Ganache 私有链上的账户,以第一个为例,
在这里插入图片描述
复制第一个地址的私钥导入 MetaMask 中。
在这里插入图片描述
在这里还需要在 MetaMask 的设置中手动添加 Ganache 的私有链网络,基本的信息都可以在 Ganache 首界面找到,添加好切换至新添加的网络。
在这里插入图片描述
在这里插入图片描述
打开在线版的 Remix 导入前面示例的 solidity 代码, 连接刚刚创建好的 MetaMask 账户
在这里插入图片描述
需要在 Remix 的编译选项中把 Advanced Configurations 中的 EVM VERSION 改为跟 Ganache 的配置一致才能运行智能合约。
在这里插入图片描述
在这里插入图片描述
否则会报错(Gas estimation errored with the following message (see below). The transactionexecution will likely fail. Do you want to force sending?
在这里插入图片描述
输入初始化信息进行部署
在这里插入图片描述
复制部署好的合约地址 和 abi 到 web3.js 的代码中, 如下所示
在这里插入图片描述
在这里插入图片描述
给 fromAccount 账户锻造 200 个代币用于功能的正常运行
在这里插入图片描述
在前端页面输入 fromAccount 账户地址可以看到显示刚才锻造的 200 个代币。
在这里插入图片描述
至此一个简单的 Dapp 的搭建和相关操作就完成了。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Linux环境grep搜索方法记录

1 grep grep 命令&#xff0c;用来搜索字符串所在位置&#xff0c;可以具体到不同文件&#xff0c;不同行&#xff1b; 在Linux 下&#xff0c;查看命令释义如下 zhaocubuntu2004:~$ grep --help Usage: grep [OPTION]... PATTERNS [FILE]... Search for PATTERNS in each FI…

安装Hadoop:Hadoop的单机模式、伪分布式模式——备赛笔记——2024全国职业院校技能大赛“大数据应用开发”赛项

前言 Hadoop包括三种安装模式&#xff1a; 单机模式&#xff1a;只在一台机器上运行&#xff0c;存储是采用本地文件系统&#xff0c;没有采用分布式文件系统HDFS&#xff1b;伪分布式模式&#xff1a;存储采用分布式文件系统HDFS&#xff0c;但是&#xff0c;HDFS的名称节点…

【VRTK】【VR开发】【Unity】18-VRTK与Unity UI控制的融合使用

课程配套学习项目源码资源下载 https://download.csdn.net/download/weixin_41697242/88485426?spm=1001.2014.3001.5503 【背景】 VRTK和Unity自身的UI控制包可以配合使用发挥效果。本篇就讨论这方面的实战内容。 之前可以互动的立体UI并不是传统的2D UI对象,在实际使用中…

Mac 安装 Minikube 及解决 “[ERROR ImagePull]: failed to pull image“ 问题

文章目录 1. 引言2. Mac 安装 Docker3. Mac 安装 Minikube4. 拉起集群4.1 启动4.2 验证 5. 参考 1. 引言 Minikube 是一种轻量化的 Kubernetes 集群&#xff0c;旨在帮助开发者和学习者更好地学习和体验 Kubernetes 的功能。它可以在个人计算机的虚拟化环境中快速构建和启动 Ku…

网易有道词典不能截屏翻译,不能联网解决办法

对应版本&#xff1a; win10系统&#xff0c;联想拯救者笔记本&#xff0c;网易有道词典8.10.2.0。 网易有道词典免费下载链接&#xff1a;https://download.csdn.net/download/qq_42755734/88684985 修改代理&#xff1a; youdao.com 0 取消勾选---不更新 效果&#xff1a…

移动应用开发:揭秘内侧APP封装台的高效

在数字化浪席卷下&#xff0c;移应用已经成连接企业与用户纽带。为了抢占市场先机&#xff0c;快速发布高质量的移动应用成为业竞争的关键。侧APP封装平因此而诞生&#xff0c;成为了应开发者的得助手。以下是内侧APP封装台的全面解读&#xff0c;助在应用开发海洋中乘风破浪。…

告别HTTP,拥抱HTTPS!免费SSL证书领取指南

为什么选择HTTPS&#xff1f; HTTP和HTTPS之间的主要区别在于安全性。HTTP是一种不安全的协议&#xff0c;数据在传输过程中是明文的&#xff0c;容易受到中间人攻击。而HTTPS通过SSL&#xff08;Secure Sockets Layer&#xff09;或TLS&#xff08;Transport Layer Security&…

python期刊稿件在线投稿系统q2ud0

本系统的用户可分为管理员、投稿者、审稿人和编辑四个用户角色组成。管理员可以管理系统内所有功能&#xff0c;主要有个人中心、投稿者管理、审稿人管理、编辑管理、个人稿件管理、审核稿件管理、稿件信息管理、类型管理等功能&#xff1b;编辑登录系统主要有个人中心、审核稿…

2023-12-22 LeetCode每日一题(得到山形数组的最少删除次数)

2023-12-22每日一题 一、题目编号 1671. 得到山形数组的最少删除次数二、题目链接 点击跳转到题目位置 三、题目描述 我们定义 arr 是 山形数组 当且仅当它满足&#xff1a; arr.length > 3存在某个下标 i &#xff08;从 0 开始&#xff09; 满足 0 < i < arr.…

阿里云PolarDB数据库优惠价格表11元一天起

阿里云数据库PolarDB租用价格表&#xff0c;云数据库PolarDB MySQL版2核4GB&#xff08;通用&#xff09;、2个节点、60 GB存储空间55元5天&#xff0c;云数据库 PolarDB 分布式版标准版2核16G&#xff08;通用&#xff09;57.6元3天&#xff0c;阿里云百科aliyunbaike.com分享…

UE蓝图 RPG动作游戏(一) day15

角色状态制作 制作角色动画混合空间 创建一个动混合空间 添加动作在混合空间 动画蓝图 创建一个动画蓝图 先使用混合空间进行移动&#xff0c;后续优化后再使用状态机 编写垂直水平速度逻辑初始化&#xff0c;获取到此动画的角色组件 获取Horizontal与Vertical的速度逻辑 …

vue项目表单使用正则过滤ip、手机号

import useFormValidate from /hooks/useFormValidatesetup(props, { emit }) {const { validateName, validateIPAndPort } useFormValidate()const state reactive({workFaceInfo: props.info?.id ? props.info : {},sysTypeData: props.sysType,formRules: {name: [{req…

QDockWidget学习

一、使用方法 QT之QDockWidget使用详解_qt dockwidget-CSDN博客 重点摘抄&#xff1a; 二、QDockWidget的常用函数&#xff1a; QDockWidget嵌套布局详解-实现Visual Studio布局_http://blog.csdn.net/czyt1988/article/details/5120-CSDN博客 摘抄&#xff1a;

【Redis-05】Redis如何实现保存键值对的保存及过期键的管理策略

在之前的文章我们介绍过&#xff0c;Redis服务器在启动之初&#xff0c;会初始化RedisServer的实例&#xff0c;在这个实例中存在很多重要的属性结构&#xff0c;同理本篇博客中介绍的数据库实现原理也会和其中的某些属性相关&#xff0c;我们继续看一下吧。 1.服务器和客户端…

大数据概念:数据网格和DataOps

数据网格&#xff08;Data Mesh&#xff09; 一种新型的数据架构模式&#xff0c;旨在解决传统数据架构中存在的一些问题&#xff0c;例如数据孤岛、数据冗余、数据安全等。数据网格将数据作为一种服务&#xff0c;通过在分布式环境中提供数据服务&#xff0c;实现数据的共享和…

Apollo自动驾驶系统:实现城市可持续交通的迈向

前言 「作者主页」&#xff1a;雪碧有白泡泡 「个人网站」&#xff1a;雪碧的个人网站 ChatGPT体验地址 文章目录 前言引言&#xff1a;1. 什么是微服务架构&#xff1f;2. 微服务架构的组成要素3. 微服务架构的挑战和解决方案4. 微服务架构的可扩展性和弹性 第二部分&#x…

回首2023: 程序员跳出舒适圈

1 前言 今天的冬日暖阳高照&#xff0c;照耀着我穿着羽绒服的身体&#xff0c;让我感到火一般的燥热&#xff0c;仿佛错觉中已经到了阳春三月。刚刚把孩子洗好&#xff0c;我坐在电脑前&#xff0c;准备整理一下思绪&#xff0c;回顾一下2023年的生活和工作。 2 2023 回顾 回…

信号与线性系统翻转课堂笔记17——z变换及其性质

信号与线性系统翻转课堂笔记17——z变换及其性质 The Flipped Classroom17 of Signals and Linear Systems 对应教材&#xff1a;《信号与线性系统分析&#xff08;第五版&#xff09;》高等教育出版社&#xff0c;吴大正著 一、要点 &#xff08;1&#xff09;序列的z变换…

Hive生产调优介绍

1.Fetch抓取 Fetch抓取是指&#xff0c;Hive中对某些情况的查询可以不必使用MapReduce计算。例如&#xff1a;SELECT * FROM employees;在这种情况下&#xff0c;Hive可以简单地读取employee对应的存储目录下的文件&#xff0c;然后输出查询结果到控制台。 在hive-default.xml…

DFS

目录 DFS 实现数字全排列 N 皇后问题 DFS 算法的理解 优先考虑深度&#xff0c;换句话说就是一条路走到黑&#xff0c;直到无路可走的情况下&#xff0c;才会选择回头&#xff0c;然后重新选择一条路。空间复杂度&#xff1a;O&#xff08;h&#xff09;和高度成正比 不具…