unity【动画】脚本_角色动画控制器 c#

news2025/1/11 8:42:39

首先创建一个代码文件夹Scripts

从人物角色Player的基类开始   创建IPlayer类

首先我们考虑到如果不挂载MonoBehaviour需要将角色设置成预制体实例化到场景上十分麻烦,

所以我们采用继承MonoBehaviour类的角色基类方法写代码

也就是说这个脚本直接绑定在角色物体身上,就不需要实例化了,相对降低了复杂程度。

首先我们需要在unity场景制作一个Renderer类型的目标点

首先创建一个平面Plane改名为MovementTargetSign

修改位置

移除掉碰撞器

添加一个材质 

对材质进行修改 点击Shader选项

选择一个纹理

代码:

using UnityEngine;
//抽象角色类-包括玩家和NPC
public class IPlayer : MonoBehaviour{
    protected Animator _animator;//动画器组件引用
    private IWeapon _weapon = null;//武器的引用
}

public class IWeapon{

}

using UnityEngine;
public class Player : IPlayer{
    private Renderer movementSign;//移动标志
    private Collider attackPlane;//攻击区
    private Collider floorPlane;//普通地面(非攻击区)
    private Vector3 lookatPos;//面向位置
    private float rotSpeed = 20f;//移动旋转速度
    private float attackRotSpeed = 10f;//攻击旋转速度
}

public class FemaleWarrior : Player{

}
将FemaleWarrior代码挂载Player对象身上

对角色Player添加胶囊碰撞器

调整胶囊碰撞器位于角色中心

再添加刚体

关掉 使用重力Use Gravity 勾选

在约束上constraints 冻结旋转 x y z

如果制作的角色不需要重力则用碰撞器实现,

如果制作的角色需要重力    则用刚体   实现

先用刚体实现:

代码如下:

using UnityEngine;
//抽象角色类-包括玩家和NPC
public class IPlayer : MonoBehaviour{
    protected Animator _animator;//动画器组件引用
    private IWeapon _weapon = null;//武器的引用
}

public class IWeapon{

}

using UnityEngine;
public class Player : IPlayer{
    private Renderer movementSign;//移动标志
    private Collider attackPlane;//攻击区
    private Collider floorPlane;//普通地面(非攻击区)
    private Vector3 lookatPos;//面向位置
    private float rotSpeed = 20f;//移动旋转速度
    private float attackRotSpeed = 10f;//攻击旋转速度
    protected virtual void Awake() {
        //移动标志
        if (movementSign == null)
            movementSign = GameObject.Find("MovementTargetSign").GetComponent<Renderer>();
    }
    protected virtual void Start(){
        //移动标志默认放在玩家脚下
        movementSign.transform.position = transform.position + new Vector3(0,0.02f,0);
        movementSign.enabled = false;//关闭移动标志的显示
    }
    //跟随鼠标旋转
    public void RotateWithCursorPos() {
        RaycastHit hit;
        //构建一条从摄像机到鼠标位置的射线
        var Ray = Camera.main.ScreenPointToRay(Input.mousePosition);
        if (Physics.Raycast(Ray, out hit)) {
            //计算方向
            Vector3 mousePos = new Vector3(hit.point.x, transform.position.y, hit.point.z);
            Vector3 playerDirection = mousePos - transform.position;
            if (playerDirection != Vector3.zero) {
                //旋转到目标方向
                transform.rotation = Quaternion.Slerp(transform.rotation,
                    Quaternion.LookRotation(playerDirection), Time.deltaTime * attackRotSpeed);
            }
        }
    }
    protected virtual void Update() {
        RotateWithCursorPos();
    }
}

public class FemaleWarrior : Player{
    protected override void Awake(){
        base.Awake();
    }
    protected override void Start(){
        base.Start();
    }
}
完成人物转向之后,开始做人物移动功能

首先添加代码,将动画机参数转换为哈希值

using UnityEngine;
public class AnimaterConsteantVelues {
    public static int WeaponID = Animator.StringToHash("WeaponID");
    public static int isCombat = Animator.StringToHash("isCombat");
    public static int isIdle = Animator.StringToHash("isIdle");
    public static int Attack = Animator.StringToHash("Attack");
}
在角色基类中添加函数

