Python面试记录

某一次

  1. Python 的 initnew 的区别

  2. Python 的内存重用机制

  3. Python 的协程是什么

  4. Python 的基本类型有哪些

  5. Python 的上下文是怎么实现的

  6. WSGI/ASGI 是什么

  7. MySQL 的索引原理是什么

  8. MySQL 的四种隔离级别

  9. MySQL 的事务原理是什么
    Atomicity(原子性):事务是一个不可分割的操作单元,要么全部成功,要么全部失败。
    Consistency(一致性):事务执行后,数据库从一个一致状态转换到另一个一致状态。
    Isolation(隔离性):多个事务并发执行时,彼此隔离,互不干扰。
    Durability(持久性):事务提交后,对数据库的修改是永久性的,即使系统崩溃也不会丢失。

某一次

  1. MySQL 的关键字的执行顺序是什么

    1. FROM 和 JOIN
      首先确定查询的数据来源,包括表和连接操作。

    2. WHERE
      过滤符合条件的行。

    3. GROUP BY
      按指定列分组。

    4. HAVING
      对分组后的结果进行过滤。

    5. SELECT
      选择要返回的列,并可以应用聚合函数或表达式。

    6. ORDER BY
      对结果进行排序。

    7. LIMIT 和 OFFSET
      限制返回的行数或跳过指定行。

  2. /login 符合RESTful 规范嘛

  3. Oauth 的四种模式
    隐式
    密码
    客户端
    授权码

某一次

  1. Python 的异步原理
  2. Python 的GC, Java 的GC
  3. 怎么查询中位数
  4. ExecuteService 的处理请求是 maxPoolSize 和 queue 逻辑
  5. 反向传播的原理

某一次

  1. 生成器和迭代器的区别

rsync笔记

背景

之前尝试在openwrt上上启动anything, 实际体验是会把设置直接弄崩,现象是ssh都连不进去,所以也没法debug, 只能手动同步了, ssh 又不支持增量传,所以使用了rsync

1
rsync -avz -e ssh /mnt/share-media/ pi:/mnt/share-media
1
rsync -avz -e ssh /www/luci-static/argon/background/ pi:/www/luci-static/argon/background/

2024在powershell中给桌面端软件签名

背景

使用Electron 打的包直接安装会有提醒, 网上很多方式是使用cmd 的, 我补充一下powershell的
reason

解决步骤

  1. 在服务商买 代码签名证书, 注意不是常见的 SSL证书

这里我先使用自签名证书了

设置环境变量

1
2
3
4
5
6
7
8
9
# 密码
$certPassword = ConvertTo-SecureString -String "123456" -Force -AsPlainText
# 文件
$fileToSign= "D:\niumag\niumagfiber\NiumagSeed-linux-4.1.3-beta.610-x64.exe"
# 执行文件入口
$signtoolPath = "C:\Program Files (x86)\Windows Kits\10\bin\10.0.26100.0\x64\signtool.exe"
# 时间服务器
$timestampServer = "http://timestamp.digicert.com"

创建根证书并导出

1
2
3
4
# 创建
$cert = New-SelfSignedCertificate -Type CodeSigningCert -Subject "CN=AmanoCert" -KeyUsage DigitalSignature -KeyAlgorithm RSA -KeyLength 2048 -CertStoreLocation "Cert:\CurrentUser\My"
# 导出
Export-PfxCertificate -Cert $cert -FilePath "AmanoCert4.pfx" -Password $certPassword

签名

1
& "$signtoolPath" sign /debug -f AmanoCert4.pfx  -p 123456 /fd SHA256 /tr $timestampServer /td SHA256 -a $fileToSign

效果

签名效果

alt text

属性效果

alt text

重装机兵重构4-功能篇

上一篇重装机兵重构2-地图篇的后续

Day24

  1. 感谢 Jax’s Studio 的无私奉献, 复用背包系统
    inventory

Day25

  1. 背包系统是使用 UGUI实现的, 利用RenderTexture 渲染在 UI Document里面

Day26

  1. RenderTexture 之后的背包不能点击, 使用UGUI重新画了一个tab, 然后把背包重新放到 UGUI里面

Day27

  1. 开源的背包物品交换有bug,直接购买了 Ultimate Inventory System, 竟然还赠送商店系统, 准备直接站在巨人肩膀上

