IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 游戏开发 -> [Unity] GameFramework 学习记录 3 -> 正文阅读

[游戏开发][Unity] GameFramework 学习记录 3

首先要仿照飞船的创建流程创建一个主角

首先直接导入 Starter Assets - Third Person Character Controller
太爽了

我看看啊
他这个是除了 ThirdPersonController 之外还有一个处理刚体碰撞的,一个玩家跟随摄像机,一个 InputSystem 相关的
我在想能不能把他们搬到框架里面


我在看战机的时候,看这两行

IDataTable<DRAircraft> dtAircraft = GameEntry.DataTable.GetDataTable<DRAircraft>();
DRAircraft drAircraft = dtAircraft.GetDataRow(TypeId);

我就一直在想,为什么 GetDataRow 输入的是一个 TypeId
我乍一看上去还以为是,输入行号 TypeId,获得数据表的一行
所以我就奇怪,为什么 typeid 会对应行
因为战机的数据表里面有很多行嘛,那按照我这么想的话,就感觉是,一个表的很多行对应很多 typeid,但是一个战机怎么会在内部有很多 typeid 呢

之后才发现,原来他这个 GetDataRow 原来说的就是获取一整个 DR+数据表名 的类的实例啊hhh


哦我在整理流程的时候最后还是溯源到了一个接口的实例
之前我一直都被这个实例难住,因为我只看得到这个实例,用转到定义也看不到接口中的函数的实现
然后我就不知道怎么看了
之后去网上搜了一下

在这里插入图片描述

他一说构造函数,我才想起来要看这个接口的实例的初始化hhhh

比如我现在一路转到定义,转到了

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs

        public void ShowEntity(int entityId, Type entityLogicType, string entityAssetName, string entityGroupName, int priority, object userData)
        {
            if (entityLogicType == null)
            {
                Log.Error("Entity type is invalid.");
                return;
            }

            m_EntityManager.ShowEntity(entityId, entityAssetName, entityGroupName, priority, ShowEntityInfo.Create(entityLogicType, userData));
        }

然后我就想看 m_EntityManager.ShowEntity,但是如果直接对 ShowEntity() 转到定义,会看到接口类中的函数定义,却看不到实现

