Unity-2D游戏-打击感与敌人AI

news2025/1/19 16:58:35

前言

最近快搞毕设了,学一些Unity2D游戏开发的知识,发现b站宝藏up主奥飒姆Awesome的两个蛮不错的教程,我想简单记录一下它这个游戏设计的方法。

我不一点点实现了,就是分析一下大致框架(方便以后套用)


资源

打击感

Red hood pixel character by Legnops

Pixel Fantasy Caves by Szadi art.

Pixelated Attack/Hit Animations by Viktor

成品项目链接:

GitHub - RedFF0000/AttackSense

敌人AI

Animated Pixel Adventurer by rvros

Skeleton Sprite Pack by Jesse Munguia

成品项目链接:

GitHub - RedFF0000/Finite-state-machine


动作游戏打击感

玩家配置

我们来看主角的动画状态:

其中,对于标准运动定义了很多,比如idle进入run,通过分析浮点型变量Horizontal来设置,因为左右都要有动画,故设置了两个Transition:

 可以看到,除了基本的idle跑跳动作,还有一个Any State,监听了任意时刻的动作,例如我选中的any state到Light Attack1,就是当触发变量LightAttack触发时,ComboStep为1时则进入,应该就是进行轻攻击的第一下:

 这些内容的控制,主要都在代码里,我们看一下人物面板配置:

 截图中的speed调大,参考数据为 moveSpeed:180、lightSpeed:25、heavySpeed:35

来看一下PlayerController脚本,我已经把大概的注释写好了:

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

public class PlayerController : MonoBehaviour
{
    [Header("补偿速度")]
    public float lightSpeed;
    public float heavySpeed;
    [Header("打击感")]
    public float shakeTime;
    public int lightPause;
    public float lightStrength;
    public int heavyPause;
    public float heavyStrength;

    [Space]
    public float interval = 2f;
    private float timer;
    private bool isAttack;
    private string attackType; //攻击类型
    private int comboStep; //第几段攻击

    public float moveSpeed;
    public float jumpForce;
    new private Rigidbody2D rigidbody;
    private Animator animator;
    private float input;
    private bool isGround;
    [SerializeField] private LayerMask layer;

    [SerializeField] private Vector3 check;

    void Start()
    {
        //得到刚体
        rigidbody = GetComponent<Rigidbody2D>();
        //得到动画状态机
        animator = GetComponent<Animator>();
    }
    /*
     * 按帧执行,适合来处理GetKeyDown
     */
    void Update()
    {
        //得到水平输入
        input = Input.GetAxisRaw("Horizontal");
        //得到地面检测(layer)
        isGround = Physics2D.OverlapCircle(transform.position + new Vector3(check.x, check.y, 0), check.z, layer);
        //将刚体水平方向速度传给动画机
        animator.SetFloat("Horizontal", rigidbody.velocity.x);
        //将刚体竖直方向速度传给动画机
        animator.SetFloat("Vertical", rigidbody.velocity.y);
        //将是否触碰地面状态传给动画机
        animator.SetBool("isGround", isGround);
        //攻击处理
        Attack();
        Jump();
    }
    void Jump(){
        //进行跳跃
        if (Input.GetButtonDown("Jump") && isGround)
        {
            //给刚体一个y方向的力
            rigidbody.velocity = new Vector2(0, jumpForce);
            //同步动画机
            animator.SetTrigger("Jump");
        }
    }
    /*
     * 按固定时间间隔执行,适合来处理刚体效果
     */
    private void FixedUpdate()
    {
        //移动处理
        Move();
    }


    /*
     * 攻击处理函数
     */
    void Attack()
    {
        if (Input.GetKeyDown(KeyCode.J) && !isAttack) //轻击
        {
            //进入攻击状态
            isAttack = true;
            //设置攻击类型
            attackType = "Light";
            //攻击段数加一
            comboStep++;
            //还原攻击段数
            if (comboStep > 3)
                comboStep = 1;
            //设置攻击冷却间隔(负责自动还原攻击段数)
            timer = interval;
            //通知动画机
            animator.SetTrigger("LightAttack");
            animator.SetInteger("ComboStep", comboStep);
        }
        if (Input.GetKeyDown(KeyCode.K) && !isAttack) //重击
        {
            //重击
            isAttack = true;
            attackType = "Heavy";
            comboStep++;
            if (comboStep > 3)
                comboStep = 1;
            timer = interval;
            animator.SetTrigger("HeavyAttack");
            animator.SetInteger("ComboStep", comboStep);
        }


        if (timer != 0)
        {
            timer -= Time.deltaTime;
            if (timer <= 0)
            {
                timer = 0;
                comboStep = 0;
            }
        }
    }