using UnityEngine;
//抽象角色类-包括玩家和NPC
public class IPlayer : MonoBehaviour{
    protected Animator _animator;//动画器组件引用
    private IWeapon _weapon = null;//武器的引用
    public IWeapon Weapon {
        get => _weapon;
        set => _weapon = value;
    }
    public void Attack() {
        if (_weapon != null)
            _weapon.Attack();
    }
}
修改抽象武器基类

using UnityEngine;
public abstract class IWeapon{
    public string WeaponName { get; set; }
    protected GameObject _weaponModel;
    protected GameObject _weaponPrefab;
    private int weaponID { get; set; }
    protected IPlayer _player { get; set; }
    public GameObject WeaponPrefab {
        get => _weaponModel; 
        set => _weaponModel = value;
    }
    public virtual void Attack() { }
    public virtual void RefreshLine() { }
    public IWeapon(int weaponID, string name, string weaponModelPath, IPlayer player){
        this.weaponID = weaponID;
        WeaponName = name;
        _player = player;
        if (weaponModelPath != "") {
            _weaponModel = Resources.Load<GameObject>(weaponModelPath);
            if (_weaponModel != null) {
                var weaponPos = ((Player)_player).handleWeaponPosList[weaponID];
                _weaponPrefab = GameObject.Instantiate(_weaponModel, weaponPos.position, weaponPos.rotation);
                _weaponPrefab.transform.SetParent(weaponPos);
                _weaponPrefab.SetActive(false);
            }
        }
    }
}
修改角色子类

using System.Collections.Generic;
using UnityEngine;
public class Player : IPlayer{
    private Renderer movementSign;//移动标志
    private Collider attackPlane;//攻击区
    private Collider floorPlane;//普通地面(非攻击区)
    private Vector3 lookatPos;//面向位置
    private float rotSpeed = 20f;//移动旋转速度
    private float attackRotSpeed = 10f;//攻击旋转速度
    private float fallSpeed;
    //列表
    [SerializeField]
    public List<Transform>handleWeaponPosList = new List<Transform>();
    protected virtual void Awake() {
        _animator = GetComponent<Animator>();
        floorPlane = GameObject.Find("Plane").GetComponent<Collider>();
        //移动标志
        if (movementSign == null)
            movementSign = GameObject.Find("MovementTargetSign").GetComponent<Renderer>();
    }
    protected virtual void Start(){
        //移动标志默认放在玩家脚下
        movementSign.transform.position = transform.position + new Vector3(0,2f,0);
        movementSign.enabled = false;//关闭移动标志的显示
    }
    //跟随鼠标旋转
    public void RotateWithCursorPos() {
        RaycastHit hit;
        //构建一条从摄像机到鼠标位置的射线
        var Ray = Camera.main.ScreenPointToRay(Input.mousePosition);
        if (Physics.Raycast(Ray, out hit)) {
            //计算方向
            Vector3 mousePos = new Vector3(hit.point.x, transform.position.y, hit.point.z);
            Vector3 playerDirection = mousePos - transform.position;
            if (playerDirection != Vector3.zero) {
                //旋转到目标方向
                transform.rotation = Quaternion.Slerp(transform.rotation,
                    Quaternion.LookRotation(playerDirection), Time.deltaTime * attackRotSpeed);
            }
        }
    }
    public void Move(){
        RaycastHit hit;
        var ray = Camera.main.ScreenPointToRay(Input.mousePosition);
        if (Input.GetMouseButton(0)){
            if (floorPlane.Raycast(ray, out hit, 50f)){
                movementSign.transform.position = hit.point + new Vector3(0, 0.01f, 0);
                movementSign.enabled = true;
                lookatPos = hit.point;
            }
        }
        else if(Input.GetMouseButtonUp(1)){
            _animator.SetBool(AnimaterConsteantVelues.isCombat, true);
            _animator.SetTrigger(AnimaterConsteantVelues.Attack);
            Attack();
        }
        lookatPos.y = transform.position.y;
        var playerDirection = lookatPos - transform.position;
        if (playerDirection != Vector3.zero) 
            transform.rotation = Quaternion.Slerp(transform.rotation, Quaternion.LookRotation(playerDirection), Time.deltaTime * rotSpeed);
        var offset = movementSign.transform.position - transform.position;
        var sqrDistance = offset.sqrMagnitude;
        if (sqrDistance > 0.1f){
            _animator.SetBool(AnimaterConsteantVelues.isIdle, false);
            rotSpeed = 20f;
        }
        else {
            _animator.SetBool(AnimaterConsteantVelues.isIdle,true);
            movementSign.enabled = false;
            movementSign.transform.position = transform.position;
            rotSpeed = 0;
        }
        var bodyRay = new Ray(transform.position + transform.up, transform.up * -1);
        if (floorPlane.Raycast(bodyRay, out hit, 1.0f)) {
            if (hit.point.y > transform.position.y + 0.02f)
                transform.position = hit.point + new Vector3(0, 0.02f, 0);
            else if (floorPlane.Raycast(bodyRay, out hit, 1.2f))
                if (hit.point.y > transform.position.y - 0.02f) 
                    transform.position = hit.point + new Vector3(0, -0.02f, 0);
            else {
                fallSpeed += 0.1f;
                var v = new Vector3(0, fallSpeed * Time.deltaTime, 0);
                transform.position -= v;
                movementSign.transform.position = transform.position + new Vector3(0, 0.01f, 0);
            }
        }
    }
    protected virtual void Update() {
        RotateWithCursorPos();
        Move();
    }
}