C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\832e6a5ddbb7461ba47bec9b9ce43ea6\IEntityManager.cs

    //
    // Summary:
    //     实体管理器接口。
    public interface IEntityManager
    {
        //
        // Summary:
        //     显示实体。
        //
        // Parameters:
        //   entityId:
        //     实体编号。
        //
        //   entityAssetName:
        //     实体资源名称。
        //
        //   entityGroupName:
        //     实体组名称。
        void ShowEntity(int entityId, string entityAssetName, string entityGroupName);

        //
        // Summary:
        //     显示实体。
        //
        // Parameters:
        //   entityId:
        //     实体编号。
        //
        //   entityAssetName:
        //     实体资源名称。
        //
        //   entityGroupName:
        //     实体组名称。
        //
        //   priority:
        //     加载实体资源的优先级。
        void ShowEntity(int entityId, string entityAssetName, string entityGroupName, int priority);

        //
        // Summary:
        //     显示实体。
        //
        // Parameters:
        //   entityId:
        //     实体编号。
        //
        //   entityAssetName:
        //     实体资源名称。
        //
        //   entityGroupName:
        //     实体组名称。
        //
        //   userData:
        //     用户自定义数据。
        void ShowEntity(int entityId, string entityAssetName, string entityGroupName, object userData);

        //
        // Summary:
        //     显示实体。
        //
        // Parameters:
        //   entityId:
        //     实体编号。
        //
        //   entityAssetName:
        //     实体资源名称。
        //
        //   entityGroupName:
        //     实体组名称。
        //
        //   priority:
        //     加载实体资源的优先级。
        //
        //   userData:
        //     用户自定义数据。
        void ShowEntity(int entityId, string entityAssetName, string entityGroupName, int priority, object userData);

这个时候应该看 m_EntityManager 的初始化

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs

        protected override void Awake()
        {
            base.Awake();

            m_EntityManager = GameFrameworkEntry.GetModule<IEntityManager>();

往下找

C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\53ef5f794854447fae0b6697b6c091ae\GameFrameworkEntry.cs

        //
        // Summary:
        //     获取游戏框架模块。
        //
        // Type parameters:
        //   T:
        //     要获取的游戏框架模块类型。
        //
        // Returns:
        //     要获取的游戏框架模块。
        //
        // Remarks:
        //     如果要获取的游戏框架模块不存在,则自动创建该游戏框架模块。
        public static T GetModule<T>() where T : class
        {
            Type typeFromHandle = typeof(T);
            if (!typeFromHandle.IsInterface)
            {
                throw new GameFrameworkException(Utility.Text.Format("You must get module by interface, but '{0}' is not.", typeFromHandle.FullName));
            }

            if (!typeFromHandle.FullName.StartsWith("GameFramework.", StringComparison.Ordinal))
            {
                throw new GameFrameworkException(Utility.Text.Format("You must get a Game Framework module, but '{0}' is not.", typeFromHandle.FullName));
            }

            string text = Utility.Text.Format("{0}.{1}", typeFromHandle.Namespace, typeFromHandle.Name.Substring(1));
            return GetModule(Type.GetType(text) ?? throw new GameFrameworkException(Utility.Text.Format("Can not find Game Framework module type '{0}'.", text))) as T;
        }

其实之前也找到过这里,但是脑子里没有一个思路,就没有深究
现在的思路就是,他是用来初始化接口实例的
然后这里把它叫做 module,嗯……很神奇

这个函数就是检查输入的类型是否是接口,是否是框架内的模块,都是的话才进行下一步
下一步是

C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\53ef5f794854447fae0b6697b6c091ae\GameFrameworkEntry.cs

        //
        // Summary:
        //     获取游戏框架模块。
        //
        // Parameters:
        //   moduleType:
        //     要获取的游戏框架模块类型。
        //
        // Returns:
        //     要获取的游戏框架模块。
        //
        // Remarks:
        //     如果要获取的游戏框架模块不存在,则自动创建该游戏框架模块。
        private static GameFrameworkModule GetModule(Type moduleType)
        {
            foreach (GameFrameworkModule s_GameFrameworkModule in s_GameFrameworkModules)
            {
                if ((object)s_GameFrameworkModule.GetType() == moduleType)
                {
                    return s_GameFrameworkModule;
                }
            }

            return CreateModule(moduleType);
        }

遍历已知的模块的列表,没有就创建
这个 s_GameFrameworkModules 是没有别的引用的,那就说明模块唯一的来源就是 CreateModule(moduleType)
那么我要知道接口实例的接口函数是怎么实现的,我还是要看这个模块是怎么创建的

C:\Users\18221\AppData\Local\Temp\MetadataAsSource\7f73a0cffcfb4b64ae957e64e6ab0bfe\53ef5f794854447fae0b6697b6c091ae\GameFrameworkEntry.cs

        //
        // Summary:
        //     创建游戏框架模块。
        //
        // Parameters:
        //   moduleType:
        //     要创建的游戏框架模块类型。
        //
        // Returns:
        //     要创建的游戏框架模块。
        private static GameFrameworkModule CreateModule(Type moduleType)
        {
            GameFrameworkModule gameFrameworkModule = (GameFrameworkModule)Activator.CreateInstance(moduleType);
            if (gameFrameworkModule == null)
            {
                throw new GameFrameworkException(Utility.Text.Format("Can not create module '{0}'.", moduleType.FullName));
            }

            LinkedListNode<GameFrameworkModule> linkedListNode = s_GameFrameworkModules.First;
            while (linkedListNode != null && gameFrameworkModule.Priority <= linkedListNode.Value.Priority)
            {
                linkedListNode = linkedListNode.Next;
            }

            if (linkedListNode != null)
            {
                s_GameFrameworkModules.AddBefore(linkedListNode, gameFrameworkModule);
            }
            else
            {
                s_GameFrameworkModules.AddLast(gameFrameworkModule);
            }

            return gameFrameworkModule;
        }

后面是插入优先级链表,创建的部分是在前面 GameFrameworkModule gameFrameworkModule = (GameFrameworkModule)Activator.CreateInstance(moduleType);
哦好吧我记得我之前确实是看到了这个Activator.CreateInstance,然后我之后不知道这个 C# 的 Activator.CreateInstance 是怎么创建出实现了接口函数的

好吧,现在的问题就是
它有一个 接口类的实例,是用 Activator.CreateInstance 创建的
那么我怎么知道这个实例创建出来之后,他对接口的实现是什么

我在网上看到的 Activator.CreateInstance 的作用是 使用与指定参数匹配程度最高的构造函数来创建指定类型的实例
那我应该是找构造函数才对,说不定我就找到了真正创建的那个类

但是那个创建类的代码是只读的,难顶
就是说我还不能加一个打印的代码
这样的话我就看不到他这个到底是输入了个什么函数
我还不想加断点
先试试慢慢分析把

要想知道向 Activator.CreateInstance 输入了什么,源文件只读,但是仍然可以仿照 GetModule 里面的函数,来得到输入的字符串。比如说在流程里面复现一遍

System.Type typeFromHandle = typeof(GameFramework.Entity.IEntityManager);
string text = GameFramework.Utility.Text.Format("{0}.{1}", typeFromHandle.Namespace, typeFromHandle.Name.Substring(1));
GameFramework.GameFrameworkLog.Debug("!!!!!!!!!!!!!" + text);

这里打印的是 GameFramework.Entity.EntityManager

……哦我悟了,原来这就是源码的部分,这就需要看别人 dll 里面的内容了!
太顶了,我去问别人,别人说看源码,我在工程里面找半天不知道啥是源码

就是这个!
https://github.com/EllanJiang/GameFramework/blob/master/GameFramework/Entity/EntityManager.cs

GameFramework/Entity/EntityManager.cs

        /// <summary>
        /// 显示实体。
        /// </summary>
        /// <param name="entityId">实体编号。</param>
        /// <param name="entityAssetName">实体资源名称。</param>
        /// <param name="entityGroupName">实体组名称。</param>
        /// <param name="priority">加载实体资源的优先级。</param>
        /// <param name="userData">用户自定义数据。</param>
        public void ShowEntity(int entityId, string entityAssetName, string entityGroupName, int priority, object userData)
        {
        	//---
        	
            EntityInstanceObject entityInstanceObject = entityGroup.SpawnEntityInstanceObject(entityAssetName);
            if (entityInstanceObject == null)
            {
                int serialId = ++m_Serial;
                m_EntitiesBeingLoaded.Add(entityId, serialId);
                m_ResourceManager.LoadAsset(entityAssetName, priority, m_LoadAssetCallbacks, ShowEntityInfo.Create(serialId, entityId, entityGroup, userData));
                return;
            }

            InternalShowEntity(entityId, entityAssetName, entityGroup, entityInstanceObject.Target, false, 0f, userData);

GameFramework/Entity/EntityManager.cs

        private void InternalShowEntity(int entityId, string entityAssetName, EntityGroup entityGroup, object entityInstance, bool isNewInstance, float duration, object userData)
        {
            try
            {
                IEntity entity = m_EntityHelper.CreateEntity(entityInstance, entityGroup, userData);
                if (entity == null)
                {
                    throw new GameFrameworkException("Can not create entity in entity helper.");
                }

                EntityInfo entityInfo = EntityInfo.Create(entity);
                m_EntityInfos.Add(entityId, entityInfo);
                entityInfo.Status = EntityStatus.WillInit;
                entity.OnInit(entityId, entityAssetName, entityGroup, isNewInstance, userData);
                entityInfo.Status = EntityStatus.Inited;
                entityGroup.AddEntity(entity);
                entityInfo.Status = EntityStatus.WillShow;
                entity.OnShow(userData);
                entityInfo.Status = EntityStatus.Showed;

                if (m_ShowEntitySuccessEventHandler != null)
                {
                    ShowEntitySuccessEventArgs showEntitySuccessEventArgs = ShowEntitySuccessEventArgs.Create(entity, duration, userData);
                    m_ShowEntitySuccessEventHandler(this, showEntitySuccessEventArgs);
                    ReferencePool.Release(showEntitySuccessEventArgs);
                }
            }
            catch (Exception exception)
            {
                if (m_ShowEntityFailureEventHandler != null)
                {
                    ShowEntityFailureEventArgs showEntityFailureEventArgs = ShowEntityFailureEventArgs.Create(entityId, entityAssetName, entityGroup.Name, exception.ToString(), userData);
                    m_ShowEntityFailureEventHandler(this, showEntityFailureEventArgs);
                    ReferencePool.Release(showEntityFailureEventArgs);
                    return;
                }

                throw;
            }
        }

原来在这里控制了 OnInit() OnShow(),还有 ShowEntity 的事件

要看最开头的 m_EntityHelper.CreateEntity 又感觉有点难办,这也是一个接口

GameFramework/Entity/EntityManager.cs

        /// <summary>
        /// 设置实体辅助器。
        /// </summary>
        /// <param name="entityHelper">实体辅助器。</param>
        public void SetEntityHelper(IEntityHelper entityHelper)
        {
            if (entityHelper == null)
            {
                throw new GameFrameworkException("Entity helper is invalid.");
            }

            m_EntityHelper = entityHelper;
        }

在框架源码里面没找到 SetEntityHelper 的引用
居然是在游戏工程里面的

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs

    public sealed partial class EntityComponent : GameFrameworkComponent
    {
		[SerializeField]
        private string m_EntityHelperTypeName = "UnityGameFramework.Runtime.DefaultEntityHelper";
		[SerializeField]
        private EntityHelperBase m_CustomEntityHelper = null;

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs

        private void Start()
        {
        	// ...
        	EntityHelperBase entityHelper = Helper.CreateHelper(m_EntityHelperTypeName, m_CustomEntityHelper);
			m_EntityManager.SetEntityHelper(entityHelper);

要知道 m_EntityHelper.CreateEntity 就要知道 m_EntityHelper 巴拉巴拉
哎呀,我突然想到或许直接写出关系更简单

m_EntityHelper.CreateEntity <- m_EntityHelper <- Helper.CreateHelper

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Utility\Helper.cs

        /// <summary>
        /// 创建辅助器。
        /// </summary>
        /// <typeparam name="T">要创建的辅助器类型。</typeparam>
        /// <param name="helperTypeName">要创建的辅助器类型名称。</param>
        /// <param name="customHelper">若要创建的辅助器类型为空时,使用的自定义辅助器类型。</param>
        /// <param name="index">要创建的辅助器索引。</param>
        /// <returns>创建的辅助器。</returns>
        public static T CreateHelper<T>(string helperTypeName, T customHelper, int index) where T : MonoBehaviour
        {
            T helper = null;
            if (!string.IsNullOrEmpty(helperTypeName))
            {
                System.Type helperType = Utility.Assembly.GetType(helperTypeName);
                if (helperType == null)
                {
                    Log.Warning("Can not find helper type '{0}'.", helperTypeName);
                    return null;
                }

                if (!typeof(T).IsAssignableFrom(helperType))
                {
                    Log.Warning("Type '{0}' is not assignable from '{1}'.", typeof(T).FullName, helperType.FullName);
                    return null;
                }

                helper = (T)new GameObject().AddComponent(helperType);
            }
            else if (customHelper == null)
            {
                Log.Warning("You must set custom helper with '{0}' type first.", typeof(T).FullName);
                return null;
            }
            else if (customHelper.gameObject.InScene())
            {
                helper = index > 0 ? Object.Instantiate(customHelper) : customHelper;
            }
            else
            {
                helper = Object.Instantiate(customHelper);
            }

            return helper;
        }

又是一个动态创建的
由于 EntityHelper 是定义好的,所以他应该会返回一个添加了 UnityGameFramework.Runtime.DefaultEntityHelper 组件的 GameObject
嗯……那我还是要看源码中的 UnityGameFramework.Runtime.DefaultEntityHelper
哦不用,这个就是工程中的

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\DefaultEntityHelper.cs

        public override IEntity CreateEntity(object entityInstance, IEntityGroup entityGroup, object userData)
        {
            GameObject gameObject = entityInstance as GameObject;
            if (gameObject == null)
            {
                Log.Error("Entity instance is invalid.");
                return null;
            }

            Transform transform = gameObject.transform;
            transform.SetParent(((MonoBehaviour)entityGroup.Helper).transform);

            return gameObject.GetOrAddComponent<Entity>();
        }

啊……好像出乎意料地简单
不过在 helper 之前就已经创建了 GameObject 的嘛
我得看看是哪里创建了

GameFramework/Entity/EntityManager.cs

			EntityInstanceObject entityInstanceObject = entityGroup.SpawnEntityInstanceObject(entityAssetName);
            if (entityInstanceObject == null)
            {
                int serialId = ++m_Serial;
                m_EntitiesBeingLoaded.Add(entityId, serialId);
                m_ResourceManager.LoadAsset(entityAssetName, priority, m_LoadAssetCallbacks, ShowEntityInfo.Create(serialId, entityId, entityGroup, userData));
                return;
            }

一开始的 entityGroup.SpawnEntityInstanceObject(entityAssetName) 会返回 m_InstancePool.Spawn(name)
这看上去像是从对象池中生成,如果池中没有对象……呃好像不对 entityInstanceObject == null 之后就直接返回了,那么如果到了 InternalShowEntity 的话那必然是要有 entityInstanceObject
那还是要继续看 entityInstanceObject 是怎么生成的

GameFramework\Entity\EntityManager.EntityGroup.cs

            public EntityInstanceObject SpawnEntityInstanceObject(string name)
            {
                return m_InstancePool.Spawn(name);
            }

要看 m_InstancePool 的初始化

GameFramework\Entity\EntityManager.EntityGroup.cs

            public EntityGroup(string name, float instanceAutoReleaseInterval, int instanceCapacity, float instanceExpireTime, int instancePriority, IEntityGroupHelper entityGroupHelper, IObjectPoolManager objectPoolManager)
            {
				m_InstancePool = objectPoolManager.CreateSingleSpawnObjectPool<EntityInstanceObject>(Utility.Text.Format("Entity Instance Pool ({0})", name), instanceCapacity, instanceExpireTime, instancePriority);
                

objectPoolManager 是输入的参数,又要看参数是哪来的,所以要看 哪里调用了 EntityGroup()

GameFramework\Entity\EntityManager.cs

        public bool AddEntityGroup(string entityGroupName, float instanceAutoReleaseInterval, int instanceCapacity, float instanceExpireTime, int instancePriority, IEntityGroupHelper entityGroupHelper)
        {
            m_EntityGroups.Add(entityGroupName, new EntityGroup(entityGroupName, instanceAutoReleaseInterval, instanceCapacity, instanceExpireTime, instancePriority, entityGroupHelper, m_ObjectPoolManager));

输入到 EntityGroup() 的参数是 m_ObjectPoolManager,又要看 m_ObjectPoolManager 的初始化

GameFramework\Entity\EntityManager.cs

        public void SetObjectPoolManager(IObjectPoolManager objectPoolManager)
        {
            if (objectPoolManager == null)
            {
                throw new GameFrameworkException("Object pool manager is invalid.");
            }

            m_ObjectPoolManager = objectPoolManager;
        }

要看 m_ObjectPoolManager 的初始化,又回到了游戏工程

Assets\GameFramework\UnityGameFramework-b2d2ef63517d2cab5f0def57e691a2d794b6a7f5\Scripts\Runtime\Entity\EntityComponent.cs

        private void Start()
        {
        	m_EntityManager.SetObjectPoolManager(GameFrameworkEntry.GetModule<IObjectPoolManager>());

那么还是要看回游戏框架中的 ObjectPoolManager
回顾一下,我要找的是 CreateSingleSpawnObjectPool 方法

在这里,他们都会跳转到 InternalCreateObjectPool

GameFramework\ObjectPool\ObjectPoolManager.cs

        private ObjectPoolBase InternalCreateObjectPool(Type objectType, string name, bool allowMultiSpawn, float autoReleaseInterval, int capacity, float expireTime, int priority)
        {
            Type objectPoolType = typeof(ObjectPool<>).MakeGenericType(objectType);
            ObjectPoolBase objectPool = (ObjectPoolBase)Activator.CreateInstance(objectPoolType, name, allowMultiSpawn, autoReleaseInterval, capacity, expireTime, priority);
            m_ObjectPools.Add(typeNamePair, objectPool);
            return objectPool;
        }

哦……只是根据对象的类型创造了相应类型的 ObjectPoolBase

那再往回看,我还要看对象池的 Spawn 方法
但是只在 GameFramework\ObjectPool\ObjectPoolBase.cs 中找到 AllowMultiSpawn 方法
怪了,我要跳转回……

哦我知道了,在框架里面使用转到实现就能看到了
我靠我刚刚在干嘛,我记得是在游戏工程里转到定义没作用,但是刚刚在源码里可以
习惯性地没有尝试过转到定义
难受

GameFramework\ObjectPool\ObjectPoolManager.ObjectPool.cs

            public T Spawn(string name)
            {
                if (name == null)
                {
                    throw new GameFrameworkException("Name is invalid.");
                }

                GameFrameworkLinkedListRange<Object<T>> objectRange = default(GameFrameworkLinkedListRange<Object<T>>);
                if (m_Objects.TryGetValue(name, out objectRange))
                {
                    foreach (Object<T> internalObject in objectRange)
                    {
                        if (m_AllowMultiSpawn || !internalObject.IsInUse)
                        {
                            return internalObject.Spawn();
                        }
                    }
                }

                return null;
            }

GameFramework\ObjectPool\ObjectPoolManager.Object.cs

            public T Spawn()
            {
                m_SpawnCount++;
                m_Object.LastUseTime = DateTime.UtcNow;
                m_Object.OnSpawn();
                return m_Object;
            }

哦……生成数量加1,最近一次使用时间设置为当前时间,然后调用生成回调,然后返回 m_Object

然后又要看 m_Object 的初始化……啊……累了
或许应该设断点来看

在这里插入图片描述

the breakpoint will not currently be hit. Unable to find a corresponding location

哦我记起来我为什么一直没有用调试
我之前尝试过调试,但是我设的断点不会停的……令人抓狂

看了一下别人的问题,感觉是 visual studio 的问题
https://forum.unity.com/threads/vs2015-the-breakpoint-will-not-currently-be-hit-unable-to-find-corresponding-location.454924/

好吧,我去下了一个 jetbrain rider 运行之后,代码原封不动,确实能够打到断点了,nb

在这里插入图片描述

但是他还是只是在调用接口的地方,嗯,就直接步进下去了,没有进到 m_EntityManager.ShowEntity 的里面
麻了,时间成本好高,已经不想再看了……

  游戏开发 最新文章
6、英飞凌-AURIX-TC3XX: PWM实验之使用 GT
泛型自动装箱
CubeMax添加Rtthread操作系统 组件STM32F10
python多线程编程:如何优雅地关闭线程
数据类型隐式转换导致的阻塞
WebAPi实现多文件上传,并附带参数
from origin ‘null‘ has been blocked by
UE4 蓝图调用C++函数(附带项目工程)
Unity学习笔记(一)结构体的简单理解与应用
【Memory As a Programming Concept in C a
上一篇文章      下一篇文章      查看所有文章
加:2022-03-16 22:53:21  更:2022-03-16 22:53:55 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 17:50:03-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码