    //攻击结束(此函数被每个攻击动画的快结束处调用)
    public void AttackOver()
    {
        isAttack = false;
    }

    /*
    * 移动处理函数
    */
    void Move()
    {
        if (!isAttack)
            //如果不在攻击中,则根据水平输入进行移动
            rigidbody.velocity = new Vector2(input * moveSpeed * Time.fixedDeltaTime, rigidbody.velocity.y);
        else
        {
            //根据攻击类型进行适量的移动(补偿速度)
            if (attackType == "Light")
                rigidbody.velocity = new Vector2(transform.localScale.x * lightSpeed * Time.fixedDeltaTime, rigidbody.velocity.y);
            else if (attackType == "Heavy")
                rigidbody.velocity = new Vector2(transform.localScale.x * heavySpeed * Time.fixedDeltaTime, rigidbody.velocity.y);
        }

        //根据速度方向调整本地缩放方向(实现转向)
        if (rigidbody.velocity.x < 0)
            transform.localScale = new Vector3(-1, 1, 1);
        else if (rigidbody.velocity.x > 0)
            transform.localScale = new Vector3(1, 1, 1);
    }

    /*
     * 触发检测
     * 玩家物体的子物体身上挂载有触发盒子
     * 玩家的攻击动画会动态激活子物体并调整其子物体的区域大小
     * 这里触发检测的就是子物体所触发区域
     */
    private void OnTriggerEnter2D(Collider2D other)
    {
        //触碰敌人
        if (other.CompareTag("Enemy"))
        {
            if (attackType == "Light") //轻击中
            {
                //通知攻击管理进行轻击中暂停
                AttackSense.Instance.HitPause(lightPause);
                //通知攻击管理进行轻击中镜头摇晃
                AttackSense.Instance.CameraShake(shakeTime, lightStrength); 
            }
            else if (attackType == "Heavy") //重击中
            {
                AttackSense.Instance.HitPause(heavyPause);
                AttackSense.Instance.CameraShake(shakeTime, heavyStrength);
            }

            //根据自身方向调整敌人的转向
            if (transform.localScale.x > 0)
                other.GetComponent<Enemy>().GetHit(Vector2.right);
            else if (transform.localScale.x < 0)
                other.GetComponent<Enemy>().GetHit(Vector2.left);
        }
    }
}

将移动放在fixedupdate中,跳跃放在update中。
移动和跳跃都是通过施加力实现的。跳跃有isGround变量限制,必须触地才能跳一次,故按帧执行或者按时间执行都不影响,而按时间执行还可能导致“按键失效”,所以我们放入按帧执行中,即update中;而移动是去持续施加力,在不同帧率的主机上效果不同,所以我们应该放入fixedupdate中,按时间执行

细看代码不难理解,其中,对攻击的触发检测,触发盒挂在在子物体上,我么可以看一下大概思路:

可以看到,打击感的核心交给了AttackSense这个单例的类,算是个攻击管理类(攻击意识);而敌人接收到攻击的响应,则是通过触发器获取敌人对象来设置,这样很好的对Enemy进行了封装,不管什么敌人,统统调用GetHit即可。

攻击意识

下面我们来看看攻击管理的单例类,这是一个挂载到相机上的类:

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

public class AttackSense : MonoBehaviour
{
    private static AttackSense instance;
    public static AttackSense Instance
    {
        get
        {
            if (instance == null)
                instance = Transform.FindObjectOfType<AttackSense>();
            return instance;
        }
    }
    private bool isShake;

    public void HitPause(int duration)
    {
        StartCoroutine(Pause(duration));
    }

    IEnumerator Pause(int duration)
    {
        float pauseTime = duration / 60f;
        Time.timeScale = 0;
        yield return new WaitForSecondsRealtime(pauseTime);
        Time.timeScale = 1;
    }