回到Unity场景中拖拽填充武器刷新

人物移动完成

接下来完成背包系统 

放进预制体包后完全解压缩

调整好Slot0-4的背景位置,将四个子物体预制体包后删除

子物体位置都改成0

隐藏/取消激活

接下来完成拾取道具

加碰撞器

调节碰撞器

加触发器

加刚体

修改名字

创建脚本

using JetBrains.Annotations;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
//可拾取道具
public class CanPickupItem : MonoBehaviour{
    public AudioClip pickUpSound;//拾取声音
    private void OnTriggerEnter(Collider other)
    {
        if (other.CompareTag("Player")) {
            //播放声音
            if (pickUpSound != null)
                AudioSource.PlayClipAtPoint(pickUpSound,transform.position);
            //将本道具更新到背包列表中
            GameObject.FindGameObjectWithTag("InventoryUITag").GetComponent<InventoryManager>().ItemNames.Add(gameObject.name);
            Destroy(gameObject);

        }
    }
}
 

挂载脚本

同样挂载到手枪上

添加标签

创建脚本

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;

public class InventoryManager : MonoBehaviour
{
    //道具名称列表
    public List<string> ItemNames = new List<string>();

}
 

给人物标签

标签

运行即可触发拾取道具

接下来做背包系统

首先这里不用字典代码,运用简单方式制作,前提必须保证道具 和 道具图片的英文名字存在包含关系

在管理类中写一个打开背包的方法

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;

public class InventoryManager : MonoBehaviour
{
    //道具名称列表
    public List<string> ItemNames = new List<string>();
    //打开或关闭背包
    public void OpenOrCloseInventoryUI(bool isOpen) {
        transform.Find("Panel").gameObject.SetActive(isOpen);
    }
}

using System.Collections;
using System.Collections.Generic;
using Unity.VisualScripting.Antlr3.Runtime;
using UnityEngine;
using UnityEngine.UI;

public class InventoryManager : MonoBehaviour
{
    //道具名称列表
    public List<string> ItemNames = new List<string>();
    //是否显示背包UI
    private bool isShowInventoryUI = false;
    //打开或关闭背包
    public void OpenOrCloseInventoryUI(bool isOpen) {
        transform.Find("Panel").gameObject.SetActive(isOpen);
    }
    private void Update()
    {
        if (Input.GetKeyDown(KeyCode.B)) {
            isShowInventoryUI = !isShowInventoryUI;
            //打开或关闭UI
            OpenOrCloseInventoryUI(isShowInventoryUI);
        }
    }
}

即可实现背包按B键开关背包UI

接下来我们需要做一个背包图标的UI,点击UI也能打开背包

即完成鼠标点击显隐背包UI 及 键盘B键显隐背包UI

接下来做UI文本更新,意义不大可省略,存在的意义在于完善体系

