【行为型之观察者模式】游戏开发实战——Unity事件驱动架构的核心实现策略
文章目录
- 🎯 观察者模式(Observer Pattern)深度解析
- 一、模式本质与核心价值
- 二、经典UML结构
- 三、Unity实战代码(玩家血量监控系统)
- 1. 定义观察者接口与主题基类
- 2. 实现具体主题(玩家血量)
- 3. 实现具体观察者
- 4. 客户端使用
- 四、模式进阶技巧
- 1. 事件总线系统
- 2. 条件过滤器
- 3. 异步事件处理
- 五、游戏开发典型应用场景
- 六、性能优化策略
- 七、模式对比与选择
- 八、最佳实践原则
- 九、常见问题解决方案
🎯 观察者模式(Observer Pattern)深度解析
——以Unity实现动态事件通知与跨系统响应为核心案例
一、模式本质与核心价值
核心目标:
✅ 建立对象间的一对多依赖,实现状态变化自动通知
✅ 解耦主题与观察者,提升系统扩展性与维护性
✅ 支持动态订阅机制,灵活管理观察关系
关键术语:
- Subject(主题):状态变化的对象(如玩家血量)
- Observer(观察者):监听状态变化的对象(如UI、成就系统)
- Notification(通知):主题向观察者传递的信息
数学表达:
设主题S有观察者集合O={o₁, o₂, …, oₙ},当S变化时:
S.Update() → ∀o ∈ O, o.OnNotify()
二、经典UML结构
三、Unity实战代码(玩家血量监控系统)
1. 定义观察者接口与主题基类
public interface IObserver {void OnNotify(HealthData data);
}public interface ISubject {void RegisterObserver(IObserver observer);void RemoveObserver(IObserver observer);void NotifyObservers();
}[System.Serializable]
public struct HealthData {public float CurrentHealth;public float MaxHealth;public float DamageAmount;
}
2. 实现具体主题(玩家血量)
public class PlayerHealth : MonoBehaviour, ISubject {[SerializeField] private float maxHealth = 100f;private float _currentHealth;private List<IObserver> _observers = new();void Start() {_currentHealth = maxHealth;}public void TakeDamage(float damage) {_currentHealth = Mathf.Max(_currentHealth - damage, 0);NotifyObservers(new HealthData {CurrentHealth = _currentHealth,MaxHealth = maxHealth,DamageAmount = damage});if(_currentHealth <= 0) Die();}public void RegisterObserver(IObserver observer) => _observers.Add(observer);public void RemoveObserver(IObserver observer) => _observers.Remove(observer);private void NotifyObservers(HealthData data) {foreach(var observer in _observers) {observer.OnNotify(data);}}
}
3. 实现具体观察者
// 血条UI
public class HealthBarUI : MonoBehaviour, IObserver {[SerializeField] private Image healthFill;public void OnNotify(HealthData data) {healthFill.fillAmount = data.CurrentHealth / data.MaxHealth;Debug.Log($"血条更新:{healthFill.fillAmount:P0}");}
}// 成就系统
public class AchievementTracker : MonoBehaviour, IObserver {private int _consecutiveHits;public void OnNotify(HealthData data) {if(data.DamageAmount > 0) {_consecutiveHits++;if(_consecutiveHits >= 3) {Debug.Log("解锁成就:连续受伤三次!");}} else {_consecutiveHits = 0;}}
}
4. 客户端使用
public class GameManager : MonoBehaviour {[SerializeField] private PlayerHealth playerHealth;[SerializeField] private HealthBarUI healthUI;[SerializeField] private AchievementTracker achievementTracker;void Start() {playerHealth.RegisterObserver(healthUI);playerHealth.RegisterObserver(achievementTracker);}void Update() {if(Input.GetKeyDown(KeyCode.Space)) {playerHealth.TakeDamage(10);}}
}
四、模式进阶技巧
1. 事件总线系统
public static class EventBus {private static Dictionary<Type, List<Action<object>>> _handlers = new();public static void Subscribe<T>(Action<T> handler) {Type type = typeof(T);if(!_handlers.ContainsKey(type)) _handlers[type] = new List<Action<object>>();_handlers[type].Add(obj => handler((T)obj));}public static void Publish<T>(T eventData) {Type type = typeof(T);if(_handlers.TryGetValue(type, out var handlers)) {foreach(var h in handlers) h(eventData);}}
}// 使用示例
EventBus.Subscribe<HealthData>(data => {// 处理健康数据...
});
2. 条件过滤器
public class FilteredObserver : IObserver {private Predicate<HealthData> _filter;private Action<HealthData> _action;public FilteredObserver(Predicate<HealthData> filter, Action<HealthData> action) {_filter = filter;_action = action;}public void OnNotify(HealthData data) {if(_filter(data)) _action(data);}
}// 使用示例:仅在血量低于30%时触发
var lowHealthObserver = new FilteredObserver(data => data.CurrentHealth/data.MaxHealth < 0.3f,data => ShowWarning()
);
3. 异步事件处理
public class AsyncEventProcessor : MonoBehaviour {private Queue<HealthData> _eventQueue = new();public void QueueEvent(HealthData data) {_eventQueue.Enqueue(data);}void Update() {while(_eventQueue.Count > 0) {StartCoroutine(ProcessEvent(_eventQueue.Dequeue()));}}private IEnumerator ProcessEvent(HealthData data) {// 异步处理逻辑yield return null;}
}
五、游戏开发典型应用场景
-
成就系统触发
public class AchievementSystem : IObserver {public void OnNotify(EnemyDeathData data) {if(data.EnemyType == EnemyType.Boss) {UnlockAchievement("BOSS_SLAYER");}} }
-
全局事件通知
public class GlobalEvent : ISubject {private static GlobalEvent _instance;public static GlobalEvent Instance => _instance ??= new GlobalEvent();// 实现观察者注册/通知逻辑... }
-
技能冷却系统
public class SkillManager : IObserver {public void OnNotify(SkillEventData data) {if(data.EventType == SkillEventType.Used) {StartCooldown(data.SkillID);}} }
-
环境互动反馈
public class EnvironmentFX : IObserver {public void OnNotify(PlayerMoveData data) {if(data.IsInWater) PlayWaterRippleFX(data.Position);} }
六、性能优化策略
策略 | 实现方式 | 适用场景 |
---|---|---|
事件过滤 | 前置条件检查 | 高频事件 |
批处理 | 合并多个事件 | 物理系统更新 |
对象池 | 重用事件对象 | 频繁事件触发 |
分层处理 | 优先级队列 | 关键事件优先 |
七、模式对比与选择
维度 | 观察者模式 | 发布-订阅模式 |
---|---|---|
耦合度 | 观察者直接注册到主题 | 通过中间件解耦 |
灵活性 | 需要知道具体主题 | 无需知道发布者 |
性能 | 直接调用更高效 | 中间件可能引入开销 |
典型应用 | 组件间直接通信 | 系统级事件管理 |
八、最佳实践原则
- 避免过度通知:仅在状态真正变化时触发通知
private float _previousHealth;void Update() {if(Mathf.Abs(_currentHealth - _previousHealth) > 0.01f) {NotifyObservers();_previousHealth = _currentHealth;} }
- 内存管理:及时取消不再需要的订阅
void OnDestroy() {playerHealth.RemoveObserver(this); }
- 线程安全:在多线程环境使用锁机制
private readonly object _lock = new object(); public void RegisterObserver(IObserver observer) {lock(_lock) {_observers.Add(observer);} }
- 事件数据不可变:
public readonly struct ImmutableHealthData {public readonly float Current;public ImmutableHealthData(float current) => Current = current; }
九、常见问题解决方案
Q1:如何处理循环通知?
→ 实现事件标记防止递归
private bool _isNotifying;
public void NotifyObservers() {if(_isNotifying) return;_isNotifying = true;// 通知逻辑..._isNotifying = false;
}
Q2:如何优化大量观察者的性能?
→ 使用分层观察者
public class TieredObserverSystem {private Dictionary<EventPriority, List<IObserver>> _tiers = new();public void NotifyByPriority() {foreach(var tier in Enum.GetValues(typeof(EventPriority))) {foreach(var observer in _tiers[(EventPriority)tier]) {observer.OnNotify();}}}
}
Q3:如何调试复杂事件流?
→ 实现事件追踪器
public class EventDebugger : IObserver {public void OnNotify(object data) {Debug.Log($"[Event] {DateTime.Now:HH:mm:ss.fff} - {data.GetType().Name}");// 记录到文件或调试窗口...}
}
上一篇 【行为型之备忘录模式】游戏开发实战——Unity存档系统与状态管理的终极解决方案
下一篇 【行为型之状态模式】深度剖析——Unity角色行为控制与AI决策的终极解决方案
相关文章:
【行为型之观察者模式】游戏开发实战——Unity事件驱动架构的核心实现策略
文章目录 🎯 观察者模式(Observer Pattern)深度解析一、模式本质与核心价值二、经典UML结构三、Unity实战代码(玩家血量监控系统)1. 定义观察者接口与主题基类2. 实现具体主题(玩家血量)3. 实现…...
Java基础语法之数组
数组 一、认识数组 1.什么是数组 数组就是一个容器,用来存一批同种类型的数据。 举例 20, 10, 80, 60, 90 int[] arr {20, 10, 80, 60, 90};张三, 李四, 王五 String[] names {"张三", "李四", "王五"};2.为什么要使用数组 假设…...
Vue3学习(组合式API——计算属性computed详解)
目录 一、计算属性computed。 Vue官方提供的案例。(普通写法与计算属性写法) 使用计算属性computed重构——>简化描述响应式状态的复杂逻辑。 (1)计算属性computed小案例。 <1>需求说明。(筛选原数组——>得新数组) &…...
高海拔和远距离的人员识别:面部、体型和步态的融合
大家读完就觉得有帮助记得关注和点赞!!! 摘要 我们解决了在无约束环境中进行全身人体识别的问题。这个问题出现在诸如IARPA高空和远距离生物识别与身份识别(BRIAR)计划等监视场景中,其中生物识别数据是在长…...
《P2345 [USACO04OPEN] MooFest G》
题目背景 P5094 [USACO04OPEN] MooFest G 加强版 题目描述 约翰的 n 头奶牛每年都会参加“哞哞大会”。 哞哞大会是奶牛界的盛事。集会上的活动很多,比如堆干草,跨栅栏,摸牛仔的屁股等等。 它们参加活动时会聚在一起,第 i 头…...
浅浅学:DoIP工作流程及基于DoIP的诊断/刷写工具
注:阅读本文需要对UDS及BootLoader有一定了解,基础内容不做赘述。 在汽车"新四化"浪潮的推动下,智能座舱、自动驾驶、车路协同等创新技术正加速重构行业格局。随着车载ECU数量突破百个量级,软件代码量呈指数级增长——…...
首个专业AI设计Agent发布-Lovart
Lovart是什么 Lovart 是为设计师打造的世界上首个专业设计 Agent。Lovart 能像专业设计师一样思考和执行设计任务,提供高水平的设计方案。基于自然语言交互,用户能快速调整布局、颜色和构图。Lovart 支持从创意拆解到专业交付的全链路设计,单…...
二叉树(中序遍历)
嘿,欢迎来到小巫blog!小巫又来啦!看到你对二叉树中序遍历这道题有点困惑,别担心,我会一步步带你搞定它!这道题是树的基础题目,掌握了它,你对树的遍历就会有很深的理解。我相信&#…...
Ubuntu 系统默认已安装 python,此处只需添加一个超链接即可
步骤 1:确认 Python 3 的安装路径 查看当前 Python 3 的路径: which python3 输出类似: /usr/bin/python3 步骤 2:创建符号链接 使用 ln -s 创建符号链接,将 python 指向 python3: sudo ln -s /usr/b…...
AcroForm JavaScript Promise 对象应用示例: 异步加载PDF文件
这段代码演示了在Adobe Acrobat DC Pro 的 JavaScript 环境中如何使用 Promise 对象处理异步操作。具体功能是: 定义了一个loadFile函数,模拟异步加载PDF文件的操作使用Promise对象封装异步操作,提供成功(resolve)和失败(reject)两种状态通过…...
LeetCode 热题 100 114. 二叉树展开为链表
LeetCode 热题 100 | 114. 二叉树展开为链表 大家好,今天我们来解决一道经典的二叉树问题——二叉树展开为链表。这道题在 LeetCode 上被标记为中等难度,要求将二叉树展开为一个单链表,展开后的单链表应该与二叉树的先序遍历顺序相同。 问题…...
DML和DQL
1. 设置MySQL的储存引擎 上一章的附录里已经将ini设置好了,不用再次设置 2. DML语句 插入单数据记录 插入多数据记录 将查询结果插入新表 更新数据 删除数据 注意:delete删除只会删除数据,不会重置表的现有逻辑,truncate会重置表逻…...
多线程与线程互斥
我们初步学习完线程之后,就要来试着写一写多线程了。在写之前,我们需要继续来学习一个线程接口——叫做线程分离。 默认情况下,新创建的线程是joinable的,线程退出后,需要对其进行pthread_join操作,否则无法…...
BMS工具箱用来执行贝叶斯模型平均(BMA)计算模块
贝叶斯模型平均(Bayesian Model Averaging,BMA)是一种用于处理模型不确定性的统计方法,通过结合多个模型的预测结果来提高预测的准确性和鲁棒性。在 MATLAB 中,可以使用专门的工具箱(如 BMS 工具箱…...
Java死锁排查:线上救火实战指南
想象一下,你正在值班,突然监控告警红成一片,用户反馈雪花般飘来:“系统卡死了!用不了了!” —— 这很可能就是Java应用遭遇了“死锁”这个大魔王。这时候,你就是救火队长,首要任务不…...
第十九次博客打卡
今天学习的内容是Java中的常见循环。 在 Java 中,常见的循环结构主要有以下几种:for 循环、while 循环、do-while 循环以及增强型 for 循环(也称为 for-each 循环)。 1. for 循环 for 循环是一种非常灵活的循环结构,…...
智能体制作学习笔记1——智能体
01 智能体_哔哩哔哩_bilibili 大语言模型可以理解成一个很厉害的人。 但是要完成一些特定的工作,除了大语言模型,还需要一些工具和业务流程,这样才能自动化帮我们完成特定的工作,这个就叫做智能体。 突然发现放视频的时候出现了…...
Python常见问题
文章目录 1.python有哪些数据类型2.python中的元组和列表的区别是什么?3.python中的break、continue、pass代表什么意思?4.如何在python中生成一个随机数?5.Python有哪些常见的内置函数?6.请用自己最擅长的编程语言,将…...
小程序 存存上下滑动的页面
推荐阅读文档: Vue3组合式API之getCurrentInstance详解 - 且行且思 - 博客园Vue2中,可以通过this来获取当前组件实例; Vue3中,在setup中无法通过this获取组件实例,console.log(this)打印出来的值是undefined。 在Vue3…...
更换git位置并在pycharm中重新配置
更新 PyCharm 中的 Git 路径 更新 PyCharm 终端的 Shell 路径 检查环境变量 确保系统环境变量中的 Path 包含了新的 Git 安装路径 ,如果使用unins0000自动卸载就不会有旧路径。...
AI世界的崩塌:当人类思考枯竭引发数据生态链断裂
AI世界的崩塌:当人类思考枯竭引发数据生态链断裂 ——论过度依赖AI创作对技术进化的反噬 一、数据生态的恶性循环:AI的“自噬危机” 当前AI模型的训练依赖于人类创造的原始数据——书籍、论文、艺术作品、社交媒体动态等。据统计,2025年全球…...
OkHttp连接池
🧰 调整连接池的核心参数 ✅ 最大空闲连接数(maxIdleConnections): 含义:连接池中最多保留的空闲连接数量。默认值:5建议值:10~50(视并发量而定) ✅ 连接保持时间&…...
哈希表的实现01
文章目录 哈希表的实现01哈希概念直接定址法哈希冲突负载因子将关键字转换为整数 哈希函数除法散列法:乘法散列法(了解)全域散列法(了解) 处理哈希冲突(开放定址法)线性探测:二次探测…...
学习日志06 java
还有四天要去比赛了,能赢吗?逼自己一把。。。!!加油! 1 对比一下java重写还是不重写tostring的区别 1. 不重写 toString() 的情况 java class Point {private int x;private int y;public Point(int x, int y) {th…...
spring中的@MapperScan注解详解
一、核心功能与作用 MapperScan是Spring与MyBatis框架集成时用于批量扫描Mapper接口的核心注解,其主要功能包括: 自动注册Mapper接口 通过指定包路径,Spring会自动扫描该路径下的所有Mapper接口,并将其注册为Spring Bean&#x…...
PYTHON训练营DAY25
BUG与报错 一、try else try:# 可能会引发异常的代码 except ExceptionType: # 最好指定具体的异常类型,例如 ZeroDivisionError, FileNotFoundError# 当 try 块中发生 ExceptionType 类型的异常时执行的代码 except: # 不推荐:捕获所有类型的异常&…...
视频图像压缩领域中 DCT 的 DC 系数和 AC 系数详解
引言 在数字图像与视频压缩领域,离散余弦变换(Discrete Cosine Transform, DCT)凭借其卓越的能量集中特性,成为JPEG、MPEG等国际标准的核心技术。DCT通过将空域信号映射到频域,分离出DC系数(直流分量&…...
YOLO v1:目标检测领域的革命性突破
引言 在计算机视觉领域,目标检测一直是一个核心任务,它不仅要识别图像中的物体类别,还要确定物体的精确位置。传统目标检测方法如R-CNN系列虽然准确率高,但计算复杂度高、速度慢。2016年,Joseph Redmon等人提出的YOLO…...
AI智能体 | 使用Coze一键制作“假如书籍会说话”视频,18个作品狂吸17.6万粉,读书博主新标杆!(附保姆级教程)
目录 一、整体工作流设计 二、制作工作流 2.1 开始节点 2.2 大模型_生成对话文案 2.3 代码_字幕切割 2.4 画板_对话背景 2.5 循环_对话语音01 2.5.1 选择器_2 2.5.2 语音合成02 2.5.3 语音合成03 2.5.4 变量聚合_1 2.5.5 视频合成01 2.6 循环_3 2.6.1 选择器_3 …...
HVV蓝队实战面试题
HVV蓝队实战,防守筹备之“部署蜜罐捕获横向扫描行为”。 蜜罐通过模拟内网脆弱服务(如SMB、SSH、数据库端口),诱捕攻击者突破边界后的横向探测行为。 通过监测高频端口扫描、非常规协议请求及非授权IP段遍历,结合多源…...
正则表达式(二)-高级应用_谨慎使用
没事建议别瞎用正则表达式,能让后端处理好的数据,尽量后端处理好,减少前端对数据的处理,保证数据原始的完整性,减少前端耗能。(其实就是懒╮(╯▽╰)╭) 1. 分组捕获 分组捕获用于提取匹配的子字符串,使用 () 定义分组。 示例:提取日期中的年、月、日 (\d{4})-(\d{2…...
在K8S集群中部署EFK日志收集
目录 引言环境准备安装自定义资源部署ElasticsearchMaster 节点与 Data 节点的区别生产优化建议安装好以后测试ES是否正常部署Fluentd测试filebeat是否正常推送日志部署Kibana获取账号密码,账号是:elastic集群测试 引言 系统版本为 Centos7.9内核版本为…...
解决常见数据库问题:保障数据安全与稳定的全方位指南
本文结合行业最佳实践与前沿技术,系统性总结数据库运维中的核心问题与解决方案,助力开发者构建高可靠、高性能的数据服务) 一、性能优化:从SQL到架构的全面调优 性能问题是数据库运维中最常见的挑战,直接影响用户体验…...
武汉科技大学人工智能与演化计算实验室许志伟课题组参加2025中国膜计算论坛
武汉科技大学人工智能与演化计算实验室许志伟课题组参加2025中国膜计算论坛 2025年5月9日至11日,第五届中国膜计算论坛(CWMC 2025)在成都信息工程大学隆重召开。会议由 国际膜计算学会(IMCS) 主办,汇聚了来…...
Femap许可硬件绑定
在电磁仿真领域,Femap软件因其卓越的性能和广泛的应用场景而备受用户青睐。为了确保软件的安全与稳定运行,Femap提供了许可硬件绑定的功能。本文将详细介绍Femap许可硬件绑定的概念和优势,帮助您了解并充分利用这一功能,确保软件在…...
构建优雅对象的艺术:Java 建造者模式的架构解析与工程实践
一、建造者模式的本质与核心价值 在面向对象的软件设计中,创建复杂对象一直是一个需要精心处理的问题。当一个对象的构建需要多个步骤,并且这些步骤具有不同的组合方式时,传统的构造函数方式会显得力不从心。建造者模式(Builder …...
vim启动的时候,执行gg
在 Vim 编辑器中,gg 命令是一个非常有用的命令,它可以将光标快速移动到当前窗口的顶部(即第一行)。如果你想在 Vim 启动时自动执行 gg 命令,有几种方法可以实现这一点: 方法 1:使用 Vim 的启动…...
【SSL部署与优化】HTTP/2与HTTPS的协同效应
HTTP/2与HTTPS的协同效应:为何HTTP/2强制要求TLS 1.2? HTTP/2是HTTP协议的现代升级版,旨在通过多路复用、头部压缩等技术提升性能。然而,HTTP/2的设计与部署与HTTPS(TLS加密)紧密相关,甚至强制…...
JavaScript篇:揭秘函数式与命令式编程的思维碰撞
大家好,我是江城开朗的豌豆,一名拥有6年以上前端开发经验的工程师。我精通HTML、CSS、JavaScript等基础前端技术,并深入掌握Vue、React、Uniapp、Flutter等主流框架,能够高效解决各类前端开发问题。在我的技术栈中,除了…...
ubuntu24.04上安装NVIDIA driver+CUDA+cuDNN+Anaconda+Pytorch
一、NVIDIA driver 使用Ubuntu系统的:软件和更新——>附加驱动,安装NVIDIA驱动。 二、CUDA 安装命令:sudo apt install nvidia-cuda-toolkit 三、cuDNN cuDNN 9.10.0 Downloads | NVIDIA Developer 四、Anaconda Download Anaconda Di…...
vue3基础学习(上) [简单标签] (vscode)
目录 1. Vue简介 2. 创建Vue应用 2.1 下载JS文件 2.2 引用JS文件 2.3 调用Vue方法编辑 2.4 运行一下试试: 2.5 代码如下 3.模块化开发模式 3.1 Live Server插件 3.2 运行 4. 常用的标签 4.1 reactive 4.1.1 运行结果 4.1.2 代码: 4.2 ref 4.2.1 运行结果 4.2.2…...
.Net HttpClient 使用代理功能
HttpClient 使用代理功能 实际开发中,HttpClient 通过代理访问目标服务器是常见的需求。 本文将全面介绍如何在 .NET 中配置 HttpClient 使用代理(Proxy)功能,包括基础使用方式、代码示例、以及与依赖注入结合的最佳实践。 注意…...
深入理解Java适配器模式:从接口兼容到设计哲学
引言:接口不兼容的困局 在软件开发中,我们经常遇到这样的场景: 旧系统有一个「RS232串口设备」(仅支持sendByRS232(String data)方法),新系统需要通过「USB接口」(要求sendByUSB(String data)…...
非异步信号安全函数
这个程序演示了如何使用sigaction来捕获和处理信号(特别是SIGINT,即CtrlC)。以下是关键点和潜在问题的分析: 程序功能 信号捕获:注册自定义处理函数handler来捕获信号2(SIGINT,通常由CtrlC触发…...
PHP黑白胶卷底片图转彩图功能 V2025.05.15
关于底片转彩图 传统照片底片是摄影过程中生成的反色图像,为了欣赏照片,需要通过冲印过程将底片转化为正像。而随着数字技术的发展,我们现在可以使用数字工具不仅将底片转为正像,还可以添加色彩,重现照片原本的色彩效…...
【C++ / STL】封装红黑树实现map和set
文章目录 一. 源码及框架分析1.决定搜索类型的传参思考:为什么要传第一个参数 2.KeyOfValue的作用 二. 模拟实现map和set1. 实现出复用红黑树框架,并支持insert2. 支持iterator的实现iterator实现思路分析【iterator操作实现详解】 3.支持map的[ ]操作4.map和set代码…...
记录: Windows下远程Liunx 系统xrdp 用到的一些小问题(免费踩坑 记录)
采用liunx Ubuntu22.04版本以下,需要安装 xrdp 或者VNC 具体过程就是下载 在linux命令行里 首先更新软件包:sudo apt update 安装xrdp服务:sudo apt install xrdp 启动XRDP:sudo systemctl start xrdp(如果在启动的…...
WordPress 文章和页面:它们的区别是什么?
很多刚接触WordPress的用户,在创建网站内容时往往会遇到这样一个问题:“我应该用‘文章’还是‘页面’?”虽然两者都能发布内容,但它们之间到底有什么区别呢?这篇文章将从易于理解的角度,帮助大家厘清WordP…...
【工具变量】各省市场化指数-杨兴权版共三个方法(1997-2023年)
市场化指数是衡量中国各地区市场化改革进程的重要指标。本次数据基于杨兴全、马连福和夏立军三位学者的研究成果,系统整理并更新了我国1997-2023年间31个省、自治区、直辖市的市场化指数,便于研究者进行横向和纵向比较分析。 一、数据介绍 数据名称&…...
Android App View——团结引擎车机版实现安卓应用原生嵌入 3D 开发场景
团结引擎 1.5.0 版本已于 4 月 14 日正式发布,从 1.5.0 版本开始,团结引擎车机版带来了一个激动人心的新能力 —— Android App View。现在,开发者可以将任意第三方安卓应用以 2D 组件或 3D 组件的形式,原生嵌入到 Tuanjie 开发的…...