    public void CameraShake(float duration, float strength)
    {
        if (!isShake)
            StartCoroutine(Shake(duration, strength));
    }

    IEnumerator Shake(float duration, float strength)
    {
        isShake = true;
        Transform camera = Camera.main.transform;
        Vector3 startPosition = camera.position;

        while (duration > 0)
        {
            camera.position = Random.insideUnitSphere * strength + startPosition;
            duration -= Time.deltaTime;
            yield return null;
        }
        camera.position = startPosition;
        isShake = false;
    }
}

很简单

敌人配置

最后我们来看一下Enemy的配置,这里敌人的动画很简单:

 

代码如下:

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

public class Enemy : MonoBehaviour
{
    public float speed;
    private Vector2 direction;
    private bool isHit;

    new private Rigidbody2D rigidbody;

    private AnimatorStateInfo info;
    private Animator animator;

    void Start()
    {
        //得到自身动画机
        animator = transform.GetComponent<Animator>();
        //得到刚体
        rigidbody = transform.GetComponent<Rigidbody2D>();
    }

    void Update()
    {
        //得到自身动画机动画状态信息
        info = animator.GetCurrentAnimatorStateInfo(0);
        if (isHit)
        {
            //受到攻击回退
            rigidbody.velocity = direction * speed; 
            if (info.normalizedTime >= .6f)
                isHit = false;
        }
    }

    public void GetHit(Vector2 direction)
    {
        //与受攻击方向同步方向
        transform.localScale = new Vector3(-direction.x, 1, 1); 
        isHit = true;
        //重置方向,便于回退Update中
        this.direction = direction;
        //同步动画状态
        animator.SetTrigger("Hit");
    }
}

结语

基本配置完成,打击感满满

 

敌方AI

玩家配置

玩家的配置很简单,就是正常的移动跳跃,需要更高级的配置在后面可以自己添加:

 

敌人配置

我们重点来看敌人配置:

在Enemy下有两个子物体:

  • Area:一个长条的空物体。触发器,模拟敌人的检测视野
  • AttackArea:一个空物体(一个点)。将在脚本中搭配一个半径值来构成一个圆形区域,作为敌人施展攻击的触发区域

在Enemy上,绑定着脚本FSM(有限状态机),其内容我已注释:

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

//定义枚举类型
public enum StateType
{
    //待机、巡逻、追逐、反应、攻击、被攻击、死亡
    Idle, Patrol, Chase, React, Attack, Hit, Death
}

[Serializable] //可序列化-在检视面板可以设置
public class Parameter
{
    //生命值
    public int health;
    //巡逻速度
    public float moveSpeed;
    //追逐速度
    public float chaseSpeed;
    //待机时长(巡逻到头后待机)
    public float idleTime;
    //巡逻范围
    public Transform[] patrolPoints;
    //追逐范围
    public Transform[] chasePoints;
    //目标
    public Transform target;
    //目标所在层
    public LayerMask targetLayer;
    //攻击位置点
    public Transform attackPoint;
    //攻击位置点的半径(结合攻击位置点规划出一个攻击触发区)
    public float attackArea;
    //动画机
    public Animator animator;
    //是否受攻击
    public bool getHit;
}
//定义有限状态机类
public class FSM : MonoBehaviour
{
    //当前状态
    private IState currentState;
    //状态字典
    private Dictionary<StateType, IState> states = new Dictionary<StateType, IState>();
    //实例化一个parameter对象
    public Parameter parameter;
    void Start()
    {
        //给字典添加各个枚举类型对应的各个对象
        states.Add(StateType.Idle, new IdleState(this));
        states.Add(StateType.Patrol, new PatrolState(this));
        states.Add(StateType.Chase, new ChaseState(this));
        states.Add(StateType.React, new ReactState(this));
        states.Add(StateType.Attack, new AttackState(this));
        states.Add(StateType.Hit, new HitState(this));
        states.Add(StateType.Death, new DeathState(this));

        //初始化状态为Idle状态(转换为Idle状态)
        TransitionState(StateType.Idle);

        //初始化参数中的动画机
        parameter.animator = transform.GetComponent<Animator>();
    }