using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class InventoryManager : MonoBehaviour{
    //道具名称列表
    public List<string> ItemNames = new List<string>();
    //是否显示背包UI
    private bool isShowInventoryUI = false;
    //UI界面中的文本
    public Text[] textUI;
    //激活或关闭背包UI显示
    public void OpenOrCloseInventoryUI(bool isOpen) {
        transform.Find("Panel").gameObject.SetActive(isOpen);
        //更新文本
        UpdateInventoryTextUI();
    }
    private void Update(){
        if (Input.GetKeyDown(KeyCode.B))
            InventoryUIState();
    }
    //打开或关闭背包
    public void InventoryUIState() {
        isShowInventoryUI = !isShowInventoryUI;
        //打开或关闭UI
        OpenOrCloseInventoryUI(isShowInventoryUI);
    }
    //更新文本UI 
    private void UpdateInventoryTextUI(){
        for(int i = 0;i< ItemNames.Count;i++)
            textUI[i].text = ItemNames[i];
    }
}

调整文本在背包UI中的位置

取消激活面板

using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class InventoryManager : MonoBehaviour{
    public List<string> ItemNames = new List<string>();//道具名称列表
    private bool isShowInventoryUI = false;//是否显示背包UI
    public Text[] textUI;//UI界面中的文本
    public Image[] availableItemIcons;//可以获取的道具图标
    public void OpenOrCloseInventoryUI(bool isOpen){//激活或关闭背包UI显示
        transform.Find("Panel").gameObject.SetActive(isOpen);
        UpdateInventoryTextUI(); //更新文本
        UpdateInventoryIconUI();//更新图标
    }
    private void Update(){
        if (Input.GetKeyDown(KeyCode.B))
            InventoryUIState();
    }
    public void InventoryUIState(){//打开或关闭背包
        isShowInventoryUI = !isShowInventoryUI;
        OpenOrCloseInventoryUI(isShowInventoryUI);//打开或关闭UI
    }
    private void UpdateInventoryTextUI(){//更新文本UI 
        for (int i = 0;i < ItemNames.Count;i++)
            textUI[i].text = ItemNames[i];
    }
    private void UpdateInventoryIconUI(){//更新图标UI
        for (int i = 0; i < ItemNames.Count; i++){
            Image itemIcon = GetIconPrefabByItemName(ItemNames[i]);//根据道具名称返回对应的图标
            if (itemIcon != null){
                Image newItemIcon = Instantiate(itemIcon);//将图标克隆到对应的Image中
                newItemIcon.transform.SetParent(textUI[i].transform.parent);//更改父物体
                RectTransform rt = newItemIcon.GetComponent<RectTransform>();//调整位置
                rt.anchoredPosition = Vector3.zero;
            }
            else
                Debug.LogError("没找到对应图标");
        }
    }
    private Image GetIconPrefabByItemName(string name){//根据道具名称返回对应的图标
        for (int i = 0; i < availableItemIcons.Length; i++){
            if (availableItemIcons[i].name.Contains(name))
                return availableItemIcons[i];
        }
        return null;
    }
}

隐藏文字部分

运行即完成

挂移动摄像机

using UnityEngine;
public class CameraMove : MonoBehaviour{ //第三人称摄像机简单版
    public Transform target;//摄像机的跟随目标
    public float distance = 8.0f;//摄像机与目标之间的距离
    private float x, y, z;
    private float xSpeed = 250.0f;
    private float ySpeed = 120.0f;
    public float yMinlimit = -45.0f;//限制上下移动角度
    public float yMaxlimit = 45.0f;
    private void Awake(){
        //注册场景加载完毕事件
        //EventCenter.AddListener(EventType.SceneLoadComplete, SetTarget);
    }
    private void SetTarget(){
        //将标签为Player的物体设置为跟踪目标
        Transform player = GameObject.FindGameObjectWithTag("Player").transform;
        if (player != null && target == null)
            target = player;
    }
    private void Start(){
        Vector3 angles = transform.eulerAngles;//获取摄像机的当前角度
        x = angles.x;
        y = angles.y;
        z = -distance;
        GoRight();
    }

    private void LateUpdate(){
        float temp = Input.GetAxis("Mouse ScrollWheel");//获取滚轮数值
        if (target != null){
            if (Input.GetMouseButton(0)){
                x += Input.GetAxis("Mouse X") * xSpeed * 0.02f;
                y -= Input.GetAxis("Mouse Y") * ySpeed * 0.02f;
            }
        }
        //钳制上下移动的角度
        y = ClampAngle(y,yMinlimit,yMaxlimit);
        z += temp * 100f * 0.02f;//数值按照自己喜好设定
        z = Mathf.Clamp(z,-20f,-3.0f);//距离限制,最远是距离玩家20米,最近是3米
        GoRight();//作用于摄像机
    }
    float ClampAngle(float angle,float min,float max){
        if (angle < -360)
            angle += 360;
        if (angle > 360)
            angle -= 360;
        return Mathf.Clamp(angle,min,max);
    }
    //摄像机控制位置及角度的核心方法
    void GoRight(){
        if (target == null)
            return;
        Quaternion rotation = Quaternion.Euler(y,x,0);//摄像机角度
        Vector3 position = rotation * new Vector3(0.0f,0.0f,z)+target.position;
        transform.position = position;//摄像机位置
        transform.rotation = rotation;//摄像机角度
    }
}

