前端技术
HTML
CSS
Javascript
前端框架和UI库
VUE
ReactJS
AngularJS
JQuery
NodeJS
JSON
Element-UI
Bootstrap
Material UI
服务端和客户端
Java
Python
PHP
Golang
Scala
Kotlin
Groovy
Ruby
Lua
.net
c#
c++
后端WEB和工程框架
SpringBoot
SpringCloud
Struts2
MyBatis
Hibernate
Tornado
Beego
Go-Spring
Go Gin
Go Iris
Dubbo
HessianRPC
Maven
Gradle
数据库
MySQL
Oracle
Mongo
中间件与web容器
Redis
MemCache
Etcd
Cassandra
Kafka
RabbitMQ
RocketMQ
ActiveMQ
Nacos
Consul
Tomcat
Nginx
Netty
大数据技术
Hive
Impala
ClickHouse
DorisDB
Greenplum
PostgreSQL
HBase
Kylin
Hadoop
Apache Pig
ZooKeeper
SeaTunnel
Sqoop
Datax
Flink
Spark
Mahout
数据搜索与日志
ElasticSearch
Apache Lucene
Apache Solr
Kibana
Logstash
数据可视化与OLAP
Apache Atlas
Superset
Saiku
Tesseract
系统与容器
Linux
Shell
Docker
Kubernetes
[采用可再生能源减少交通污染的影响]的搜索结果
这里是文章列表。热门标签的颜色随机变换,标签颜色没有特殊含义。
点击某个标签可搜索标签相关的文章。
点击某个标签可搜索标签相关的文章。
HTML
...变化、全球变暖、环境污染、生物灭绝等,这些问题对我们的生活和发展产生着巨大的影响。因此,维护家园已经成为了人类面临的重要任务。 维护家园的方法 人们应该采取一系列方法来维护和改善家园的环境,例如: 降低二氧化碳排放,调节温室气体的增加,以防止全球气候变暖。 循环利用资源,降低能耗,降低垃圾污染,维护家园生态平衡。 推动绿色交通,鼓励使用可再生能源,降低交通污染与噪声。 倡导环保意识,倡导低碳生活,从个人做起,逐步改变生活习惯,减轻家园负担。 我们的行动 作为家园的维护者和建设者,我们要从自身做起,积极参与到家园环保的行动中来,让我们的生活、工作和生产更加环保、持久的,为我们的子孙后代留下更加美好的世界。 <html> <header> <title>维护家园的必要性</title> </header> <body> <h1>维护家园,涉及我们的未来</h1> <p>家园是我们生活的根基,它供给了我们所需要的空气、水、食物、药品等必需品,但随着人类活动的不断增加,家园面临着越来越多的破坏和威胁,例如:气候变化、全球变暖、环境污染、生物灭绝等,这些问题对我们的生活和发展产生着巨大的影响。因此,维护家园已经成为了人类面临的重要任务。</p> <h2>维护家园的方法</h2> <p>人们应该采取一系列方法来维护和改善家园的环境,例如:</p> <ul> <li>降低二氧化碳排放,调节温室气体的增加,以防止全球气候变暖。</li> <li>循环利用资源,降低能耗,降低垃圾污染,维护家园生态平衡。</li> <li>推动绿色交通,鼓励使用可再生能源,降低交通污染与噪声。</li> <li>倡导环保意识,倡导低碳生活,从个人做起,逐步改变生活习惯,减轻家园负担。</li> </ul> <h2>我们的行动</h2> <p>作为家园的维护者和建设者,我们要从自身做起,积极参与到家园环保的行动中来,让我们的生活、工作和生产更加环保、持久的,为我们的子孙后代留下更加美好的世界。</p> </body> </html>
2024-01-01 15:43:53
457
程序媛
CSS
...化,每个CSS模块只影响其对应的组件或区域,避免了全局作用域下的样式冲突和冗余。在实际应用中,CSS Modules技术会为每个类名生成唯一的标识符,确保样式在不同模块间的隔离性和一致性,从而提升代码的可维护性和复用性。 样式重复引用问题 , 样式重复引用问题是CSS编写和管理中常见的一种现象,尤其在大型复杂的Web项目中尤为突出。当多个页面或模块引用了相同的CSS样式规则时,如果每处都单独编写相同的样式,会导致代码重复,增加文件体积,降低加载效率,并可能引发潜在的样式覆盖和维护困难等问题。通过合理的CSS模块化设计和引用机制,可以有效解决样式重复引用问题。 CSS-in-JS , CSS-in-JS是一种新兴的样式处理方案,它允许开发者直接在JavaScript中编写和操作CSS样式。这种技术打破了传统CSS与HTML分离的方式,使得样式能够根据运行时的状态动态生成和应用,增强了样式的灵活性和可编程性。CSS-in-JS不仅可以减少全局样式污染,实现按需加载,提高性能,还能更好地配合现代前端框架(如React、Vue等)进行组件化的开发实践,提高代码质量和开发效率。
2023-09-11 12:29:02
408
算法侠
CSS
...一些最佳实践提倡尽量减少甚至避免使用“!important”,转而利用CSS Modules或CSS-in-JS等方案增强样式作用域隔离,确保样式声明的明确性和可维护性。 同时,考虑到响应式设计和无障碍访问的需求,过度依赖“!important”可能会导致在不同设备和用户代理下的样式失效或者冲突。因此,不少开发者倡导通过提升选择器特定性和优化CSS代码结构来控制样式层级,从而达到预期的渲染效果。 此外,对于团队协作的项目,良好的CSS编码规范是必不可少的,其中应明确规定“!important”的使用条件和限制,以防止因个人习惯差异导致的全局样式污染问题。 总之,在实际开发过程中,理解和掌握“!important”的适用范围和潜在影响,结合最新的前端技术和最佳实践,才能真正做到精细化、高效化的样式管理,打造出既美观又易于维护的网页应用。
2023-04-18 17:52:39
558
逻辑鬼才
MemCache
...等待。这种机制就像个交通警察,它能确保多个线程不会同时对一份数据动手脚,这样一来,就相当于拦住了可能导致数据混乱的各种“撞车”事件,让数据始终保持一致性和准确性。 三、Memcache 的锁机制 Memcache 使用了一种称为“互斥锁(mutex)”的锁机制。当一个线程需要访问某个键对应的值时,它首先会尝试获取这个键的锁。如果锁已经被其他线程占用,那么当前线程就需要等待锁被释放。一旦锁被释放,当前线程就可以安全地读取或修改这个键对应的值。 四、多线程环境下锁机制冲突的原因 在多线程环境中,由于锁的粒度是键级别的,而不同的线程可能会操作相同的键,这就可能导致锁的竞争和冲突。具体来说,以下两种情况可能会导致锁的冲突: 1. 锁竞争 当多个线程同时尝试获取同一个键的锁时,就会发生锁竞争。 2. 锁膨胀 当一个线程已经获取了某个键的锁,但又试图获取另一个键的锁时,如果这两个键都在同一个数据库行中,那么就可能发生锁膨胀。 五、解决锁机制冲突的方法 为了防止锁的冲突,我们可以采取以下几种方法: 1. 分布式锁 使用分布式锁可以有效解决锁的竞争问题。分布式锁啊,就好比是多个小哥一起共用的一把钥匙,当其中一个线程小弟想要拿到这把钥匙的时候,它会先给所有节点大哥们发个消息:“喂喂喂,我要拿钥匙啦!”然后呢,就看哪个节点大哥反应最快,最先回应它,那这个线程小弟就从这位大哥手里接过钥匙,成功获取到锁啦。 2. 延迟锁 延迟锁是一种特殊的锁,它可以保证在一段时间内只有一个线程可以访问某个资源。当一个线程想去获取锁的时候,假如这个锁已经被其他线程给霸占了,那么它不会硬碰硬,而是会选择先歇一会儿,过段时间再尝试去抢夺这把锁。 3. 减少锁的数量 减少锁的数量可以有效地减少锁的竞争。比如,我们能够把一个看着头疼的复杂操作,拆分成几个轻轻松松就能理解的小步骤,每一步只专注处理一点点数据,就像拼图一样简单明了。 六、代码示例 以下是一个使用 Memcache 的代码示例,展示了如何使用互斥锁来保护共享资源: python import threading from memcache import Client 创建一个 Memcache 客户端 mc = Client(['localhost:11211']) 创建一个锁 lock = threading.Lock() def get(key): 获取锁 lock.acquire() try: 从 Memcache 中获取数据 value = mc.get(key) if value is not None: return value finally: 释放锁 lock.release() def set(key, value): 获取锁 lock.acquire() try: 将数据存储到 Memcache 中 mc.set(key, value) finally: 释放锁 lock.release() 以上代码中的 get 和 set 方法都使用了一个锁来保护 Memcache 中的数据。这样,即使在多线程环境下,也可以保证数据的一致性。 七、总结 在多线程环境下,Memcache 的锁机制冲突是一个常见的问题。了解了锁的真正含义和它的工作原理后,我们就能找到对症下药的办法,保证咱们的程序既不出错,又稳如泰山。希望这篇文章对你有所帮助。
2024-01-06 22:54:25
78
岁月如歌-t
Beego
...要。代码提交规则就像交通规则一样,能让我们这些开发者都遵守同一套玩法,避免在项目里撞车,还能把代码搞得更靠谱些。试想一下,要是团队里没有一套统一的编码规范,那代码库岂不是跟被龙卷风刮过似的,乱七八糟的,以后要维护起来简直就像是在找针一样难。再说呢,每个程序员都有自己的小癖好嘛,这就导致大家的写代码风格五花八门。有时候看着别人的代码就像在猜谜,这事儿挺影响咱们团队干活儿的效率的。 3. 实际案例分析 接下来,让我们通过几个具体的案例来看看不遵守代码提交规则可能带来的问题。 3.1 案例一:代码风格不一致 假设我们在一个Beego项目中,有的开发者喜欢用单引号,而有的开发者喜欢用双引号。这就造成了代码风格五花八门,读起来费劲不说,还容易出些莫名其妙的bug。比如,在Beego中,如果我们使用了不一致的引号风格,可能会导致字符串解析错误。下面是一个简单的示例: go // 不同的引号风格 func main() { name := 'John' // 使用单引号 age := "30" // 使用双引号 } 这样的一段代码在编译时可能会报错,因为Go语言的标准是使用双引号作为字符串的分隔符。如果团队内部没有统一的规则,这样的错误就很容易发生。 3.2 案例二:缺少必要的注释 另一个常见的问题是缺乏必要的注释。在Beego项目里,我们有时得花时间解释那些烧脑的逻辑,或者是给API接口写点使用说明啥的。如果这些重要的信息没有被记录下来,后续维护人员将会面临很大的困扰。例如,我们可以看看下面这个简单的Beego控制器示例: go package controllers import ( "github.com/astaxie/beego" ) type UserController struct { beego.Controller } // 获取用户列表 func (this UserController) GetUserList() { users := []User{} // 假设User是定义好的结构体 this.Data["json"] = users this.ServeJSON() } 在这个例子中,如果没有任何注释,其他开发者很难理解这个函数的具体作用。因此,添加必要的注释是非常重要的。 3.3 案例三:没有遵循版本控制的最佳实践 最后,我们来看看版本控制的问题。在Beego项目中,我们通常会使用Git来进行版本控制。不过,要是团队里的小伙伴不按套路出牌,比如压根不用分支管理,或者是提交信息简单得让人摸不着头脑,那后续的代码管理和维护可就头大了。举个例子: bash 不正确的提交信息 $ git commit -m "修改了一些东西" 这样的提交信息没有任何具体的内容,对于后续的代码审查和维护都是不利的。正确的做法应该是提供更详细的提交信息,比如: bash $ git commit -m "修复了用户列表接口的bug,增加了错误处理逻辑" 4. 如何改进? 既然我们已经了解了不遵守代码提交规则可能带来的问题,那么接下来我们该如何改进呢? 4.1 制定并遵守统一的编码规范 首先,我们需要制定一套统一的编码规范,并确保所有团队成员都严格遵守。比如说,我们可以定个规矩,所有的字符串都得用双引号包起来,变量的名字呢,就用驼峰那种一高一低的方式起名。这不仅可以提高代码的可读性,还能减少不必要的错误。 4.2 添加必要的注释 其次,我们应该养成良好的注释习惯。在编写代码的同时,应该为重要的逻辑和接口添加详细的注释。这样,即使后续维护人员不是原作者,也能快速理解代码的意图。例如: go // 获取用户列表 // @router /api/users [get] func (this UserController) GetUserList() { users := []User{} // 假设User是定义好的结构体 this.Data["json"] = users this.ServeJSON() } 4.3 遵循版本控制的最佳实践 最后,我们还需要遵循版本控制的最佳实践。比如说,当你用分支管理功能时,提交的信息可得越详细越好,这样以后自己或别人看代码时才会更容易,审查和维护起来也更轻松。例如: bash 正确的提交信息 $ git commit -m "修复了用户列表接口的bug,增加了错误处理逻辑" 5. 结语 总之,代码提交规则的严格遵守对于Beego项目的成功至关重要。虽然开始时可能会觉得有点麻烦,但习惯了之后,你会发现这能大大提升团队的工作效率和代码质量。希望各位开发者能够认真对待这个问题,共同维护一个高质量的代码库。
2024-12-26 15:33:14
92
红尘漫步
Mongo
...功能特性有着决定性的影响。那么,咱们就来聊一聊MongoDB这家伙到底用的是哪种存储引擎吧!在这篇文章里,我会手把手地带你们深入探索这个问题,还会通过一些实实在在的代码实例,教大家如何查看以及亲自指定这个存储引擎,就像在玩一场技术揭秘的游戏一样。 1. MongoDB存储引擎概述 MongoDB在其发展历程中曾支持过多种存储引擎,包括早期版本中的MMAPv1以及后续逐渐成为默认选择的WiredTiger。当前(2024年),WiredTiger 已经是MongoDB社区版和企业版的标准配置,自MongoDB 3.2版本后被确立为默认存储引擎。这个决策背后的真正原因是,WiredTiger这家伙拥有更先进的并发控制技术,就像个超级交通管理员,能同时处理好多任务还不混乱;它的压缩机制呢,就像是个空间魔法师,能把数据压缩得妥妥的,节省不少空间;再者,它的检查点技术就像个严谨的安全员,总能确保系统状态的一致性和稳定性。所以,在应对大部分工作负载时,WiredTiger的表现那可真是更胜一筹,让人不得不爱! 1.1 WiredTiger的优势 - 文档级并发控制:WiredTiger实现了行级锁,这意味着它可以在同一时间对多个文档进行读写操作,极大地提高了并发性能,特别是在多用户环境和高并发场景下。 - 数据压缩:WiredTiger支持数据压缩功能,能够有效减少磁盘空间占用,这对于大规模数据存储和传输极为重要。 - 检查点与恢复机制:定期创建检查点以确保数据持久化,即使在系统崩溃的情况下也能快速恢复到一个一致的状态。 2. 如何查看MongoDB的存储引擎? 要确定您的MongoDB实例当前使用的存储引擎类型,可以通过运行Mongo Shell并执行以下命令: javascript db.serverStatus().storageEngine 这将返回一个对象,其中包含了存储引擎的名称和其他详细信息,如引擎类型是否为wiredTiger。 3. 指定MongoDB存储引擎 在启动MongoDB服务时,可以通过mongod服务的命令行参数来指定存储引擎。例如,若要明确指定使用WiredTiger引擎启动MongoDB服务器,可以这样做: bash mongod --storageEngine wiredTiger --dbpath /path/to/your/data/directory 这里,--storageEngine 参数用于设置存储引擎类型,而--dbpath 参数则指定了数据库文件存放的位置。 请注意,虽然InMemory存储引擎也存在,但它主要适用于纯内存计算场景,即所有数据仅存储在内存中且不持久化,因此不适合常规数据存储需求。 4. 探讨与思考 选择合适的存储引擎对于任何数据库架构设计都是至关重要的。随着MongoDB的不断成长和进步,核心团队慧眼识珠,挑中了WiredTiger作为默认配置。这背后的原因呢,可不光是因为这家伙在性能上表现得超级给力,更因为它对现代应用程序的各种需求“拿捏”得恰到好处。比如咱们常见的实时分析呀、移动应用开发这些热门领域,它都能妥妥地满足,提供强大支持。不过呢,每个项目都有自己独特的一套规矩和限制,摸清楚不同存储引擎是怎么运转的、适合用在哪些场合,能帮我们更聪明地做出选择,让整个系统的性能表现更上一层楼。 总结来说,MongoDB如今已经将WiredTiger作为其默认且推荐的存储引擎,但这并不妨碍我们在深入研究和评估后根据实际业务场景选择或切换存储引擎。就像一个经验老道的手艺人,面对各种不同的原料和工具,咱们得瞅准具体要干的活儿和环境条件,然后灵活使上最趁手的那个“秘密武器”,才能真正鼓捣出既快又稳、超好用的数据库系统来。
2024-01-29 11:05:49
202
岁月如歌
ZooKeeper
...盘I/O错误的表现及影响 当ZooKeeper日志中频繁出现“Disk is full”、“No space left on device”或“I/O error”的警告时,表明存在磁盘I/O问题。这种状况会导致ZooKeeper没法顺利完成事务日志和快照文件的写入工作,这样一来,那些关键的数据持久化,还有服务器之间的选举、同步等核心功能都会受到连带影响。到了严重的时候,甚至会让整个服务直接罢工,无法提供服务。 4. 探究原因与解决方案 (1)磁盘空间不足 这是最直观的原因,可以通过清理不必要的数据文件或增加磁盘空间来解决。例如,定期清理ZooKeeper的事务日志和快照文件,可以使用自带的zkCleanup.sh脚本进行自动维护: bash ./zkCleanup.sh -n myServer1:2181/myZooKeeperCluster -p /data/zookeeper/version-2 (2)磁盘I/O性能瓶颈 如果磁盘读写速度过慢,也会影响ZooKeeper的正常运行。此时应考虑更换为高性能的SSD硬盘,或者优化磁盘阵列配置,提高I/O吞吐量。另外,一个蛮实用的办法就是灵活调整ZooKeeper的刷盘策略。比如说,我们可以适当地给syncLimit和tickTime这两个参数值加加油,让它们变大一些,这样一来,就能有效地降低刷盘操作的频率,让它不用那么频繁地进行写入操作,更贴近咱们日常的工作节奏啦。 (3)并发写入压力大 高并发场景下,大量写入请求可能会导致磁盘I/O瞬间飙升。对于这个问题,我们可以采取一些措施,比如运用负载均衡技术,让ZooKeeper集群的压力得到分散缓解,就像大家一起扛米袋,别让一个节点给累垮了。另外,针对实际情况,咱们也可以灵活调整,对ZooKeeper客户端API的调用来个“交通管制”,根据业务需求合理限流控制,避免拥堵,保持运行流畅。 5. 结论 面对ZooKeeper运行过程中出现的磁盘I/O错误,我们需要具体问题具体分析,结合监控数据、日志信息以及系统资源状况综合判断,采取相应措施进行优化。此外,良好的运维习惯和预防性管理同样重要,如定期检查磁盘空间、合理分配资源、优化系统配置等,都是避免这类问题的关键所在。说真的,ZooKeeper就相当于我们分布式系统的那个“底座大石头”,没它不行。只有把这块基石稳稳当当地砌好,咱们的系统才能健壮得像头牛,让人放心可靠地用起来。 以上内容,不仅是我在实践中积累的经验总结,也是我不断思考与探索的过程,希望对你理解和处理类似问题有所启发和帮助。记住,技术的魅力在于持续学习与实践,让我们一起在ZooKeeper的世界里乘风破浪!
2023-02-19 10:34:57
127
夜色朦胧
Netty
...很多时候,这个问题可能源自于对系统需求的理解不足,或者是对现有技术栈的过度依赖。比如说,如果我们没意识到自己的应用得应对海量的同时请求,然后就随便选了个简单的线程池方案,那到了高峰期,系统卡成狗基本上是躲不掉的。 2.1 案例分析:一个失败的案例 假设我们正在开发一款即时通讯应用,目标是支持数千用户同时在线聊天。一开始,我们可能觉得用个固定大小的线程池挺省事儿,以为这样能简化开发流程,结果发现事情没那么简单。不过嘛,在真正的战场里,一旦用户蜂拥而至,这种方法就露馅了:线程池里的线程忙得团团转,新的请求不是被直接拒之门外,就是得乖乖排队,等老半天才轮到自己。这不仅影响了用户体验,也限制了系统的扩展能力。 3. Netty中的并发资源分配 寻找正确的路径 既然提到了Netty,那么我们就来看看如何利用Netty来解决并发资源分配的问题。Netty提供了多种机制来管理并发访问,其中最常用的莫过于EventLoopGroup和ChannelPipeline。 3.1 EventLoopGroup:并发管理的核心 EventLoopGroup是Netty中用于处理并发请求的核心组件之一。这家伙专门管理一帮EventLoop小弟,每个小弟都负责处理一类特定的活儿,比如读数据啦,写数据啦,干得可带劲了!合理地设置EventLoopGroup,就能更好地分配和管理资源,避免大家抢来抢去的尴尬局面啦。 示例代码: java // 创建两个不同的EventLoopGroup,分别用于客户端和服务端 EventLoopGroup bossGroup = new NioEventLoopGroup(1); EventLoopGroup workerGroup = new NioEventLoopGroup(); try { // 创建服务器启动器 ServerBootstrap b = new ServerBootstrap(); b.group(bossGroup, workerGroup) .channel(NioServerSocketChannel.class) .childHandler(new ChannelInitializer() { @Override public void initChannel(SocketChannel ch) throws Exception { ch.pipeline().addLast(new TimeServerHandler()); } }); // 绑定端口,同步等待成功 ChannelFuture f = b.bind(port).sync(); // 等待服务端监听端口关闭 f.channel().closeFuture().sync(); } finally { // 优雅地关闭所有线程组 bossGroup.shutdownGracefully(); workerGroup.shutdownGracefully(); } 在这个例子中,我们创建了两个EventLoopGroup:bossGroup和workerGroup。前者用于接收新的连接请求,后者则负责处理这些连接上的I/O操作。这样的设计不仅提高了并发处理能力,还使得代码结构更加清晰。 3.2 ChannelPipeline:灵活的请求处理管道 除了EventLoopGroup之外,Netty还提供了一个非常强大的功能——ChannelPipeline。这简直就是个超级灵活的请求处理流水线,我们可以把一堆处理器像串糖葫芦一样串起来,然后一个个按顺序来处理网络上的请求,简直不要太爽!这种方式非常适合那些需要执行复杂业务逻辑的应用场景。 示例代码: java public class TimeServerHandler extends ChannelInboundHandlerAdapter { @Override public void channelRead(ChannelHandlerContext ctx, Object msg) { ByteBuf buf = (ByteBuf) msg; try { byte[] req = new byte[buf.readableBytes()]; buf.readBytes(req); String body = new String(req, "UTF-8"); System.out.println("The time server receive order : " + body); String currentTime = "QUERY TIME ORDER".equalsIgnoreCase(body) ? new Date( System.currentTimeMillis()).toString() : "BAD ORDER"; currentTime = currentTime + System.getProperty("line.separator"); ByteBuf resp = Unpooled.copiedBuffer(currentTime.getBytes()); ctx.write(resp); } finally { buf.release(); } } @Override public void channelReadComplete(ChannelHandlerContext ctx) { ctx.flush(); } @Override public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) { // 当出现异常时,关闭Channel cause.printStackTrace(); ctx.close(); } } 在这个例子中,我们定义了一个TimeServerHandler类,继承自ChannelInboundHandlerAdapter。这个处理器的主要职责是从客户端接收请求,并返回当前时间作为响应。加个这样的处理器到ChannelPipeline里,我们就能轻轻松松地扩展或者修改请求处理的逻辑,完全不用去动那些复杂的底层网络通信代码。这样一来,调整起来就方便多了! 4. 结论 拥抱变化,不断进化 通过上述讨论,我们已经看到了正确选择并发资源分配算法的重要性,以及Netty在这方面的强大支持。当然啦,这只是个开始嘛,真正的考验在于你得根据自己实际用到的地方,不断地调整和优化这些方法。记住,优秀的软件工程师总是愿意拥抱变化,勇于尝试新的技术和方法,以求达到最佳的性能表现和用户体验。希望这篇文章能给大家带来一些启示,让我们一起在技术的海洋里继续探索吧! --- 这篇技术文章希望能够以一种更贴近实际开发的方式,让大家了解并发资源分配的重要性,并通过Netty提供的强大工具,找到适合自己的解决方案。如果有任何疑问或建议,欢迎随时留言交流!
2024-12-05 15:57:43
102
晚秋落叶
转载文章
...3年):饿了么由上海交通大学创始团队起家,发展至35人规模,日订单量维持在十万量级,由“IDC+Python”技术组合支撑业务运营,但面临Python人才难觅等困扰。 2)成长期(2014年至2015年):14年8至9月短短2个月内日均订单量增长10倍,从10万迅猛飙升至100万,业务规模主攻全国200个城市,原有IT系统架构压力极大,依靠人肉运维举步维艰,故障波动影响业务,创始人与核心技术团队坚守机房运维一线,才勉强扛住100万量级业务订单。开始借鉴阿里淘宝架构模式,人员团队也涨至500人,技术生态从Python扩展至“Java+Python”开发体系,从“人肉”支撑百万订单运营到自动化运维,并筹备同城异地容灾体系。 3)规模期(2015年至2017年):2015年7至8月,日均订单量从200万翻倍,以往积压的问题都暴露出来,技术架构面临大考验,坚定了架构上云的方案,团队扩展至1000人,架构要承载数百万量级业务时,出现峰值成本、灾备切换、IDC远程运维等种种挑战,全面战略转型采用“IDC+云计算”的混合云架构。在2016年12月25日圣诞节日订单量迎来前所未有的900万单,因此在技术架构上探索多活部署等创新性研发。 为什么选择架构转型上云?据饿了么CTO张雪峰先生所说,技术架构从IDC经典模式发展至混合云模式,主要原因是三个关键因素让管理层下定决心上云: 1) 脉冲计算:从技术架构配套业务发展分析,网络订餐业务具有明显的“脉冲计算”特征,在每日上午10:00至13:00、晚间16:00至19:00业务高峰值出现,而其他时间则业务量很低,暑假是业务高峰季,2016年5.17大促,饿了么第一次做“秒杀”,一秒订单15000笔,巨大的波峰波谷计算差异,引发了自建数据中心容量不可调和的两难处境,如果大规模投入服务器满足6小时的高峰业务量,则其余18个小时的业务低谷计算资源闲置,若满足平均业务量,则无法跟上业务快速发展节奏,落后于竞争对手;搞电商大促时,计算资源投入巨大,大促之后计算峰值下降,采用自建机房利用率仅10%,所以技术团队摸索出用云计算扛营销大促峰值的新模式,采用混合云架构满足 “潮汐业务”峰值计算,阿里云海量云计算资源弹性随需满足巨大的脉冲计算力缺口,这与每年“双11” 淘宝引入阿里云形成全球最大混合云架构具有异曲同工的创新价值。 2) 数据量爆炸:伴随饿了么近五年业务量呈几何级数的爆发式发展,数据量增速更加令人吃惊,是业务量增速的5倍,每日增量数据接近100TB,2015年短短2个月内业务量增长10倍,数据量增长了50倍,上海主生产机房不堪重负。30GB的DDoS攻击对业务系统造成较大风险,上云成为承载大数据、抗网络攻击的好方法。 3) 高可用性挑战:众所周知,IDC自建系统运维要承担从底层硬件到上层应用的“全栈运维”运营能力与维修能力,当2015年夏天上海数据中心故障发生,主核心交换机宕机时,备核心交换机Bug同时被触发,从事故发生到硬件厂商携维修设备打车赶往现场维修的整个过程中,饥饿的消费者无法订餐吃饭,技术团队第一次经历业务中断而束手无策,才下定决心大笔投入混合云灾备的建设,“吃一堑,长一智”,持续向淘宝学习电商云生产与灾备架构,以自动化运维替代人肉运维,从灾备向多活演进,成为饿了么企业架构转型的必经之路。 4) 大数据精益运营:不论网络打车还是网络订餐,共享服务平台脱颖而出的关键成功要素是智能调度算法,以大数据训练算法提升调度效率,饿了么在高峰时段内让百万“骑士”(送餐快递员)完成更多订单是算法持续优化的目标,而这背后隐藏着诸多复杂因素,包括考虑餐厅、骑士、消费者三者的实时动态位置关系,把新订单插入现有“骑士”的行进路线中,估计每家餐厅出餐时间,每个骑手的行进速度、道路熟悉程度各不相同,新老消费者获客成本、高价低价订单的优先级皆不相同。种种考量因素合并到一起,对于人类调度员来说,每天中午和晚上的高峰都是巨大的挑战。以上海商城路配送站为例,一个调度员每6秒钟就要调度1单,他需要考虑骑手已有订单量、路线熟悉度等。因此可以说,这份工作已经完全不适合人类。但对人工智能而言,阿里云ET则非常擅长处理这类超复杂、大规模、实时性要求高的“非人”问题。 饿了么是中国最大的在线外卖和即时配送平台,日订单量900万单、180万骑手、100万家餐饮店,既是史无前例的计算存储挑战,又是人无我有的战略发展机遇。饿了么携手阿里云人工智能团队,通过海量数据训练优化全球最大实时智能调度系统。在基础架构层,云计算解决弹性支撑业务量波动的基础生存问题,在数据智能层,利用大数据训练核心调度算法、提升餐饮店的商业价值,才是业务决胜的“技术神器”。 在针对大数据资源的“专家+机器”运营分析中,不断发现新的特征: 1) 区域差异性:饿了么与阿里云联合研发小组测试中发现有2个配送站点出现严重超时问题。后来才知道:2个站点均在成都,当地人民喜欢早、中餐一起吃,高峰从11点就开始了。习惯了北上广节奏的ET到成都就懵了。据阿里云人工智能专家闵万里分析:“不存在一套通用的算法可以适配所有站点,所以我们需要让ET自己学习或者向人类运营专家请教当地的风土人情、饮食习惯”。除此之外,饿了么覆盖的餐厅不仅有高大上的连锁店,还有大街小巷的各类难以琢磨的特色小吃,难度是其他智能调度业务的数倍。 2) 复杂路径规划:吃一口热饭有多难?送餐路径规划比驾车出行路径规划难度更高,要考虑“骑士”地图熟悉程度、天气状况、拼单效率、送餐顺序、时间对客户满意度影响、送达写字楼电梯等待时间等各种实际情况,究竟ET是如何实现智能派单并确保效率最优的呢?简单来说,ET会将配送站新接订单插入到每个骑手已有的任务中,重新规划一轮最短配送路径,对比哪个骑手新增时间最短。为了能够准确预估新增时间,ET需要知道全国100万家餐厅的出餐速度、超过180万骑手各自的骑行速度、每个顾客坐电梯下楼取餐的时间。一般来说,餐厅出餐等待时间占到了整个送餐时间的三分之一。ET要想提高骑手效率,必须准确预估出餐时间以减少骑手等待,但又不能让餐等人,最后饭凉了。饿了么旗下蜂鸟配送“准时达”服务单均配送时长缩短至30分钟以内。 3) 天气特殊影响:天气等环境因素对送餐响应时间影响显著,要想计算骑手的送餐路程时间,ET需要知道每个骑手在不同区域、不同天气下的送餐速度。如果北京雾霾,ET能看见吗?双方研发团队为ET内置了恶劣天气的算法模型。通常情况下,每逢恶劣天气,外卖订单将出现大涨,对应的餐厅出餐速度和骑手骑行速度都将受到影响,这些ET都会考虑在内。如果顾客在下雪天点个火锅呢?ET也知道,将自动识别其为大单,锁定某一个骑手专门完成配送。 4) 餐饮营销顾问:饿了么整体业务涉及C端(消费者)、B端(餐饮商户)、D端(物流配送)、BD端(地推营销),以往区域业务开拓考核新店数量,现在会重点关注餐饮外卖“健康度”,对于营业额忽高忽低、在线排名变化的餐饮店,都需要BD专家根据大数据帮助餐饮店经营者找出原因并给出解决建议,避免新店外卖刚开始就淹没在区域竞争中,销量平平的新店会离开平台,通过机器学习把餐饮运营专家的经验、以及人看不到的隐含规律固化下来,以数据决策来发现餐饮店经营问题、产品差异定位,让餐饮商户尝到甜头,才愿意继续经营。举个例子,饿了么员工都喜欢楼下一家鸡排店的午餐,但大数据发现这家店的外卖营收并不如实体店那么火爆,9元“鸡排+酸梅汁”是所有人都喜欢的爆款产品,可为什么同样菜品遭遇“线下火、线上冷”呢?数据预警后,BD顾问指出线上外卖鸡排产品没有写明“含免费酸梅汁一杯”的关键促销内容,导致大多数外卖消费者订一份鸡排一杯酸梅汁,却收到一份鸡排两杯酸梅汁,体验自然不好。 饿了么是数据驱动、智能算法调度的自动化生活服务平台,通过O2O数据的在线实时分析,与阿里云人工智能团队不断改进算法,以“全局最优”取代“局部最优”,保证平台上所有餐饮商户都能享受到数据智能的科技红利。 “上云用数”的外部价值诸多,从饿了么内部反馈来看,上云不仅没有让运维团队失去价值,反而带来了“云原生应用”(Cloud Native Application)、“云上多活”、“CDN云端压测”、“安全风控一体化”等创新路径与方案,通过敏捷基础设施(IaaS)、微服务架构(PaaS和SaaS)、持续交付管理、DevOps等云最佳实践,摆脱“人肉”支撑的种种困境,进而实现更快的上线速度、细致的故障探测和发现、故障时能自动隔离、故障时能够自动恢复、方便的水平扩容。饿了么CTO张雪峰先生说:“互联网平台型组织,业务量涨数倍,企业人数稳定降低,才是技术驱动的正确商业模式。” 在不久的将来,你每天订餐、出行、娱乐、工作留下的大数据,会“驯养”出无处不在、无所不能的智能机器人管家,家庭助理帮你点菜,无人机为你送餐,聊天机器人接受你的投诉……当然这个无比美妙的“未来世界”背后,皆有阿里云的数据智能母体“ET”。 本篇文章为转载内容。原文链接:https://blog.csdn.net/weixin_34126557/article/details/90592502。 该文由互联网用户投稿提供,文中观点代表作者本人意见,并不代表本站的立场。 作为信息平台,本站仅提供文章转载服务,并不拥有其所有权,也不对文章内容的真实性、准确性和合法性承担责任。 如发现本文存在侵权、违法、违规或事实不符的情况,请及时联系我们,我们将第一时间进行核实并删除相应内容。
2023-01-31 14:48:26
343
转载
转载文章
...电及顽皮狗。虽然书中采用的例子通常依据一些专门的技术,但是讨论范围远超于某个引擎或API。文中的参考及引用也非常有用,可让读者继续深入游戏开发过程的任何特定方向。 《游戏引擎架构》为一个大学程度的游戏编程课程而编写,但也适合软件工程师、业余爱好者、自学游戏程序员,以及游戏产业的从业人员。通过阅读《游戏引擎架构》,资历较浅的游戏工程师可以巩固他们所学的游戏技术及引擎架构的知识,专注某一领域的资深程序员也能从本书更为全面的介绍中获益。 內容包括: 游戏开发中的大规模C++软件架构 游戏编程所需的数学 供调试、源代码控制及性能剖析的游戏开发工具 引擎基础系统、渲染、碰撞、物理、角色动画、游戏世界对象模型等引擎子系统 多平台游戏引擎 多处理器环境下的游戏编程 工作管道及游戏资产数据库 作者/译者简介 作者介绍:Jason Gregory在1994年开始任职专业软件工程师,自1999年3月开始在游戏产业中任职软件工程师。在圣迭哥Midway Home Entertainment公司开始游戏编程的他,为《疯狂飞行员(Freaky Flyers)》及《Crank the Weasel》开发PlayStation 2/Xbox上的动画系统。在2003年,他转到洛杉矶艺电,为《荣誉勋章:血战太平洋(Medal of Honor: Pacific Assault)》开发游戏引擎及游戏性技术,并在《荣誉勋章:空降神兵(Medal of Honor: Airborne)》中担任首席工程师。他现时是顽皮狗公司的通才程序员,为《神秘海域:德雷克船长的宝藏(Uncharted: Drake's Fortune)》及《神秘海域:纵横四海(Uncharted: Among Thieves)》开发引擎及游戏性软件。他也在南加州大学教授游戏技术的课程。 译者简介:叶劲峰(Milo Yip)从小自习编程,并爱好计算机图形学。上中学时兼职开发策略RPG《王子传奇》,该游戏在1995年于台湾发行。其后他获取了香港大学认知科学学士、香港中文大学系统工程及工程管理哲学硕士。毕业后在香港理工大学设计学院从事游戏引擎及相关技术的研发,职至项目主任。除发表学术文章外,也曾合著《DirectX9游戏编程实务》。2008年往上海育碧担任引擎工程师开发《美食从天而降(Cloudy with a Chance of Meatballs)》Xbox360/PS3/Wii/PC,2009年起于麻辣马开发《爱丽丝:疯狂回归(Alice: Madness Returns)》Xbox360/PS3/PC,2011年加入腾讯互动娱乐引擎技术中心担任专家工程师,所研发的技术已用于《斗战神》、《天涯明月刀》、《众神争霸》等项目中。 推荐序1 最初拿到《Game Engine Architecture》一书的英文版,是编辑侠少邮寄给我的打印版。他建议我接下翻译此书的合同。当时我正在杭州带领一个团队开发3D游戏引擎,我和我的同事都对这本书的内容颇有兴趣,两大本打印的英文书立刻在同事间传开。可惜那段时间个人精力顾及不来,把近千页的英文读物精读而后翻译成中文对个人的业余时间是个极大的挑战,不能担此翻译任务颇为遗憾。 不久以后听说Milo Yip(叶劲峰)已开始着手翻译,甚为欣喜。翻译此巨著,他一定是比我更合适的人选。我和Milo虽未曾蒙面,但神交已久。在网络上读过一些他的成长经历,和我颇为相似,心有戚戚。他对游戏3D实时渲染技术研究精深为我所不及,我们曾通过Google Talk讨论过许多技术问题,他都有独到的见解。翻译工作开始后,Milo是香港人,英文技术术语在香港的中文译法和大陆的有许多不同。但此书由大陆出版社出版,考虑到面对的读者主要是大陆程序员,Milo希望能更符合大陆程序员的用词习惯,所以在翻译一开始就通过Google Docs创建了协作页面,邀请大家共同探讨书中技术名词的中译名。从中我们可以一窥他作为译者的慎重。 三年之后,有幸在出版之前就拿到了完整的译本。这是一本用LaTeX精心排版的800页的电子书,我只花了一周时间,几乎是一口气读完。流畅的阅读享受,绝对不仅仅是因为原著精彩的内容,精美的版面和翔实的译注也加了不少分。 在阅读本书的过程中,我不只一次地获得共鸣。例如在第5章的内存管理系统的介绍中,作者介绍的几种游戏特有的内存管理方法我都曾在项目中用过,而这是第一次有书籍专门将这些方法详尽记录;又如第11章动画系统的介绍,我们也同样在3D引擎开发过程中改进原有动画片段混合方法的经历。虽然书中介绍的每个技术点,都可能可以在某篇论文,某本其他的书的章节,某篇网络blog上见过,但之前却无一本书可以把这些东西放在一起相互参照。对于从事游戏引擎开发的程序员来说,了解各种引擎在处理每个具体问题时的方案是相当重要的。而每种方案又各有利弊,即使不做引擎开发工作而是在某一特定游戏引擎上做游戏开发,从中也可以理解引擎的局限性以及可能的改进方法。尤其是第14章介绍的对游戏性相关系统的设计,各个开发人员几乎都是凭经验设计,很少见有书籍对这些做总结。对于基于渲染引擎做开发的游戏程序员,这是必须面对的工作,这一章会有很大的借鉴意义。 本书作者是业内资深的游戏引擎开发人,他所参于的《神秘海域》和《最后生还者》都是我的个人最爱。在玩游戏的过程中,作为游戏程序员的天性,自然会不断地猜想各个技术点是如何实现的,背后需要怎样的工具支持。能在书中一一得到印证是件特别开心的事情。作者反复强调代码实践的重要性,在书中遍布着C++代码。我不认为这些代码有直接取来使用的价值,但它们极大地帮助了读者理解书中的技术点。书中列出的顽皮狗工作室用lisp方言作为游戏配置脚本的范例也给我很大的启发,有了这些具体的代码示例以及作者本身的一线工程师背景,也让我确信书中那些关于主机游戏开发相关等,我所没有接触过的内容都也绝非泛泛而谈。 国内的游戏开发社区的壮大,主要是随最近十年的MMO风潮而生。而就在大型网络游戏在中国有些畸形发展,让这类游戏偏离电子游戏游戏性的趋势时,我们有幸迎来了为移动设备开发游戏的大潮。游戏开发的重心重新回到游戏性本身。我们更需要去借鉴单机游戏是如何为玩家带来更纯粹的游戏体验,我相信书中记录的各种技术点会变的更有帮助。 资深游戏开发及创业者 云风 @简悦云风 推荐序2 在我认识的许多游戏业开发同仁中,只有少数香港同胞,Milo Yip(叶劲峰)却正是这样一位给我印象非常深刻的优秀香港游戏开发者。我俩认识,是在Milo加入腾讯互动娱乐研发部引擎技术中心后,说来到现在也只是两年多时间。其间,他为人的谦逊务实,对待技术问题的严谨求真态度,对算法设计和性能优化的娴熟技术,都为人所称道。Milo一丝不苟的工作风格,甚至表现在对待技术文档排版这类事情上(Milo常执著地用LaTeX将技术文档排到完美),我想这一定是他在香港读大学、硕士及在香港理工大学的多媒体创新中心从事研究员,一贯沿袭至今的好作风。 我很高兴腾讯游戏有实力吸引到这样优秀的技术专家;即使在其已从上海迁回香港家中,依然选择到深圳腾讯互动娱乐总部工作。叶兄从此工作日每天早晚过关,来往香港和深圳两地,虽有舟车劳顿,但是兼顾了对家庭的照顾和在游戏引擎方面的专业研究,希望这样的状况是令他满意的。 认识叶兄当时,我便知道他在进行Jason Gregory所著《游戏引擎架构》一书的中译工作。因为自己从前也有业余翻译游戏开发有关书籍的经历,所以我能理解其中的辛苦和责任重大,对叶兄也更多一分钦佩。我以为,本书以及本书的中文读者最大的幸运便是,遇到叶兄这位对游戏有着如同对家对国般强烈责任感,犹如“游戏科学工作者”般的专业译者! 现在(2013年年末)无疑是游戏史上对独立游戏制作者最友好的年代。开发设备方便获得(相对过往仅由主机厂商授权才能获得专利开发设备,现在有一台智能手机和一台个人电脑就可以开发)、技术工具友好、调试过程简单方便,且互联网上有丰富的例程和开源代码参考,也有网上社区便于交流。很多爱好者能够很快地制作出可运行的游戏原型,其中一些也能发布到应用商店。 但是不全面掌握各方面知识,尤其是游戏引擎架构知识,往往只能停留在勉强修改、凑合重用别人提供的资源的应用程度上,难以做极限的性能改进,更妄谈革命式的架构创新。这样的程度是很难在成千上万的游戏中脱颖而出的。我们所认可的真正的游戏大作,必定是在某方面大幅超越用户期待的产品。为了打造这样的产品,游戏内容创作者(策划、美术等)需要“戴着镣铐跳舞”(在当前的机能下争取更多的创作自由度),而引擎架构合理的游戏可以经得起──也值得进行──反复优化,最终可以提供更多的自由度,这是大作出现的技术前提。 书的作者、译者、出版社的编者,加上读者,大家是因书而结缘的有缘人。因叶兄这本《游戏引擎架构》译著而在线上线下相识的读者们,你们是不是因“了解游戏引擎架构,从而制作/优化好游戏”这样的理想而结了缘呢? 亲爱的读者,愿你的游戏有一天因谜题巧妙绝伦、趣味超凡、虚拟世界气势磅礴、视觉效果逼真精美等专业因素取得业界褒奖,并得到玩家真诚的赞美。希望届时曾读叶兄这本《游戏引擎架构》译作的你,也可以回馈社会,回馈游戏开发的学习社区,帮助新人。希望你也可以建立微信公众号、博客等,或翻译游戏开发书籍,造福外语不好的读者,所以如果你的外语(英语、日语、韩语之于游戏行业比较重要)水平仍需精进,现在也可以同步加油了! 腾讯《天天爱消除》游戏团队Leader 沙鹰 @也是沙鹰 译序 数千年以来,艺术家们通过文学、绘画、雕塑、建筑、音乐、舞蹈、戏剧等传统艺术形式充实人类的精神层面。自20世纪中叶,计算机的普及派生出另一种艺术形式──电子游戏。游戏结合了上述传统艺术以及近代科技派生的其他艺术(如摄影、电影、动画),并且完全脱离了艺术欣赏这种单向传递的方式──游戏必然是互动的,“玩家”并不是“读者”、“观众”或“听众”,而是进入游戏世界、感知并对世界做出反应的参与者。 基于游戏的互动本质,游戏的制作通常比其他大众艺术复杂。商业游戏的制作通常需要各种人才的参与,而他们则需要依赖各种工具及科技。游戏引擎便是专门为游戏而设计的工具及科技集成。之所以称为引擎,如同交通工具中的引擎,提供了最核心的技术部分。因为复杂,研发成本高,人们不希望制作每款游戏(或车款)时都重新设计引擎,重用性是游戏引擎的一个重要设计目标。 然而,各游戏本身的性质以及平台的差异,使研发完全通用的游戏引擎变得极困难,甚至不可能。市面上出售的游戏引擎,有一些虽然已经达到很高的技术水平,但在商业应用中,很多时候还是需要因应个别游戏项目对引擎改造、整合、扩展及优化。因此,即使能使用市面上最好的商用引擎或自研引擎,我们仍需要理解当中的架构、各种机制和技术,并且分析及解决在制作中遇到的问题。这些也是译者曾任于上海两家工作室时的主要工作范畴。 选择翻译此著作,主要原因是在阅读中得到共鸣,并且能知悉一些知名游戏作品实际上所采用的方案。有感坊间大部分游戏开发书籍并不是由业内人士执笔,内容只足够应付一些最简单的游戏开发,欠缺宏观比较各种方案,技术与当今实际情况也有很大差距。而一些Gems类丛书虽然偶有好文章,但受形式所限欠缺系统性、全面性。难得本书原作者身为世界一流游戏工作室的资深游戏开发者(注1),在繁重的游戏开发工作外,还在大学教授游戏开发课程以至编写本著作。此外,从与内地同事的交流中,了解到许多从业者不愿意阅读外文书籍。为了普及知识及反馈业界社会,希望能尽绵力。 或许有些人以为本著作是针对单机/游戏机游戏的,并不适合国内以网游为主的环境。但译者认为这是一种误解,许多游戏本身所涉及的技术是具通用性的。例如游戏性相关的游戏性系统、场景管理、人工智能、物理模拟等部分,许多时候也会同时用于网游的前台和后台。现时,一些动作为主、非MMO的国内端游甚至会直接在后台运行传统意义上的游戏引擎。至于前台相关的技术,单机和端游的区别更少。此外,随着近年移动终端的兴起,其硬件性能已超越传统掌上游戏机,开发手游所需的技术与传统掌上游戏机并无太大差异。还可预料,现时单机/游戏机的一些较高级的架构及技术,将在不远的未来着陆移动终端平台。 译者认为,本书涵括游戏开发技术的方方面面,同时适合入门及经验丰富的游戏程序员。书名中的架构二字,并不单是给出一个系统结构图,而是描述每个子系统的需求、相关技术及与其他子系统的关系。对译者本人而言,本书的第11章(动画系统)及第14章(运行时游戏性基础系统)是本书特別精彩之处,含有许多少见于其他书籍的内容。而第10章(渲染引擎)由于是游戏引擎中的一个极大的部分,有限的篇幅可能未能覆盖广度及深度,推荐读者参考[1](注2),人工智能方面也需参考其他专著。 本译作采用LaTeX排版(注3),以Inkscape编译矢量图片。为了令阅读更流畅,内文中的网址都统一改以脚注标示。另外,由于现时游戏开发相关的文献以英文为主,而且游戏开发涉及的知识面很广,本译作尽量以括号形式保留英文术语。为了方便读者查找内容,在附录中增设中英文双向索引(索引条目与原著的不同)。 本人在香港成长学习及工作,至2008年才赴内地游戏工作室工作,不黯内地的中文写作及用字习惯,翻译中曾遇到不少困难。有幸得到出版社人员以及良师益友的帮助,才能完成本译作。特别感谢周筠老师支持本作的提案,并耐心地给予协助及鼓励。编辑张春雨老师和卢鸫翔老师,以及好友余晟给予了大量翻译上的知识及指导。也感谢游戏业界专家云风、大宝和Dave给予了许多宝贵意见。此书的翻译及排版工作比预期更花时间,感谢妻子及儿女们的体谅。此次翻译工作历时三年半,因工作及家庭事宜导致严重延误,唯有在翻译及排版工作上更尽心尽力,希望求得等待此译作的读者们谅解。无论是批评或建议,诚希阁下通过电邮miloyip@gmail.com、新浪微博、豆瓣等渠道不吝赐教。 叶劲峰(Milo Yip) 2013年10月 原作者是顽皮狗(Naughty Dog)《神秘海域(Uncharted)》系列的通才程序员、《最后生还者(The Last of Us)》的首席程序员,之前还曾在EA和Midway工作。 中括号表示引用附录中的参考文献。一些参考条目加入了其中译本的信息。 具体是使用CTEX套装,它是在MiKTeX的基础上增加中文的支持。 前言 最早的电子游戏完全由硬件构成,但微处理器(microprocessor)的高速发展完全改变了游戏的面貌。现在的游戏是在多用途的PC和专门的电子游戏主机(video game console)上玩的,凭借软件带来绝妙的游戏体验。从最初的游戏诞生至今已有半个世纪,但很多人仍然认为游戏是一个未成熟的产业。即使游戏可能是个年轻的产业,若仔细观察,也会发现它正在高速发展。 现时游戏已成为一个上百亿美元的产业,覆盖不同年龄、性别的广泛受众。 千变万化的游戏,可以分为从纸牌游戏到大型多人在线游戏(massively multiplayer online game,MMOG)等多个种类(category)和“类型(genre)”(注1),也可以运行在任何装有微芯片(microchip)的设备上 。你现在可以在PC、手机及多种特别为游戏而设计的手持/电视游戏主机上玩游戏。家用电视游戏通常代表最尖端的游戏科技,又由于它们是周期性地推出新版本,因此有游戏机“世代”(generation)的说法。最新一代(注2)的游戏机包括微软的Xbox 360和索尼的PlayStation 3,但一定不可忽视长盛不衰的PC,以及最近非常流行的任天堂Wii。 最近,剧增的下载式休闲游戏,使这个多样化的商业游戏世界变得更复杂。虽然如此,大型游戏仍然是一门大生意。今天的游戏平台非常复杂,有难以置信的运算能力,这使软件的复杂度得以进一步提升。所有这些先进的软件都需要由人创造出来,这导致团队人数增加,开发成本上涨。随着产业变得成熟,开发团队要寻求更好、更高效的方式去制作产品,可复用软件(reusable software)和中间件(middleware)便应运而生,以补偿软件复杂度的提升。 由于有这么多风格迥异的游戏及多种游戏平台,因此不可能存在单一理想的软件方案。然而,业界已经发展出一些模式 ,也有大量的潜在方案可供选择。现今的问题是如何找到一个合适的方案去迎合某个项目的需要。再进一步,开发团队必须考虑项目的方方面面,以及如何把各方面集成。对于一个崭新的游戏设计,鲜有可能找到一个完美搭配游戏设计各方面的软件包。 现时业界内的老手,入行时都是“开荒牛”。我们这代人很少是计算机科学专业出身(Matt的专业是航空工程、Jason的专业是系统设计工程),但现时很多学院已设有游戏开发的课程和学位。时至今日,为了获取有用的游戏开发信息,学生和开发者必须找到好的途径。对于高端的图形技术,从研究到实践都有大量高质量的信息。可是,这些信息经常不能直接应用到游戏的生产环境,或者没有一个生产级质量的实现。对于图形以外的游戏开发技术,市面上有一些所谓的入门书籍,没提及参考文献就描述很多内容细节,像自己发明的一样。这种做法根本没有用处,甚至经常带有不准确的内容。另一方面,市场上有一些高端的专门领域书籍,例如物理、碰撞、人工智能等。可是,这类书或者啰嗦到让你难以忍受,或者高深到让部分读者无法理解,又或者内容过于零散而难于融会贯通。有一些甚至会直接和某项技术挂钩,软硬件一旦改动,其内容就会迅速过时。 此外,互联网也是收集相关知识的绝佳工具。可是,除非你确实知道要找些什么,否则断链、不准确的资料、质量差的内容也会成为学习障碍。 好在,我们有Jason Gregory,他是一位拥有在顽皮狗(Naughty Dog)工作经验的业界老手,而顽皮狗是全球高度瞩目的游戏工作室之一。Jason在南加州大学教授游戏编程课程时,找不到概括游戏架构的教科书。值得庆幸的是,他承担了这个任务,填补了这个空白。 Jason把应用到实际发行游戏的生产级别知识,以及整个游戏开发的大局编集于本书。他凭经验,不仅融汇了游戏开发的概念和技巧,还用实际的代码示例及实现例子去说明怎样贯通知识来制作游戏。本书的引用及参考文献可以让读者更深入探索游戏开发过程的各方面。虽然例子经常是基于某些技术的,但是概念和技巧是用来实际创作游戏的,它们可以超越个别引擎或API的束缚。 本书是一本我们入行做游戏时想要的书。我们认为本书能让入门者增长知识,也能为有经验者开拓更大的视野。 Jeff Lander(注3) Matthew Whiting(注4) 译注:Genre一词在文学中为体裁。电影和游戏里通常译作类型。不同的游戏类型可见1.2节。 译注:按一般说法,2005年至今属于第7个游戏机世代。这3款游戏机的发行年份为Xbox 360(2005)、PlayStation 3(2006)、Wii(2006)。有关游戏机世代可参考维基百科。 译注:Jeff Lander现时为Darwin 3D公司的首席技术总监、Game Tech公司创始人,曾为艺电首席程序员、Luxoflux公司游戏性及动画技术程序员。 译注:Matthew Whiting现时为Wholesale Algorithms公司程序员,曾为Luxoflux公司首席软件工程师、Insomniac Games公司程序员。 序言 欢迎来到《游戏引擎架构》世界。本书旨在全面探讨典型商业游戏引擎的主要组件。游戏编程是一个庞大的主题,有许多内容需要讨论。不过相信你会发现,我们讨论的深度将足以使你充分理解本书所涵盖的工程理论及常用实践的方方面面。话虽如此,令人着迷的漫长游戏编程之旅其实才刚刚启程。与此相关的每项技术都包含丰富内容,本书将为你打下基础,并引领你进入更广阔的学习空间。 本书焦点在于游戏引擎的技术及架构。我们会探讨商业游戏引擎中,各个子系统的相关理论,以及实现这些理论所需要的典型数据结构、算法和软件接口。游戏引擎与游戏的界限颇为模糊。我们将把注意力集中在引擎本身,包括多个低阶基础系统(low-level foundation system)、渲染引擎(rendering engine)、碰撞系统(collision system)、物理模拟(physics simulation)、人物动画(character animation),及一个我称为游戏性基础层(gameplay foundation layer)的深入讨论。此层包括游戏对象模型(game object model)、世界编辑器(world editor)、事件系统(event system)及脚本系统(scripting system)。我们也将会接触游戏性编程(gameplay programming)的多个方面,包括玩家机制(player mechanics)、摄像机(camera)及人工智能(artificial intelligence,AI)。然而,这类讨论会被限制在游戏性系统和引擎接口范围。 本书可以作为大学中等级游戏程序设计中两到三门课程的教材。当然,本书也适合软件工程师、业余爱好者、自学的游戏程序员,以及游戏行业从业人员。通过阅读本书,资历较浅的游戏程序员可以巩固他们所学的游戏数学、引擎架构及游戏科技方面的知识。专注某一领域的资深程序员也能从本书更为全面的介绍中获益。 为了更好地学习本书内容,你需要掌握基本的面向对象编程概念并至少拥有一些C++编程经验。尽管游戏行业已经开始尝试使用一些新的、令人兴奋的编程语言,然而工业级的3D游戏引擎仍然是用C或C++编写的,任何认真的游戏程序员都应该掌握C++。我们将在第3章重温一些面向对象编程的基本原则,毫无疑问,你还会从本书学到一些C++的小技巧,不过C++的基础最好还是通过阅读[39]、[31]及[32]来获得。如果你对C++已经有点生疏,建议你在阅读本书的同时,最好能重温这几本或者类似书籍。如果你完全没有C++经验,在看本书之前,可以考虑先阅读[39]的前几章,或者尝试学习一些C++的在线教程。 学习编程技能最好的方法就是写代码。在阅读本书时,强烈建议你选择一些特别感兴趣的主题付诸实践。举例来说,如果你觉得人物动画很有趣,那么可以首先安装OGRE,并测试一下它的蒙皮动画示范。接着还可以尝试用OGRE实现本书谈及的一些动画混合技巧。下一步你可能会打算用游戏手柄控制人物在平面上行走。等你能玩转一些简单的东西了,就应该以此为基础,继续前进!之后可以转移到另一个游戏技术范畴,周而复始。这些项目是什么并不重要,重要的是你在实践游戏编程的艺术,而不是纸上谈兵。 游戏科技是一个活生生、会呼吸的家伙 ,永远不可能将之束缚于书本之上 。因此,附加的资源、勘误、更新、示例代码、项目构思等已经发到本书的网站。 目录 推荐序1 iii推荐序2 v译序 vii序言 xvii前言 xix致谢 xxi第一部分 基础 1第1章 导论 31.1 典型游戏团队的结构 41.2 游戏是什么 71.3 游戏引擎是什么 101.4 不同游戏类型中的引擎差异 111.5 游戏引擎概观 221.6 运行时引擎架构 271.7 工具及资产管道 46第2章 专业工具 532.1 版本控制 532.2 微软Visual Studio 612.3 剖析工具 782.4 内存泄漏和损坏检测 792.5 其他工具 80第3章 游戏软件工程基础 833.1 重温C++及最佳实践 833.2 C/C++的数据、代码及内存 903.3 捕捉及处理错误 118第4章 游戏所需的三维数学 1254.1 在二维中解决三维问题 1254.2 点和矢量 1254.3 矩阵 1394.4 四元数 1564.5 比较各种旋转表达方式 1644.6 其他数学对象 1684.7 硬件加速的SIMD运算 1734.8 产生随机数 180第二部分 低阶引擎系统 183第5章 游戏支持系统 1855.1 子系统的启动和终止 1855.2 内存管理 1935.3 容器 2085.4 字符串 2255.5 引擎配置 234第6章 资源及文件系统 2416.1 文件系统 2416.2 资源管理器 251第7章 游戏循环及实时模拟 2777.1 渲染循环 2777.2 游戏循环 2787.3 游戏循环的架构风格 2807.4 抽象时间线 2837.5 测量及处理时间 2857.6 多处理器的游戏循环 2967.7 网络多人游戏循环 304第8章 人体学接口设备(HID) 3098.1 各种人体学接口设备 3098.2 人体学接口设备的接口技术 3118.3 输入类型 3128.4 输出类型 3168.5 游戏引擎的人体学接口设备系统 3188.6 人体学接口设备使用实践 332第9章 调试及开发工具 3339.1 日志及跟踪 3339.2 调试用的绘图功能 3379.3 游戏内置菜单 3449.4 游戏内置主控台 3479.5 调试用摄像机和游戏暂停 3489.6 作弊 3489.7 屏幕截图及录像 3499.8 游戏内置性能剖析 3499.9 游戏内置的内存统计和泄漏检测 356第三部分 图形及动画 359第10章 渲染引擎 36110.1 采用深度缓冲的三角形光栅化基础 36110.2 渲染管道 40410.3 高级光照及全局光照 42610.4 视觉效果和覆盖层 43810.5 延伸阅读 446第11章 动画系统 44711.1 角色动画的类型 44711.2 骨骼 45211.3 姿势 45411.4 动画片段 45911.5 蒙皮及生成矩阵调色板 47111.6 动画混合 47611.7 后期处理 49311.8 压缩技术 49611.9 动画系统架构 50111.10 动画管道 50211.11 动作状态机 51511.12 动画控制器 535第12章 碰撞及刚体动力学 53712.1 你想在游戏中加入物理吗 53712.2 碰撞/物理中间件 54212.3 碰撞检测系统 54412.4 刚体动力学 56912.5 整合物理引擎至游戏 60112.6 展望:高级物理功能 616第四部分 游戏性 617第13章 游戏性系统简介 61913.1 剖析游戏世界 61913.2 实现动态元素:游戏对象 62313.3 数据驱动游戏引擎 62613.4 游戏世界编辑器 627第14章 运行时游戏性基础系统 63714.1 游戏性基础系统的组件 63714.2 各种运行时对象模型架构 64014.3 世界组块的数据格式 65714.4 游戏世界的加载和串流 66314.5 对象引用与世界查询 67014.6 实时更新游戏对象 67614.7 事件与消息泵 69014.8 脚本 70714.9 高层次的游戏流程 726第五部分 总结 727第15章 还有更多内容吗 72915.1 一些未谈及的引擎系统 72915.2 游戏性系统 730参考文献 733中文索引 737英文索引 755 参考文献 Tomas Akenine-Moller, Eric Haines, and Naty Hoffman. Real-Time Rendering (3rd Edition). Wellesley, MA: A K Peters, 2008. 中译本:《实时计算机图形学(第2版)》,普建涛译,北京大学出版社,2004. Andrei Alexandrescu. Modern C++ Design: Generic Programming and Design Patterns Applied. Resding, MA: Addison-Wesley, 2001. 中译本:《C++设计新思维:泛型编程与设计模式之应用》,侯捷/於春景译,华中科技大学出版社,2003. Grenville Armitage, Mark Claypool and Philip Branch. Networking and Online Games: Understanding and Engineering Multiplayer Internet Games. New York, NY: John Wiley and Sons, 2006. James Arvo (editor). Graphics Gems II. San Diego, CA: Academic Press, 1991. Grady Booch, Robert A. Maksimchuk, Michael W. Engel, Bobbi J. Young, Jim Conallen, and Kelli A. Houston. Object-Oriented Analysis and Design with Applications (3rd Edition). Reading, MA: Addison-Wesley, 2007. 中译本:《面向对象分析与设计(第3版)》,王海鹏/潘加宇译,电子工业出版社,2012. Mark DeLoura (editor). Game Programming Gems. Hingham, MA: Charles River Media, 2000. 中译本:《游戏编程精粹 1》, 王淑礼译,人民邮电出版社,2004. Mark DeLoura (editor). Game Programming Gems 2. Hingham, MA: Charles River Media, 2001. 中译本:《游戏编程精粹 2》,袁国忠译,人民邮电出版社,2003. Philip Dutré, Kavita Bala and Philippe Bekaert. Advanced Global Illumination (2nd Edition). Wellesley, MA: A K Peters, 2006. David H. Eberly. 3D Game Engine Design: A Practical Approach to Real-Time Computer Graphics. San Francisco, CA: Morgan Kaufmann, 2001. 国内英文版:《3D游戏引擎设计:实时计算机图形学的应用方法(第2版)》,人民邮电出版社,2009. David H. Eberly. 3D Game Engine Architecture: Engineering Real-Time Applications with Wild Magic. San Francisco, CA: Morgan Kaufmann, 2005. David H. Eberly. Game Physics. San Francisco, CA: Morgan Kaufmann, 2003. Christer Ericson. Real-Time Collision Detection. San Francisco, CA: Morgan Kaufmann, 2005. 中译本:《实时碰撞检测算法技术》,刘天慧译,清华大学出版社,2010. Randima Fernando (editor). GPU Gems: Programming Techniques, Tips and Tricks for Real-Time Graphics. Reading, MA: Addison-Wesley, 2004. 中译本:《GPU精粹:实时图形编程的技术、技巧和技艺》,姚勇译,人民邮电出版社,2006. James D. Foley, Andries van Dam, Steven K. Feiner, and John F. Hughes. Computer Graphics: Principles and Practice in C (2nd Edition). Reading, MA: Addison-Wesley, 1995. 中译本:《计算机图形学原理及实践──C语言描述》,唐泽圣/董士海/李华/吴恩华/汪国平译,机械工业出版社,2004. Grant R. Fowles and George L. Cassiday. Analytical Mechanics (7th Edition). Pacific Grove, CA: Brooks Cole, 2005. John David Funge. AI for Games and Animation: A Cognitive Modeling Approach. Wellesley, MA: A K Peters, 1999. Erich Gamma, Richard Helm, Ralph Johnson, and John M. Vlissiddes. Design Patterns: Elements of Reusable Object-Oriented Software. Reading, MA: Addison-Wesley, 1994. 中译本:《设计模式:可复用面向对象软件的基础》,李英军/马晓星/蔡敏/刘建中译,机械工业出版社,2005. Andrew S. Glassner (editor). Graphics Gems I. San Francisco, CA: Morgan Kaufmann, 1990. Paul S. Heckbert (editor). Graphics Gems IV. San Diego, CA: Academic Press, 1994. Maurice Herlihy, Nir Shavit. The Art of Multiprocessor Programming. San Francisco, CA: Morgan Kaufmann, 2008. 中译本:《多处理器编程的艺术》,金海/胡侃译,机械工业出版社,2009. Roberto Ierusalimschy, Luiz Henrique de Figueiredo and Waldemar Celes. Lua 5.1 Reference Manual. Lua.org, 2006. Roberto Ierusalimschy. Programming in Lua, 2nd Edition. Lua.org, 2006. 中译本:《Lua程序设计(第2版)》,周惟迪译,电子工业出版社,2008. Isaac Victor Kerlow. The Art of 3-D Computer Animation and Imaging (2nd Edition). New York, NY: John Wiley and Sons, 2000. David Kirk (editor). Graphics Gems III. San Francisco, CA: Morgan Kaufmann, 1994. Danny Kodicek. Mathematics and Physics for Game Programmers. Hingham, MA: Charles River Media, 2005. Raph Koster. A Theory of Fun for Game Design. Phoenix, AZ: Paraglyph, 2004. 中译本:《快乐之道:游戏设计的黄金法则》,姜文斌等译,百家出版社,2005. John Lakos. Large-Scale C++ Software Design. Reading, MA: Addison-Wesley, 1995. 中译本:《大规模C++程序设计》,李师贤/明仲/曾新红/刘显明译,中国电力出版社,2003. Eric Lengyel. Mathematics for 3D Game Programming and Computer Graphics (2nd Edition). Hingham, MA: Charles River Media, 2003. Tuoc V. Luong, James S. H. Lok, David J. Taylor and Kevin Driscoll. Internationalization: Developing Software for Global Markets. New York, NY: John Wiley & Sons, 1995. Steve Maguire. Writing Solid Code: Microsoft's Techniques for Developing Bug Free C Programs. Bellevue, WA: Microsoft Press, 1993. 国内英文版:《编程精粹:编写高质量C语言代码》,人民邮电出版社,2009. Scott Meyers. Effective C++: 55 Specific Ways to Improve Your Programs and Designs (3rd Edition). Reading, MA: Addison-Wesley, 2005. 中译本:《Effective C++:改善程序与设计的55个具体做法(第3版)》,侯捷译,电子工业出版社,2011. Scott Meyers. More Effective C++: 35 New Ways to Improve Your Programs and Designs. Reading, MA: Addison-Wesley, 1996. 中译本:《More Effective C++:35个改善编程与设计的有效方法(中文版)》,侯捷译,电子工业出版社,2011. Scott Meyers. Effective STL: 50 Specific Ways to Improve Your Use of the Standard Template Library. Reading, MA: Addison-Wesley, 2001. 中译本:《Effective STL:50条有效使用STL的经验》,潘爱民/陈铭/邹开红译,电子工业出版社,2013. Ian Millington. Game Physics Engine Development. San Francisco, CA: Morgan Kaufmann, 2007. Hubert Nguyen (editor). GPU Gems 3. Reading, MA: Addison-Wesley, 2007. 中译本:《GPU精粹3》,杨柏林/陈根浪/王聪译,清华大学出版社,2010. Alan W. Paeth (editor). Graphics Gems V. San Francisco, CA: Morgan Kaufmann, 1995. C. Michael Pilato, Ben Collins-Sussman, and Brian W. Fitzpatrick. Version Control with Subversion (2nd Edition). Sebastopol , CA: O'Reilly Media, 2008. (常被称作“The Subversion Book”,线上版本.) 国内英文版:《使用Subversion进行版本控制》,开明出版社,2009. Matt Pharr (editor). GPU Gems 2: Programming Techniques for High-Performance Graphics and General-Purpose Computation. Reading, MA: Addison-Wesley, 2005. 中译本:《GPU精粹2:高性能图形芯片和通用计算编程技巧》,龚敏敏译,清华大学出版社,2007. Bjarne Stroustrup. The C++ Programming Language, Special Edition (3rd Edition). Reading, MA: Addison-Wesley, 2000. 中译本《C++程序设计语言(特别版)》,裘宗燕译,机械工业出版社,2010. Dante Treglia (editor). Game Programming Gems 3. Hingham, MA: Charles River Media, 2002. 中译本:《游戏编程精粹3》,张磊译,人民邮电出版社,2003. Gino van den Bergen. Collision Detection in Interactive 3D Environments. San Francisco, CA: Morgan Kaufmann, 2003. Alan Watt. 3D Computer Graphics (3rd Edition). Reading, MA: Addison Wesley, 1999. James Whitehead II, Bryan McLemore and Matthew Orlando. World of Warcraft Programming: A Guide and Reference for Creating WoW Addons. New York, NY: John Wiley & Sons, 2008. 中译本:《魔兽世界编程宝典:World of Warcraft Addons完全参考手册》,杨柏林/张卫星/王聪译,清华大学出版社,2010. Richard Williams. The Animator's Survival Kit. London, England: Faber & Faber, 2002. 中译本:《原动画基础教程:动画人的生存手册》,邓晓娥译,中国青年出版社,2006. 勘误 第1次印册(2014年2月) P.xviii: 译注中 Wholesale Algoithms -> Wholesale Algorithms P.10: 最后一段第一行 微软的媒体播放器 -> 微软的Windows Media Player (多谢读者OpenGPU来函指正) P.15: 1.4.3节第三点 按妞 -> 按钮 (多谢读者一个小小凡人来函指正) P.40: 正文最后一行 按扭 -> 按钮 P.50: 1.7.8节第二节第一行 同是 -> 同时 (多谢读者czfdd来函指正) P.98: 代码 writeExampleStruct(Example& ex, Stream& ex) 中 Stream& ex -> Stream& stream (多谢读者Snow来函指正) P.106: 第一段中有六处 BBS -> BSS,最后一段代码的注释也有同样错误 (多谢读者trout来函指正) P.119: 译注中 软体工程 -> 软件工程 (多谢读者Snow来函指正) P.214: 正文第一段有两处 虚内存 -> 虚拟内存 (多谢读者Snow来函指正) P.216: 脚注24应标明为译注 (多谢读者Snow来函指正) P.221: 第一段代码的第二个断言应为 ASSERT(link.m_pPrev != NULL); (多谢读者Snow来函指正) P.230: 5.4.4.1节 第二段 软体 -> 软件 P.286: 脚注4应标明为译注 (多谢读者Snow来函指正) P.322: 第二段 按扭事件字 -> 按钮事件 P.349: 9.8节第二段第二行两处 部析器 -> 剖析器 (多谢读者Snow来函指正) P.738-572: 双数页页眉 参考文献 -> 中文索引 P.755-772: 双数页页眉 参考文献 -> 英文索引 P.755: kd tree项应归入K而不是Symbols 以上的错误已于第2次印册中修正。 第2次印册及之前 P.11: 第四行 细致程度 -> 层次细节 (这是level-of-detail/LOD的内地通译,多谢读者OpenGPU来函指正) P.12: 正文第一段及图1.2标题 使命之唤 -> 使命召唤 (多谢读者OpenGPU来函指正) P.12: 正文第一段 战栗时空 -> 半条命 (多谢读者OpenGPU来函指正) P.16: 第一点 表面下散射 -> 次表面散射 (多谢读者OpenGPU来函指正) P.17: 1.4.4节第五行 次文化 -> 亚文化 (此译法在内地更常用。多谢读者OpenGPU来函提示) P.22: 战栗时空 -> 半条命 P.24: 战栗时空2 -> 半条命2 P.34: 1.6.8.2节第一行 提呈 -> 提交 (这术语在本书其他地方都写作提交。多谢读者OpenGPU来函提示) P.35: 第七行 提呈 -> 提交 (这术语在本书其他地方都写作提交。多谢读者OpenGPU来函提示) P.50: 战栗时空2 -> 半条命2 P.365: 第四段第二行: 细致程度 -> 层次细节 P.441: 10.4.3.2节第三行 细致程度 -> 层次细节 P.494: sinusiod -> sinusoid (多谢读者OpenGPU来函指正) P.511: 11.10.4节第一行 谈入 -> 淡入 (多谢读者Snow来函指正) P.541: 战栗时空2 -> 半条命2 P.627: 战栗时空2 -> 半条命2 P.654: 第二行 建康值 -> 血量 (原来是改正错别字,但译者发现应改作前后统一使用的“血量”。多谢读者Snow来函指正) P.692: 第二行 内部分式 -> 内部方式 (多谢读者Snow来函指正) P.696: 14.7.6节第四行 不设实际 -> 不切实际 (多谢读者Snow来函指正) 以上的错误已于第3次印册中修正。 其他意见 P.220: 正文第一段 m_root.m_pElement 和 P.218 第一段代码中的 m_pElem 不统一。原文有此问题,但因为它们是不同的struct,暂不列作错误。 (多谢读者Snow来函提示) P.331: 8.5.8节第二段中 “反覆”较常见的写法为“反复”,但前者也是正确的,暂不列作错误。 (多谢读者Snow来函提示) P.390: 10.1.3.3节静态光照第二段中“取而代之,我们会使用一张光照纹理贴到所有受光源影响范围内的物体上。这样做能令动态物体经过光源时得到正确的光照。” 后面的一句与前句好像难以一起理解。译者认为,作者应该是指,使用同一静态光源去为静态物件生成光照纹理,以及用于动态对象的光照,能使两者的效果维持一致性。译者会考虑对译文作出改善或加入译注解译。(多谢读者店残来函查询) P.689: 第五行 并行处理世代 -> 并行处理时代 是对era较准确的翻译。 (多谢读者Snow来函提示) 本篇文章为转载内容。原文链接:https://blog.csdn.net/mypongo/article/details/38388381。 该文由互联网用户投稿提供,文中观点代表作者本人意见,并不代表本站的立场。 作为信息平台,本站仅提供文章转载服务,并不拥有其所有权,也不对文章内容的真实性、准确性和合法性承担责任。 如发现本文存在侵权、违法、违规或事实不符的情况,请及时联系我们,我们将第一时间进行核实并删除相应内容。
2023-02-12 23:04:05
327
转载
JQuery插件下载
...示加载状态的场景。它采用了易于控制的jQuery代码实现,使得用户可以轻松地调整和定制进度条的外观与行为。特别之处在于其独特的斑马线样式设计,这一效果完全依赖于现代CSS3技术来呈现,确保了流畅且美观的视觉体验。对于那些尚未全面支持CSS3的老旧浏览器,插件还提供了备选方案,即用图片来模拟斑马线效果,从而保证了在不同设备和浏览器上的兼容性和一致性。使用这款插件,开发者不仅可以快速提升网站或应用的用户体验,还能减少开发工作量,因为它已经内置了多种优化措施。无论是用于网站加载页面、文件上传过程中的进度显示,还是任何需要动态更新和视觉反馈的场景,这款jQuery斑马线样式加载进度条都能完美胜任,为用户提供直观、美观的加载指示。此外,由于其轻量级的设计,不会给页面加载带来额外负担,确保了整体性能不受影响。总之,这是一款功能强大、操作简便、适应性广的加载进度条插件,非常适合追求高效开发和优质用户体验的前端开发者选用。 点我下载 文件大小:57.98 KB 您将下载一个JQuery插件资源包,该资源包内部文件的目录结构如下: 本网站提供JQuery插件下载功能,旨在帮助广大用户在工作学习中提升效率、节约时间。 本网站的下载内容来自于互联网。如您发现任何侵犯您权益的内容,请立即告知我们,我们将迅速响应并删除相关内容。 免责声明:站内所有资源仅供个人学习研究及参考之用,严禁将这些资源应用于商业场景。 若擅自商用导致的一切后果,由使用者承担责任。
2024-12-27 11:05:34
99
本站
JQuery插件下载
...的颜色选择器相呼应,采用直观且易于理解的界面设计,使得用户能够快速找到所需颜色。其简洁的界面设计不仅美观,还能有效减少用户学习成本,提升工作效率。灵活性与易用性支持多种调用方式,允许开发者根据项目需求灵活集成到不同场景中。无论是嵌入到表单输入框内,还是作为独立组件使用,都能轻松适应。此外,插件的API设计简单明了,便于开发者进行二次开发和自定义配置,满足个性化需求。小巧与高效作为一款注重性能的插件,"ps样式的jQuery颜色选择插件"体积小,加载速度快,对页面性能影响极小。它优化了资源使用,确保即使在资源受限的环境中也能流畅运行,适用于各种规模的项目。应用场景广泛无论是用于设计网站的配色方案,还是在开发应用程序中需要用户选择颜色的功能,此插件都能提供无缝的体验。它适用于任何需要用户与颜色进行交互的场景,极大地提升了用户体验和界面美观度。总之,"ps样式的jQuery颜色选择插件"凭借其独特的外观设计、丰富的调用方式、高效的性能表现以及广泛的适用性,成为了一个不可多得的jQuery颜色选择器解决方案,为开发者和设计师提供了强大的工具支持。 点我下载 文件大小:91.83 KB 您将下载一个JQuery插件资源包,该资源包内部文件的目录结构如下: 本网站提供JQuery插件下载功能,旨在帮助广大用户在工作学习中提升效率、节约时间。 本网站的下载内容来自于互联网。如您发现任何侵犯您权益的内容,请立即告知我们,我们将迅速响应并删除相关内容。 免责声明:站内所有资源仅供个人学习研究及参考之用,严禁将这些资源应用于商业场景。 若擅自商用导致的一切后果,由使用者承担责任。
2024-09-24 20:52:58
70
本站
HTML
...观的同时,最大限度地减少动画效果对页面加载速度和CPU使用率的影响。一些轻量级的JavaScript库如Snowfall.js,可以高效生成逼真的下雪效果,并能根据设备性能自动调整雪花数量和运动轨迹,实现了美感与性能的兼顾。 同时,在响应式网页设计领域,针对不同设备和屏幕尺寸自适应的下雪特效也成为研究热点。设计师们正在不断尝试创新方法,使得雪花飘落效果在移动端小屏上同样流畅自然,从而确保跨平台、跨设备的一致性用户体验。 总的来说,网页下雪特效作为增强网页视觉吸引力和互动性的一种手段,其背后的实现原理和技术趋势值得广大前端开发者持续关注与深入学习。而通过不断跟踪最新的实战案例与技术解析,我们将更好地理解并应用这些技术,创造出更为惊艳且高效的网页动态效果。
2023-08-21 12:02:08
458
软件工程师
Java
...进一步探究其重要性和影响力。 近期,随着Spring框架5.x版本的广泛采用以及模块化编程的日益流行,访问控制权限的重要性更加凸显。在大型企业级项目中,开发者必须精确控制类、接口和成员变量的可见性,以保证代码的封装性和安全性。例如,在微服务架构中,每个服务模块内部定义的核心业务逻辑通常会被设置为private或package-private(default),以避免被外部模块随意访问,从而降低耦合度和潜在的安全风险。 同时,protected访问控制在面向对象设计中的角色也愈发关键。在实现组件复用和继承时,父类通过protected成员变量和方法向子类提供了一种安全而灵活的扩展机制。如在JDK新特性记录中,有开发者利用protected修饰符优化了框架内部组件的设计,使子类可以方便地重写和扩展特定功能,而不影响原有框架结构的稳定性。 此外,对于开源社区而言,public API的设计直接关系到库的易用性和兼容性。许多开源库在迭代更新过程中,会严格限制新增API的访问级别,尽量减少对外暴露的public接口,转而推荐使用protected或默认访问权限的方法来指导用户按照最佳实践进行扩展开发。 综上所述,掌握Java中的访问控制修饰符并合理运用,不仅有助于编写出更安全、高效、易于维护的代码,更能顺应现代软件工程的发展趋势,适应复杂多变的技术生态。
2023-05-18 18:06:08
371
键盘勇士
JQuery
...其便捷高效的特性持续影响着开发者。最近,随着Web技术的快速发展,诸如Vue.js、React和Angular等现代前端框架日渐流行,但jQuery仍然在许多项目中扮演着重要角色,特别是在简化DOM操作、事件处理以及Ajax请求等方面。 实际上,在音乐网站及多媒体应用开发中,jQuery结合HTML5 Audio/Video API可以构建出功能丰富的播放器组件。例如,除了基础的播放、暂停和音量控制,还可以实现进度条拖拽、播放列表管理、循环播放模式切换等功能。同时,jQuery插件生态系统如jPlayer、MediaElement.js等为音乐播放器提供了更强大的定制化解决方案。 此外,值得注意的是,尽管现代浏览器原生支持音频播放功能,但在不同浏览器间的兼容性和用户体验一致性上,jQuery能够提供有效的辅助。例如,通过封装复杂的跨浏览器兼容性代码,确保音乐播放功能在各种环境下都能顺畅运行。 近期,jQuery团队仍在不断更新维护,旨在保持其在现有项目中的稳定性和对最新Web标准的支持。因此,无论是对于正在使用或考虑采用jQuery进行音乐网站开发的开发者来说,深入理解并掌握这一库的应用技巧,无疑将极大地提升项目的开发效率与用户体验。同时,关注前沿技术动态,灵活结合各类前端工具与框架,也将成为优化音乐网站性能的关键所在。
2023-09-30 11:47:52
298
数据库专家
MySQL
...优化对提升玩家体验的影响。近期,《游戏开发者》杂志的一篇文章揭示了某知名网络游戏通过优化数据库架构,成功减少了游戏内交易的延迟,显著提升了元宝充值、消耗等操作的实时性,从而提高了用户满意度和留存率。 同时,随着云计算和大数据技术的发展,许多游戏公司开始采用分布式数据库来应对高并发场景下的数据处理需求。例如,阿里云发布的最新解决方案中就详细介绍了如何借助云数据库实现动态扩容,有效支撑了大型网游在高峰期的海量元宝数值更新与查询请求。 此外,针对游戏经济系统的安全问题,也有专家提出应当强化数据库权限管理,采用加密传输技术和二次验证机制确保元宝等虚拟财产的安全存储与变更。最近一起因数据库漏洞导致的游戏元宝被盗事件,再次敲响了游戏数据安全的警钟,促使业界加大对数据库防护措施的研究和投入。 总的来说,从基本的MySQL操作到复杂的数据库架构设计与优化,再到数据安全防护,游戏开发过程中对于数据库技术的应用和探索是一个持续且深入的过程,它不仅影响着游戏功能的实现,更是关乎游戏生态健康与用户体验的关键因素。
2023-04-20 08:05:28
62
软件工程师
Java
...的数量对性能有很大的影响。如果哈希函数不好,会导致槽位中的元素数量过多,从而降低性能。因此,在使用HashMap和HashSet时,应该尽可能保证键或元素的哈希函数是高质量的。
2023-10-10 17:34:26
308
编程狂人
CSS
...定性,为此推荐开发者采用“内联关键CSS”或“CSS资源优先级标记”的方法,以确保快速渲染初始视口内容。 近期一项研究显示,将关键CSS直接内嵌在HTML文档头部可以显著提升页面交互性,尤其对于移动端用户而言,这种做法能够有效减少“首次输入延迟”(First Input Delay, FID),从而提高用户体验评分。同时,针对非关键或者按需加载的CSS资源,则可通过异步加载方式,在不影响首屏内容展示的情况下完成样式更新。 然而,将CSS置于底部并非一劳永逸的解决方案,它可能导致“无样式内容闪烁”(Flash of Unstyled Content, FOUC)的问题,影响用户体验。因此,更现代的做法是结合最新的前端性能优化工具和技术,如Webpack、Parcel等进行代码分割与智能加载,并利用浏览器缓存机制进一步加快重复访问时的页面加载速度。 此外,值得深入探究的是如何在保证页面加载速度的同时,兼顾SEO优化及无障碍阅读的需求。一些最佳实践指出,合理布局CSS并遵循语义化HTML标准,既能改善搜索引擎抓取效率,也能增强辅助技术对网站内容的理解和呈现,最终实现多维度的网页性能优化目标。
2023-12-20 17:00:57
449
软件工程师
Java
...的长时间暂停。ZGC采用颜色指针和读屏障等先进技术,实现了并发标记与整理,极大地提升了大规模应用在高并发、低延迟场景下的性能表现。 同时,OpenJDK社区也在持续优化其他垃圾回收器。例如,Shenandoah GC是OpenJDK的一个实验性项目,它通过使用“并发压缩”技术来减少GC暂停时间,适用于那些无法接受长时间STW(Stop-The-World)的应用程序。尽管其设计理念与ZGC有相似之处,但Shenandoah更加注重降低中等规模堆内存环境下的停顿时间。 此外,对于云原生和容器化环境下的Java应用,新一代的Epsilon垃圾回收器提供了“无操作”模式,仅专注于资源占用最小化,特别适合于短生命周期或对响应时间要求极为严格的微服务场景。 综上所述,随着技术的发展,Java垃圾回收领域的研究和创新从未止步,不断为开发者提供更高效、更灵活的内存管理工具,以适应日益复杂的软件系统需求。对于系统管理员和技术决策者而言,紧跟这些最新的垃圾回收技术动态,结合实际业务场景进行合理选择和调优,是提升系统整体性能和稳定性的关键所在。
2023-11-22 10:36:57
339
逻辑鬼才
VUE
...用于构建用户界面。它采用组件化的方式来组织和复用代码,通过数据绑定和指令系统提供了声明式的数据绑定及DOM操作方式。在本文中,Vue.js因其组件化和数据响应式特性被广泛应用在复杂Web应用开发中,并且提供了一系列内置的性能优化技术,如计算属性、过滤器等,以解决随着页面复杂度提高而带来的噪点问题,提升页面性能。 计算属性(Computed Properties) , 在Vue.js框架中,计算属性是一种特殊的属性,它的值依赖于其他数据的变化并自动进行重新计算。开发者可以定义一个计算属性方法,当其依赖的数据发生变化时,Vue会自动调用该方法来更新视图。在文章中,计算属性被用来处理含有噪点的数据,通过封装复杂的逻辑处理,确保渲染的是经过优化后的数据,从而避免了不必要的重复计算和渲染,提升了页面性能。 过滤器(Filters) , Vue.js中的过滤器主要用于数据预处理,它们可以在Vue模板表达式中方便地对变量的值进行格式化或转换。过滤器通常应用于展示层,例如对文本进行格式化、对数组进行筛选或排序等操作。在本文上下文中,过滤器作为一种去噪技术,被用来对原始数据进行筛选、排序、去重等处理,减少页面渲染的工作量,从而优化页面性能。 去噪技术(Noise Reduction Techniques) , 在前端开发领域,去噪技术主要是指通过特定的方法去除影响页面性能的无效、冗余或无关的数据,这些数据被称为“噪点”。在Vue.js中,通过使用计算属性和过滤器等机制,开发者能够有针对性地清理和优化需要渲染的数据,降低页面渲染负担,进而提升页面加载速度和运行流畅度。
2023-10-30 09:32:35
105
算法侠
Python
...过优化垃圾回收机制以减少内存泄漏的风险,这使得开发者在处理大数据或长时间运行任务时能更好地把控程序内存占用情况。 同时,针对多线程编程中的安全问题,Python 3.9版本引入了新的并发工具与同步原语,如asyncio库的增强和contextvars模块的完善,帮助开发者更方便地处理多线程间的资源竞争和互斥问题,从而降低因并发控制不当引发段错误的可能性。 此外,对于递归深度过大的问题,除了限制递归调用层数外,还可以采用尾递归优化、循环替代递归等编程技巧,或者利用堆栈检查机制预防栈溢出。例如,一些现代Python解释器已经开始支持尾递归优化,为深递归场景提供更好的解决方案。 实践层面,Google V8引擎团队最近分享了一篇关于JavaScript(其内存管理和Python有相似之处)中的内存泄漏检测和修复策略的文章,其中的很多方法论同样适用于Python开发人员,有助于他们在实际项目中排查并修复潜在的段错误源头。 综上所述,持续关注Python语言的最新发展动态和技术文章,结合理论知识与实践经验,将有助于我们编写出更为健壮、稳定且高效的Python应用程序,有效规避诸如段错误这类严重影响程序运行的问题。
2023-06-07 20:35:26
132
算法侠
转载文章
...的排序组件时,就考虑采用了计数排序等非比较型排序算法以提升系统性能。研究人员发现,通过针对性地分析数据分布特征,并适时引入计数排序算法,可以在不影响稳定性的同时显著减少排序所需的时间成本。 然而,对于浮点数或数据范围极大的情况,计数排序则可能因为需要创建极大空间的计数数组而导致空间效率低下。因此,在实际应用中,往往需要结合其他高效排序算法(如快速排序、归并排序等)进行混合使用,根据实际情况灵活选择最优策略。 此外,深入探究排序算法背后的理论基础也十分有益,例如Knuth在其经典著作《计算机程序设计艺术》中对各种排序算法进行了详尽而深入的解读,其中包括计数排序的设计原理及其在实际问题中的应用场景分析。学习这些理论知识将有助于我们更好地理解并运用计数排序以及其他各类排序算法,从而在面对不同的工程问题时能够做出更为精准有效的决策。
2023-10-02 13:00:57
130
转载
站内搜索
用于搜索本网站内部文章,支持栏目切换。
知识学习
实践的时候请根据实际情况谨慎操作。
随机学习一条linux命令:
systemctl start|stop|restart|status service_name
- 管理systemd服务。
推荐内容
推荐本栏目内的其它文章,看看还有哪些文章让你感兴趣。
2023-04-28
2023-08-09
2023-06-18
2023-04-14
2023-02-18
2023-04-17
2024-01-11
2023-10-03
2023-09-09
2023-06-13
2023-08-07
2023-03-11
历史内容
快速导航到对应月份的历史文章列表。
随便看看
拉到页底了吧,随便看看还有哪些文章你可能感兴趣。
时光飞逝
"流光容易把人抛,红了樱桃,绿了芭蕉。"