    void Update()
    {
        //当前状态更新
        currentState.OnUpdate();

        //模拟受到攻击(按下回车)
        if (Input.GetKeyDown(KeyCode.Return))
        {
            //参数对象设置受到攻击
            parameter.getHit = true;
        }
    }

    /* 状态转换
     * 
     */
    public void TransitionState(StateType type)
    {
        //当前已有状态,做退出工作
        if (currentState != null)
            currentState.OnExit();
        //更改当前状态
        currentState = states[type];
        //做进入状态工作
        currentState.OnEnter();
    }
    /* 翻转
     * 根据目标位置转向
     */
    public void FlipTo(Transform target)
    {
        if (target != null)
        {
            if (transform.position.x > target.position.x)
            {
                transform.localScale = new Vector3(-1, 1, 1);
            }
            else if (transform.position.x < target.position.x)
            {
                transform.localScale = new Vector3(1, 1, 1);
            }
        }
    }
    //碰到玩家触发(触发器是子物体Area)
    private void OnTriggerEnter2D(Collider2D other)
    {
        if (other.CompareTag("Player"))
        {
            parameter.target = other.transform;
        }
    }
    //退出触发
    private void OnTriggerExit2D(Collider2D other)
    {
        if (other.CompareTag("Player"))
        {
            parameter.target = null;
        }
    }

    private void OnDrawGizmos()
    {
        Gizmos.DrawWireSphere(parameter.attackPoint.position, parameter.attackArea);
    }
}

可以看到,原作者是通过状态机来实现AI的,将各个状态封装成对象,全部继承自接口:

public interface IState
{
    void OnEnter();

    void OnUpdate();

    void OnExit();
}

下面我们来一个个看,代码上我就省略头文件的引用了:

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

待机状态:

/*
 * 待机状态
 */
public class IdleState : IState
{
    //一个有限状态机
    private FSM manager;
    //一个参数汇总对象
    private Parameter parameter;

    private float timer;
    //初始化
    public IdleState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        //参数对象的动画机播放待机动画(待机动画是循环的)
        parameter.animator.Play("Idle");
    }

    public void OnUpdate()
    {
        //计时器累积(单位秒)
        timer += Time.deltaTime;

        //待机状态被打
        if (parameter.getHit)
        {
            //转换状态到Hit
            manager.TransitionState(StateType.Hit);
        }
        //发现目标且目标距离在追捕范围内
        if (parameter.target != null &&
            parameter.target.position.x >= parameter.chasePoints[0].position.x &&
            parameter.target.position.x <= parameter.chasePoints[1].position.x)
        {
            //进入反应状态
            manager.TransitionState(StateType.React);
        }
        //计时器超过待机时间
        if (timer >= parameter.idleTime)
        {
            //转换回巡逻状态
            manager.TransitionState(StateType.Patrol);
        }
    }

    public void OnExit()
    {
        //计时器归零
        timer = 0;
    }
}

巡逻状态:

/*
 * 巡逻状态
 */
public class PatrolState : IState
{
    private FSM manager;
    private Parameter parameter;

    private int patrolPosition;
    //初始化
    public PatrolState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        //播放走路动画
        parameter.animator.Play("Walk");
    }

    public void OnUpdate()
    {
        //根据巡逻位置转向
        manager.FlipTo(parameter.patrolPoints[patrolPosition]);
        //位移
        manager.transform.position = Vector2.MoveTowards(manager.transform.position,
            parameter.patrolPoints[patrolPosition].position, parameter.moveSpeed * Time.deltaTime);

        if (parameter.getHit)
        {
            manager.TransitionState(StateType.Hit);
        }
        if (parameter.target != null &&
            parameter.target.position.x >= parameter.chasePoints[0].position.x &&
            parameter.target.position.x <= parameter.chasePoints[1].position.x)
        {
            manager.TransitionState(StateType.React);
        }
        if (Vector2.Distance(manager.transform.position, parameter.patrolPoints[patrolPosition].position) < .1f)
        {
            manager.TransitionState(StateType.Idle);
        }
    }

    public void OnExit()
    {
        //巡逻点下标更新
        patrolPosition++;

        if (patrolPosition >= parameter.patrolPoints.Length)
        {
            patrolPosition = 0;
        }
    }
}