即完成摄像机跟随人物移动

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

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

相关文章

Quartz之JDBC-JobStoreTX配置

一、前言 上篇 《Quartz介绍》中使用的是RAMJobStored存储调度信息&#xff0c;当进程终止调度信息会丢失&#xff0c;本篇我们介绍使用JDBCJobStore来存储调度信息&#xff08;jobs、Triggers和日历&#xff09;。 二、Quartz 表结构 可以从官网&#xff08;http://www.qua…

康耐视深度学习ViDi-ViDi四大工具介绍与主要用途

Cognex ViDi 工具是一系列机器视觉工具&#xff0c;通过深度学习解决各种难以解决的挑战。虽然这些工具共享一个引擎&#xff0c;但它们在图像中寻找的内容不同。更具体地说&#xff0c;在分析单个点、单个区域或完整图像时&#xff0c;每个工具都有不同的侧重点。 Locate&…

JUC并发编程系列(一):Java线程

前言 JUC并发编程是Java程序猿必备的知识技能&#xff0c;只有深入理解并发过程中的一些原则、概念以及相应源码原理才能更好的理解软件开发的流程。在这篇文章中荔枝会梳理并发编程的基础&#xff0c;整理有关Java线程以及线程死锁的知识&#xff0c;希望能够帮助到有需要的小…

30、JAVA进阶——Socket编程

✅作者简介:热爱国学的Java后端开发者,修心和技术同步精进。 🍎个人主页:乐趣国学的博客 🍊个人信条:不迁怒,不贰过。小知识,大智慧。 💞当前专栏:JAVA开发者成长之路 ✨特色专栏:国学周更-心性养成之路 🥭本文内容:JAVA进阶——Socket编程 更多内容点击👇 …

idea2023 PoJie以后无法修改内存无效

1. 打开电脑环境变量 2. 找到对应pojie文件 vmoptions目录 3. 修改这个文件 添加或者修改配置 -Xms128m -Xmx8192m4. 重启idea 修改成功

flutter开发报错The instance member ‘widget‘ can‘t be accessed in an initializer

文章目录 问题描述问题原因解决方法 问题描述 The instance member ‘widget’ can’t be accessed in an initializer. 问题原因 “The instance member ‘widget’ can’t be accessed in an initializer” 错误是因为在初始化器列表中&#xff08;constructor initializer…

JavaScript使用正则表达式

正则表达式(RegExp)也称规则表达式(regular expression)&#xff0c;是非常强大的字符串操作工具&#xff0c;语法格式为一组特殊字符构成的匹配模式&#xff0c;用来匹配字符串。ECMAScript 3以Perl为基础规范JavaScript正则表达式&#xff0c;实现Perl 5正则表达式的子集。Ja…

Linux常用命令——cdrecord命令

在线Linux命令查询工具 cdrecord Linux系统下光盘刻录功能命令 补充说明 cdrecord命令用于Linux系统下光盘刻录&#xff0c;它支持cd和DVD格式。linux下一般都带有cdrecord软件。 语法 cdrecord(选项)(参数)选项 -v&#xff1a;显示刻录光盘的详细过程&#xff1b; -eje…

基于社交网络算法的无人机航迹规划-附代码

基于社交网络算法的无人机航迹规划 文章目录 基于社交网络算法的无人机航迹规划1.社交网络搜索算法2.无人机飞行环境建模3.无人机航迹规划建模4.实验结果4.1地图创建4.2 航迹规划 5.参考文献6.Matlab代码 摘要&#xff1a;本文主要介绍利用社交网络算法来优化无人机航迹规划。 …

UseGalaxy.cn生信云|新增热图绘制工具:heatmap2