Day28/29/30

  1. 研究Ultimate Inventory System

MySQL表锁为什么不会出现死锁

截止 2025-02-20, 大语言模型还有进步空间哈

我的回答

首先固定一下上下文,不讨论表锁和行锁形成的死锁, 只讨论表锁与表锁之间。
目前的答案是不会形成死锁,原因官方已经说了 解锁
简单的说就是lock表A并想lock表B的时候会隐式的触发表A的解锁,所以不会存在死锁, 如果想同时锁住A和B, 需要这样

1
LOCK TABLES job WRITE, user WRITE;

快照

证明1 X表确实时互斥的

  1. 创建User 和Job 表, 每个表有 id 和name 字段
    效果

  2. 锁住 user 并查询锁的情况
    会话1

  3. 给 user 表 增加列 age, 发现存在锁竞争, 这个DDL pending 住了
    会话2

  4. 解锁 user 表, 发现age 修改好了
    会话1

会话2

结果

证明2 隐式解锁

  1. 锁住 user
    会话1

  2. 给user 增加 gender, 发现卡住
    会话2

  3. 锁住 job 表
    会话1

  4. 发现 user 的改动也结束了, 说明user表确实解锁了
    会话2

效果

补充一下大模型的回答

DeepSeek 回答的不正确

alt text

OpenAI 通过引导可以回答正确

alt text

重装机兵重构3-剧情篇

上一篇重装机兵重构2-地图篇的后续

Day16

  1. 认识一些像素人物的生成工具;
  2. 增加持久化的能力,基于json 的文件的 SaveSystem的实现

Day17

  1. 认识 Unity 的 Timeline
  2. 认识 Timeline 的 Playable Track, Signal Track

Day18

  1. 利用 ProcessFrame 实现 Timeline 中两个 playableAsset 的暂停,等待用户输入才继续播放

  2. 认识 Timeline 的 Animation Track,实现人物移动 + 动画播放
    效果图

  3. 修复 PlayableDirector.Pause 会导致人物会回到初始地点的bug:

    1
    2
    //director.Pause();
    director.playableGraph.GetRootPlayable(0).Pause();

Day19

  1. 处理TextMeshPro的中文乱码:
    从windows-> TextMeshPro -> Font Asset Creator 创建没有效果
    从*.ttf -> creat ->TextMeshPro ->Font Asset -> SPF 这样可以
  2. 简单过场黑屏动画 天亮了 使用 UI -> Panel + UI -> TextMeshPro 实现
  3. 封装打字机,过场动画 天亮了 也可以使用

Day20

  1. 解决Visual Studio 的中文在Unity 乱码的问题:
    在VS里面安装 扩展 -> 管理扩展, 搜索Force UTF-8(No BOM) 2022
  2. 增加逻辑, 开始游戏 -> 三选一角色 + 用户名称 -> 存档
  3. 增加逻辑, 记载游戏 -> 选择存档 -> 进入游戏

Day21

  1. 实现gameObject 在 UI Document 的展示(通过将GameObject 渲染到 RenderTexture)

  2. 实现Player的prefab, 根据职业选择渲染不同的人物

Day22

  1. 根据玩家选择的职业, 动态更新Timeline中animation track的人物的gameobject, 以及更新animation track 中的 animation clip

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    private void UpdateAnimationTrack(PlayableDirector director)
    {
    var playerObject = GameObject.FindWithTag("Player");
    Animator playerAnimator = playerObject.GetComponentInChildren<Animator>();
    Debug.Log("playerAnimator is " + playerAnimator);
    foreach (var track in director.playableAsset.outputs)
    {
    if (track.streamName == "PlayerPosition" || track.streamName == "PlayerAnimation")
    {
    if (track.streamName == "PlayerAnimation")
    {
    UpdateAnimationClip(track, playerAnimator);
    }
    director.SetGenericBinding(track.sourceObject, playerAnimator);
    }
    }
    }
    private void UpdateAnimationClip(PlayableBinding track, Animator animator)
    {
    if (track.sourceObject is AnimationTrack animationTrack)
    {
    // 获取该 Track 上的所有绑定(这里是 Animation Clip)
    foreach (var clip in animationTrack.GetClips())
    {
    // 替换现有的 Animation Clip
    foreach (var newClip in animator.runtimeAnimatorController.animationClips)
    {
    if (clip.displayName == newClip.name)
    {
    clip.asset = newClip; // 将新的 Animation Clip 赋值给 Track
    break;
    }
    }
    }
    }

    }

  2. 增加postTimeline 脚本, 实现剧情中人物的销毁和常规人物的位置重置

  3. 在camera上面增加audio source, *.m4a 需要转成 *.wav

    1
    ffmpeg -i input.m4a output.wav