追捕状态:

/*
 * 追捕状态
 */
public class ChaseState : IState
{
    private FSM manager;
    private Parameter parameter;

    public ChaseState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        //播放走路动画(Walk动画循环)
        parameter.animator.Play("Walk");
    }

    public void OnUpdate()
    {
        //根据目标位置转向
        manager.FlipTo(parameter.target);
        //有目标则一直走
        if (parameter.target)
            manager.transform.position = Vector2.MoveTowards(manager.transform.position,
            parameter.target.position, parameter.chaseSpeed * Time.deltaTime);

        if (parameter.getHit)
        {
            manager.TransitionState(StateType.Hit);
        }
        if (parameter.target == null ||
            manager.transform.position.x < parameter.chasePoints[0].position.x ||
            manager.transform.position.x > parameter.chasePoints[1].position.x)
        {
            manager.TransitionState(StateType.Idle);
        }
        if (Physics2D.OverlapCircle(parameter.attackPoint.position, parameter.attackArea, parameter.targetLayer))
        {
            manager.TransitionState(StateType.Attack);
        }
    }

    public void OnExit()
    {

    }
}

反应状态:

/*
 * 反应状态
 */
public class ReactState : IState
{
    private FSM manager;
    private Parameter parameter;

    //得到动画状态
    private AnimatorStateInfo info;
    public ReactState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        parameter.animator.Play("React");
    }

    public void OnUpdate()
    {
        //得到当前动画状态
        info = parameter.animator.GetCurrentAnimatorStateInfo(0);

        if (parameter.getHit)
        { 
            manager.TransitionState(StateType.Hit);
        }
        //播放动画快结束,则进入追捕状态
        if (info.normalizedTime >= .95f)
        {
            manager.TransitionState(StateType.Chase);
        }
    }

    public void OnExit()
    {

    }
}

攻击状态

/*
 * 攻击状态
 */
public class AttackState : IState
{
    private FSM manager;
    private Parameter parameter;

    private AnimatorStateInfo info;
    public AttackState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        //进入攻击动画
        parameter.animator.Play("Attack");
    }

    public void OnUpdate()
    {
        info = parameter.animator.GetCurrentAnimatorStateInfo(0);

        if (parameter.getHit)
        {
            manager.TransitionState(StateType.Hit);
        }
        if (info.normalizedTime >= .95f)
        {
            manager.TransitionState(StateType.Chase);
        }
    }

    public void OnExit()
    {

    }
}

受攻击状态:

/*
 * 受攻击状态
 */
public class HitState : IState
{
    private FSM manager;
    private Parameter parameter;

    private AnimatorStateInfo info;
    public HitState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        //播放动画,
        parameter.animator.Play("Hit");
        //掉血
        parameter.health--;
    }

    public void OnUpdate()
    {
        info = parameter.animator.GetCurrentAnimatorStateInfo(0);

        if (parameter.health <= 0)
        {
            //死亡
            manager.TransitionState(StateType.Death);
        }
        if (info.normalizedTime >= .95f)
        {
            parameter.target = GameObject.FindWithTag("Player").transform;

            manager.TransitionState(StateType.Chase);
        }
    }

    public void OnExit()
    {
        parameter.getHit = false;
    }
}

死亡状态:

/*
 * 死亡状态
 */
public class DeathState : IState
{
    private FSM manager;
    private Parameter parameter;

    public DeathState(FSM manager)
    {
        this.manager = manager;
        this.parameter = manager.parameter;
    }
    public void OnEnter()
    {
        parameter.animator.Play("Dead");
    }

    public void OnExit()
    {
        throw new System.NotImplementedException();
    }

    public void OnUpdate()
    {
        throw new System.NotImplementedException();
    }
}

结语

功能基本完成。

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

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

相关文章

广东电子MES系统在电子厂实施的功能和流程

1、电子行业特点电子行业为典型的离散性加工企业&#xff0c;其管理核心的问题在于SN号的管理和物料追溯&#xff0c;即产品、半成品、关键部件都有SN号&#xff0c;且需要实现物料追溯。2、电子行业MES解决方案针对行业需求&#xff0c;我们提供了如下的解决方案&#xff1a;采…