2023-11-05&#xff0c;Galaxy生信云平台 UseGalaxy.cn 新增绘制热图工具。 Graph/Display Data heatmap2 (Galaxy Version 3.1.3galaxy0) 使用方法 进入网址&#xff1a; https://usegalaxy.cn/root?tool_idtoolshed.g2.bx.psu.edu/repos/iuc/ggplot2_heatmap2/ggplot2_heatm…

8.接口与抽象类 深入多态

8.1 不该初始化的class 这个结构有什么不对&#xff1f; 这个class结构不算太差。如此设计已经能够维持最少的重复程序代码&#xff0c;且有需要特地实现的方法也已经被覆盖过。从多态的角度来看&#xff0c;我们也做到了适应性&#xff0c;所以任何Animal的子型&#xff0c;包…

什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码

&#x1f9f8;欢迎来到dream_ready的博客&#xff0c;&#x1f4dc;相信您对这几篇博客也感兴趣o (ˉ▽ˉ&#xff1b;) &#x1f4dc;什么是SpringMVC&#xff1f;简单好理解&#xff01;什么是应用分层&#xff1f;SpringMVC与应用分层的关系&#xff1f; 什么是三层架构&…

【Unity细节】为什么UI移动了锚点,中心点和位置,运行的时候还是不在设置的位置当中

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;元宇宙-秩沅 hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 秩沅 原创 &#x1f636;‍&#x1f32b;️收录于专栏&#xff1a;unity细节和bug &#x1f636;‍&#x1f32b;️优质专栏 ⭐【…

从行车记录仪恢复已删除/丢失视频的方法

“我的车里有行车记录仪。几天前&#xff0c;当我下班回家时&#xff0c;一辆卡车不知从哪里冒出来撞向了我。我们的两辆车都损坏了&#xff0c;但幸运的是&#xff0c;没有人受伤。我曾与卡车司机就修理我的汽车进行过会面&#xff0c;但他说我有错。我需要查看我的行车记录仪…

微服务注册中心之安装+实例搭建zookeeper

1.下载安装包并上传到Linux服务器 Apache ZooKeeper 可以使用wget或者curl命令 wget http://mirror.bit.edu.cn/apache/zookeeper/zookeeper-3.7.1/apache-zookeeper-3.7.1-bin.tar.gz连接失败也可以本地下载之后上传到服务器 scp /本地/文件的/路径 用户名远程服务器IP或主…

(免费领源码)java#SSM#mysql基于响应式的网上书店系统27119-计算机毕业设计项目选题推荐

摘 要 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理系统的实施在技术上已逐步成熟。管理系统是一个不断发展的新型学科&#xff0c;本文主要通过对响应式的网上书店系统的功能性需求分析&#xff0c;对系统的安全性和可扩展性进行了非功能性需求分析。在详细的…

【Proteus仿真】【Arduino单片机】OLED液晶显示

文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真Arduino单片机控制器&#xff0c;使用IIC OLED液晶等。 主要功能&#xff1a; 系统运行后&#xff0c;OLED液晶显示各种图形、字符、图像。 二、软件设计 /* 作者&#xff1a;嗨小…

C++标准模板(STL)- 类型支持 (类型属性,is_bounded_array,is_unbounded_array)

类型特性 类型特性定义一个编译时基于模板的结构&#xff0c;以查询或修改类型的属性。 试图特化定义于 <type_traits> 头文件的模板导致未定义行为&#xff0c;除了 std::common_type 可依照其所描述特化。 定义于<type_traits>头文件的模板可以用不完整类型实例…

时间序列预测模型实战案例(八)(Informer)BestPaper论文模型Informer代码实战讲解

论文地址->Informer论文地址PDF点击即可阅读 代码地址-> 论文官方代码地址点击即可跳转下载GIthub链接 本文介绍 本篇博客带大家看的是Informer模型进行时间序列预测的实战案例&#xff0c;它是在2019年被提出并在ICLR 2020上被评为Best Paper&#xff0c;可以说Inform…

如何再kali中下载iwebsec靶场

这个靶场有三种搭建方法&#xff1a; 第一种是在线靶场&#xff1a;http://www.iwebsec.com:81/ 第二种是虚拟机版本的&#xff0c;直接下载到本地搭建 官网地址下载&#xff1a;http://www.iwebsec.com/ 而第三种就是利用docker搭建这个靶场&#xff0c;我这里是用kali进行…