Day23

  1. 打字机支持加速(左键跳过)
  2. 利用UI Docuemnt 实现tab
    tabs

Day24

  1. 实现menu 的隐藏出现,menu里面包含退出,实现游戏的开始->退出->开始的闭环:
    注意UIDocument 的隐藏不要使用SetActive(false), 这样会导致按钮的事件丢失

至此, 素材篇基本结束了,后面进入功能篇

重装机兵重构2-地图篇

上一篇重装机兵重构的后续

Day12

  1. 认识Aseprite, 一个Animated Sprite Editor & Pixel Art Tool。这个工具可以快速把真实的画转成像素风
  2. 认识Tiled, 比Unity自带的Palette 多了一个地形集,画地图更方便
  3. 认识SuperTiled2Unity 可以把Tiled 的地图放入unity里面
  4. 把多个Aseprite 合并成一个
    1
    2
    3
    4
    C:\Users\amanoooo\Desktop\Aseprite\aseprite --batch building-2.aseprite building3.aseprite building4.aseprite building5.aseprite building6.aseprite building7.aseprite building8.aseprite --sheet buildings.png


    C:\Users\amanoooo\Desktop\Aseprite\aseprite --batch cliff1.aseprite cliff2.aseprite cliff3.aseprite garbge.aseprite grass.aseprite grass2.aseprite hedge.aseprite hedge2.aseprite hedge-left.aseprite shizijia.aseprite stone.aseprite tree1.aseprite water1.aseprite sand.aseprite 三个点.aseprite 宝箱1.aseprite 地板1.aseprite 地板2.aseprite 地板3.aseprite 工作台.aseprite 平台.aseprite 油桶1.aseprite 油桶2.aseprite --sheet envs.png```

Day13

  1. 在Tiled 上面增加对象层,增加对象, 然后unity 使用脚本实现把对象改成IsTrigger ,实现门碰撞逻辑, 通过SuperCustomProperties 来获取门的名称, 实现不同门的跳转. 脚本如下

DoorManager 放在空对象上

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
using UnityEngine;
using SuperTiled2Unity;

public class DoorManager : MonoBehaviour
{
void Start()
{
// 获取所有的 SuperObject 组件

// 使用 FindObjectsByType 替代 FindObjectsOfType
SuperObject[] superObjects = FindObjectsByType<SuperObject>(FindObjectsSortMode.None);


foreach (SuperObject superObject in superObjects)
{
// 检查对象是否有碰撞器
Collider2D collider = superObject.GetComponent<Collider2D>();
if (collider != null)
{
// 动态添加 TriggerDoorNamePrinter 脚本
superObject.gameObject.AddComponent<DoorHandler>();
}
}
}
}

DoorHandler

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
using UnityEngine;
using SuperTiled2Unity;

[RequireComponent(typeof(Collider2D))] // 确保对象有 Collider2D 组件
public class DoorHandler : MonoBehaviour
{
private SuperCustomProperties customProperties;

void Start()
{
// 获取 SuperCustomProperties 组件
customProperties = GetComponent<SuperCustomProperties>();

// 确保碰撞器是触发器
Collider2D collider = GetComponent<Collider2D>();
if (collider != null)
{
collider.isTrigger = true;
}

// 打印 doorName 的值
PrintDoorName();
}

void OnTriggerEnter2D(Collider2D other)
{
// 当触发器被触发时打印 doorName
PrintDoorName();
}

private void PrintDoorName()
{
if (customProperties != null)
{
// 检查是否有 doorName 属性
if (customProperties.TryGetCustomProperty("doorName", out CustomProperty property))
{
Debug.Log($"Door Name: {property.m_Value}", this);
}
else
{
Debug.LogWarning("No doorName property found on this object.", this);
}
}
else
{
Debug.LogWarning("No SuperCustomProperties component found on this object.", this);
}
}
}

Day14

  1. 实现不同场景人物的初始化位置:
    在Awake 里面查找自定属性 IsEntry
  2. fix迭代更新tmx对应的tsx对应的png(补充素材),出现的问题是unity的地图素材乱了,和tiled的不一致,通过更新tsx的高度修正bug

Day15

  1. 认识Tiled editor 增加类
  2. 实现多场景多入口的切换:
    进入IsDoor 的时候, 检查DoorName(场景名) ,检查 DoorIndex, 然后切换到对应的 Scene,查找EntryIndex==DoorIndex的 IsEntry 的gameObject
  3. 临时处理人物位移 x+0.5, y-0.5

Day16

  1. 处理bug:
    老版本的MetalMax有个逻辑是,场景A->B通过楼梯, B->A也是通过楼梯, 要求是到B的初始化状态是在楼梯上, 这时候按照我们的写法, B会因为OnTriggerEnter2D 立即切换到A, 解决方案是在 IsEntry 上面也增加一个Trigger, 只有 OnTriggerExit2D 才能是IsDoor 的trigger启用
    代码如下:
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138
    139
    140
    141
    142
    143
    144
    145
    146
    147
    148
    149
    150
    151
    152
    153
    154
    155
    156
    157
    158
    159
    160
    161
    162
    163
    using UnityEngine;
    using SuperTiled2Unity;
    using static UnityEngine.EventSystems.EventTrigger;

    [RequireComponent(typeof(Collider2D))] // 确保对象有 Collider2D 组件
    public class DoorHandler : MonoBehaviour
    {
    private SuperCustomProperties customProperties;
    private bool disableByEntry = false;


    void Awake()
    {
    // 获取 SuperCustomProperties 组件
    customProperties = GetComponent<SuperCustomProperties>();

    var isDoor = IsDoor();
    if (IsDoor())
    {
    // 确保碰撞器是触发器
    Collider2D collider = GetComponent<Collider2D>();
    if (collider != null)
    {
    collider.isTrigger = true;
    }
    }

    // 检查是否是入口门
    if (IsEntry())
    {

    Collider2D collider = GetComponent<Collider2D>();
    if (!isDoor)
    {
    collider.enabled = false;
    }

    if (GetEntryIndex() == SceneLoader.Instance.DoorIndex)
    {
    if (isDoor)
    {
    disableByEntry = true;
    }
    MovePlayerToThisDoor();
    }
    }
    }

    private void Update()
    {
    }


    void OnTriggerEnter2D(Collider2D other)
    {
    if (!disableByEntry)
    {
    SceneLoader.Instance.SwitchScene($"mm/scenes/{GetDoorName()}");
    } else
    {
    Debug.Log("ignore scene loader due to disableByEntry: " + disableByEntry);
    }
    }
    void OnTriggerExit2D(Collider2D other)
    {
    disableByEntry = false;
    }

    private int GetDoorIndex()
    {
    if (customProperties.TryGetCustomProperty("DoorIndex", out CustomProperty DoorIndex))
    {
    if (DoorIndex != null)
    {
    return DoorIndex.m_Value.ToInt();
    }
    }
    return 0;
    }

    private int GetEntryIndex()
    {
    if (customProperties.TryGetCustomProperty("EntryIndex", out CustomProperty Index))
    {
    if (Index != null)
    {
    return Index.m_Value.ToInt();
    }
    }
    return 0;
    }

    private string GetDoorName()
    {
    if (customProperties != null)
    {
    // 检查是否有 doorName 属性
    if (customProperties.TryGetCustomProperty("DoorName", out CustomProperty DoorName))
    {
    var DoorIndex = GetDoorIndex();
    Debug.Log($"Door Name: {DoorName.m_Value} Index:{DoorIndex}", this);
    SceneLoader.Instance.DoorIndex = DoorIndex;
    return DoorName.m_Value;
    }
    else
    {
    Debug.LogWarning("No doorName property found on this object.", this);
    return null;
    }
    }
    else
    {
    Debug.LogWarning("No SuperCustomProperties component found on this object.", this);
    return null;
    }
    }

    private bool IsEntry()
    {
    if (customProperties != null)
    {
    // 检查是否有 IsEntry 属性
    if (customProperties.TryGetCustomProperty("IsEntry", out CustomProperty property))
    {
    return property.m_Value == "true";
    }
    }
    return false;
    }

    private bool IsDoor()
    {
    if (customProperties != null)
    {
    // 检查是否有 IsEntry 属性
    if (customProperties.TryGetCustomProperty("IsDoor", out CustomProperty property))
    {
    return property.m_Value == "true";
    }
    }
    return false;
    }

    private void MovePlayerToThisDoor()
    {
    // 查找玩家对象
    GameObject player = GameObject.FindGameObjectWithTag("Player");
    Debug.Log($"player is {player}");
    if (player != null)
    {
    Vector3 alignedPosition = new Vector3(
    transform.position.x + 0.5f,
    transform.position.y - 0.5f,
    transform.position.z);
    // 将玩家移动到门的位置
    player.transform.position = alignedPosition;
    }
    else
    {
    Debug.LogWarning("Player not found in the scene.", this);
    }
    }
    }

效果

至此, 素材篇基本结束了,后面进入剧情篇

unity对话框打印机特效

分享一下我的对话框打印机特效

上下文

uGUI 已经变成弃用了,我使用的最近 unity 6 的 UI Document。

小镇Town 场景有一个 TownUI 的 UIDocument, TownUI里面有一个类似 Prefab 的Dialog 的UIDocuement, Dialog 里面是一个VisualElement 作为背景, 里面是ScrollView, 然后里面是Label, 你会看到我的代码里也是这么查询的。

TownHandler 变成static 是方便在变得场景中随时调用

showStr 变成public 是为了支持可以在 inspector 那边随时更改

实现

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
using UnityEngine;
using UnityEngine.UIElements;
using System.Collections;

public class TownHandler : MonoBehaviour
{

public static TownHandler instance { get; private set; }



public float displayTime = 4.0f;
private VisualElement m_NonPlayerDialogue;
private float m_TimerDisplay;

public string showStr; //需要打出来的字
private Label label; //打字展示的文本
private ScrollView scrollView;




// Awake is called when the script instance is being loaded (in this situation, when the game scene loads)
private void Awake()
{
instance = this;
}


// Start is called once before the first execution of Update after the MonoBehaviour is created
void Start()
{

UIDocument uiDocument = GetComponent<UIDocument>();
//m_Healthbar = uiDocument.rootVisualElement.Q<VisualElement>("HealthBar");
//SetHealthValue(1.0f);


m_NonPlayerDialogue = uiDocument.rootVisualElement.Q<VisualElement>("Dialog");
m_NonPlayerDialogue.style.display = DisplayStyle.None;
m_TimerDisplay = -1.0f;


scrollView = m_NonPlayerDialogue.Q<ScrollView>();
label = scrollView.Q<Label>(); // 使用 Label 的名称

Debug.LogWarning("label is " + label);
}

// Update is called once per frame
void Update()
{
//if (m_TimerDisplay > 0)
//{
// m_TimerDisplay -= Time.deltaTime;
// if (m_TimerDisplay < 0)
// {
// m_NonPlayerDialogue.style.display = DisplayStyle.None;
// }


//}
}

public void DisplayDialogue()
{
Debug.Log("DisplayDialogue");
m_NonPlayerDialogue.style.display = DisplayStyle.Flex;
m_TimerDisplay = displayTime;

StartCoroutine(PrintWord());

}

IEnumerator PrintWord()
{

foreach (char letter in showStr)
{
label.text += letter; // 添加一个字符
yield return new WaitForSeconds(0.1f); // 模拟打印速度,调整时间来控制速度


scrollView.scrollOffset = new Vector2(0, label.resolvedStyle.height); // 设置滚动位置到最底部
}



}

}

重装机兵重构

背景

看到 《大灾变:黑暗之日》 竟然还在维护,突然很感动,想到了重装机兵在国内应该也有一定的受众。于是,再次开始游戏吧。

过程

Day1

  1. 安装 Unity Hub
  2. 按照这个视频一步步学习简单的2d物理引擎 https://learn.unity.com/tutorial/playground-get-started-on-your-first-game?language=en#

Day2

按照这个教程开始2d游戏的高阶基础 https://learn.unity.com/tutorial/set-up-tilemap-collision?uv=2022.3&courseId=64774201edbc2a1638d25d18&projectId=6477424bedbc2a1473e5fce4# ,

人物可以移动,可以和sprite碰撞, 也可以和tilemap 碰撞

  1. 认识 Main window
  2. 认识 Context Menu
  3. 认识 Hierarchy Menu
  4. 认识 Project Menu
  5. 认识 Inspector Menu
  6. 认识 物理引擎 Rigidbody 2D
  7. 认识 碰撞器 Box Collider 2D
  8. 认识 Grid + TileMap
  9. 认识 Palette (tileSet)
  10. 认识 Rect Tool
  11. 认识 prefab 的作用

Day3

继续昨天的教程, 人物移动可以扣血回血, 并在血条上体现, enemy 的巡逻
  1. 认识 Box Collider 2D 的is Trigger 会导致可以穿透,并有事件
  2. 认识 Box Collider 2D 的 auto Tiling
  3. 认识 UI Builder(UI Document)
  4. 认识 VisualElement
  5. PlayerController 的碰撞脚本和静态的 UIHandler 进行血量交互
  6. 认识 Animator
  7. 认识 Animation
  8. 认识 Blend Tree

Day4

  1. Blend Tree 增加 transtion
  2. 认识 Layers (用于发射子弹)
  3. 认识 Edit > Project Settings… > Physics 2D -> the Layer Collision Matrix
  4. 认识 Raycasting
  5. 遇到了子弹射出去不移动的问题
  6. 遇到了子弹从enemy身上穿过去的问题
  7. 实现了对话弹出ui
  8. 认识cinimachine (摄像头碰撞)

Day5

  1. 认识 Audio Source
  2. 认识 Audio Source 的 Spatial Blend
  3. 认识 Effects -> Particle System
  4. 认识 粒子的 Radius 和 angle
  5. 如何手搓动画, 主要是关键帧
  6. Editor -> Project Setting -> Player 设置分辨率,版本
  7. File -> Building Profile -> 设置scene -> Build And Run

Day6

  1. 使用 UIDocument 增加游戏入口 Main , 监听 Enter 利用 SceneManager 切换到另一个场景
  2. unity store 购买免费的 DOTWEEN 插件, 然后实现场景切换时候的fade动画

Day7

  1. 复习实现 NPC 的弹窗对话
  2. 文字的打字机特效
  3. 自动滚动到ScrollView 的最下面

Day8

  1. 修复刚进入场景是 animator 迟缓(关闭Transitions 的 has Exit time)
  2. 在Moving(Blend Tree) 的基础上增加 Idle 的 Blend Tree
  3. 把6张真人的照片 去除背景, 然后处理成 像素图
  4. 学习战斗系统 英文教程, 在b站上看的盗版的
  5. 认识 Rigidbody 2D 的 Body Type: Dynamic / Kinematic / Static
  6. 认识 Rigidbody 2D 的 Collistion Detection: Discrete / Continuous

2025春节

Day9

  1. 参考b站上的中文教程
  2. 认识 Physics2D.OverlapCircle, 实现仇恨范围, 敌人自动追踪
  3. 认识 Cinemachine vs Camera Folllow

Day10

  1. 认识PlayerInput(Script) 和 InputAction 的区别
  2. 认识跨脚本调用,除了使用static 方法, 还可以使用 gameObject.BroadcastMessage()
  3. 在animation里面插入关键帧, 实现动画打击(假设是4帧, 就是1,2 帧 disable, 3帧enable, 4帧disable)

Day11

  1. 利用Collider 的 isTriger实现击退效果, 被伤害动画
  2. 实现死亡效果(死亡的动画插入event, 然后Destory掉 gameObject)
  3. 认识 TextMeshPro, 实现漂浮文字
  4. 实现漂浮文字的暴击效果, 动画效果

至此,动画相关除了持久化都已经测试结果,开始地图篇

docker 容器重启追踪

背景

无意中看到 docker ps 显示容器重启了, 虽然服务正常,还是准备查询一下

步骤

确认到服务重启

找到上一个停止的容器的日志

查看日志

发现容器是正常停止的, 基本排除服务器重启

确认机器没有重启

查看docker日志

可以发现是node 的状态从new 变成了down

查看系统日志

监控到软件oom 了

1
journalctl -e


可以在aliyun 的监控看到当时cpu有增加,但是只有75%

结论

prometheus 的容器oom导致的docker node 节点的down, 但还不知道为什么会导致我的traefik 的容器重启,暂时的解决方式是给 prometheus 加上资源限制