【Python模块】日期时间

在平时开发工作中&#xff0c;我们经常需要用到日期时间&#xff0c;比如日志记录、日期时间的计算、时间字段的赋值等。Python 提供了 time 模块、datatime 模块及子模块、calendar 模块等内置模块&#xff0c;可实现对日期时间的设置、获取、转换等常见操作。 一、日期时间的…

基于STM32G431嵌入式学习笔记——七、定时器定时

一、题目引入 上述为第13届蓝桥杯省赛节选内容&#xff0c;为了研究定时器的机理并独立书写计时函数&#xff0c;上述内容简化为以下要求&#xff1a; ①按下B4按键&#xff0c;LD1点亮5s后熄灭 ②按下B3按键&#xff0c;LD2以0.1秒为间隔切换亮灭状态 二、基础知识 定时器相…

请问各位程序员,是我的思维方式有错误吗?

你好呀&#xff0c;我是歪歪。 前几天知乎给我推送了一个问题&#xff0c;我点进去一看&#xff0c;好家伙&#xff0c;竟然把我血压都看上来了是怎么回事。 我先把问题复述一遍&#xff0c;聊天记录比较长&#xff0c;但是看的过程中容易冲动&#xff0c;注意控制情绪&#…

麦芽糖-聚乙二醇-人血清白蛋白,HAS-PEG-maltose

中文名称&#xff1a;麦芽糖-人血清白蛋白 英文名称&#xff1a;maltose-HAS 别称&#xff1a;人血清白蛋白修饰麦芽糖&#xff0c;HAS-麦芽糖 PEG接枝修饰麦芽糖 麦芽糖-聚乙二醇-人血清白蛋白 HAS-PEG-maltose 人血清白蛋白-PEG-麦芽糖 纯度&#xff1a;95% 存储条件…

物料搬运装置及控制系统设计(CAD+PLC)

目 录 1 绪论 1 1.1 物料搬运装置的概述 1 1.2 物料搬运装置的发展史 1 1.3 气动技术及气动物料搬运装置的发展过程 2 1.4 物料搬运装置未来的发展趋势 3 1.5 本课题研究内容 6 1.6 课题研究的意义 6 2 物料搬运装置的总体设计方案 7 2.1 物料搬运装置的工作原理及系统组成 7 2…

动态规划 DP专题

跟着ygg的dp题单刷的dp 1.代码源每日一题 Div1 连续子序列 分析&#xff1a; dp数组开成map&#xff0c;则状态转移式dp[i] max(dp[i - 1] 1, dp[i]) AC代码&#xff1a; #include <bits/stdc.h>using namespace std; typedef long long ll; #define int ll #define …

Day814.电商系统表设计优化案例分析 -Java 性能调优实战

电商系统表设计优化案例分析 Hi&#xff0c;我是阿昌&#xff0c;今天学习记录的是关于电商系统表设计优化案例分析。 如果在业务架构设计初期&#xff0c;表结构没有设计好&#xff0c;那么后期随着业务以及数据量的增多&#xff0c;系统就很容易出现瓶颈。 如果表结构扩展…

CentOS7.9 安装配置MySQL5.7.30

1.下载mysql安装包&#xff1a;mysql-5.7.30-linux-glibc2.12-x86_64.tar.gz 2. 卸载自带的mariadb和mysql 检查是否安装了mariadb和mysql&#xff0c;有时候默认安装了 rpm -qa | grep mariadb rpm -qa | grep mysql 如果没有&#xff0c;就可以安装mysql&#xff0c;如果有&a…

Centos7搭建SVN代码控制服务器

Centos7搭建SVN代码控制服务器检查SVN是否安装创建SVN版本库配置代码库设置允许访问远程仓库的用户帐号密码设置权限控制设置SVN服务配置启动svn与停止启动SVN关闭SVN访问拉取远程仓库代码检查SVN是否安装 1、centos7系统自带SVN rpm -qa subversion2、如果没有则通过yum安装 …

[论文精读|博士论文]面向文本数据的关系抽取关键技术研究

电子科技大学 2022.3.15博士论文 实体关系的方向性语义缺失&#xff0c;使得关系的判别缺乏对文本蕴含语义特征的利用 提出——基于句法关系的方向敏感型句子级关系抽取算法 利用依存句法树结构信息构建双向依存路径结构&#xff08;新的文本策略解决过度剪裁&#xff09;额外…

构建直接序列扩频系统模型(Matlab代码实现)

&#x1f352;&#x1f352;&#x1f352;欢迎关注&#x1f308;&#x1f308;&#x1f308; &#x1f4dd;个人主页&#xff1a;我爱Matlab &#x1f44d;点赞➕评论➕收藏 养成习惯&#xff08;一键三连&#xff09;&#x1f33b;&#x1f33b;&#x1f33b; &#x1f34c;希…

【分析笔记】全志 i2c-sunxi.c 控制器驱动分析

分析平台&#xff1a;全志 A64 内核版本&#xff1a;Linux 4.9 数据手册&#xff1a;Allwinner_A64_User_Manual_V1.1.pdf (whycan.com) 驱动框架 I2C 设备驱动 作为方案应用来说&#xff0c;我们是最经常要动的地方&#xff0c;这一层主要与具体的芯片功能强关联&#xff0c…

RK3588平台开发系列讲解(USB篇)内核 USB 配置

平台内核版本安卓版本RK3588Linux 5.10Android 12文章目录 一、USB PHY CONFIG二、USB Host CONFIG三、USB OTG CONFIG四、USB Gadget CONFIG沉淀、分享、成长,让自己和他人都能有所收获!😄 📢本篇将介绍USB的相关配置。 USB 模块的配置及保存和其它内核模块的配置方法一…

英语考试的作文模板

考试需要&#xff0c;但是老是忘记&#xff0c;所以发出来备忘~~~~~ 这里写目录标题1 高频考点&#xff1a;正面话题 (能力/事情/习惯/行为/品质/意识/习惯&#xff09;1.1 题干关键词与结构1.2 开头段1.3 主体段2 一起写范文【正面话题】主体段&#xff1a;重要性怎么做重要性…

【AI学习笔记】jupyter notebook 默认路径修改(超简介,超详细)

文章目录修改前&#xff1a;修改notebook默认路径&#xff1a;1. 找到 Anaconda 的安装目录2. 修改 notebook 安装位置3. 删除"%USERPROFILE%/"内容修改后&#xff1a;【声明&#xff1a;由于我的电脑有 Anaconda3的root环境 和 名为TensorFlow 的 Anaconda虚拟环境&…

操作系统学习笔记(Ⅱ):进程

目录 1 进程 1.1 定义、组成、组织方式与特征 1.定义 2.组成 ​3.组织方式 4.进程的特征 1.2 进程的状态与转换 1.状态 2.进程状态的转换 1.3 进程控制 1.基本概念 2.进程控制相关的原语 1.4 进程通信 1.共享存储 2.消息传递 3.管道通信 1.5 线程概念和多线…

Python数据分析实战-实现模型K折交叉验证(附源码和实现效果)

前面我介绍了可视化的一些方法以及机器学习在预测方面的应用&#xff0c;分为分类问题&#xff08;预测值是离散型&#xff09;和回归问题&#xff08;预测值是连续型&#xff09;&#xff08;具体见之前的文章&#xff09;。 从本期开始&#xff0c;我将做一个数据分析类实战…

mybatisplus savebatch 多数据源时候,sqlSessionFactory 不正确踩坑记录。

记录一下 mybatis-plus sharding-JDBC 的时候&#xff0c;因为配置多数据源和多个SqlSessionFactory导致 mybatisPlus 执行 saveBatch 异常的问题。 具体异常就是 saveBatch 执行的数据源&#xff0c;与期望的不一致。其实是因为 SqlSessionFactory 错误导致的。 项目中有2个…

程序员第一次接私活?记住这三点让你事半功倍

不少程序员都有接私活的想法&#xff0c;但恰恰就如开发者之间的论调一样&#xff0c;接私活其实是有一定难度的&#xff0c;想找到合适的单子&#xff0c;顺利地做完并拿到薪水&#xff0c;需要注意的事儿很多&#xff0c;接下来和大家分享一下&#xff0c;程序员第一次接私活…