阿里IM技术分享(六):闲鱼亿级IM消息系统的离线推送到达率优化
本文由阿里闲鱼技术团队逸昂分享,原题“消息链路优化之弱感知链路优化”,有修订和改动,感谢作者的分享。
为元氏等地区用户提供了全套网页设计制作服务,及元氏网站建设行业解决方案。主营业务为做网站、网站建设、元氏网站设计,以传统方式定制建设网站,并提供域名空间备案等一条龙服务,秉承以专业、用心的态度为用户提供真诚的服务。我们深信只要达到每一位用户的要求,就会得到认可,从而选择与我们长期合作。这样,我们也可以走得更远!
闲鱼的IM消息系统作为买家与卖家的沟通工具,增进理解、促进信任,对闲鱼的商品成交有重要的价值,是提升用户体验最关键的环节。
然而,随着业务体量的快速增长,当前这套消息系统正面临着诸多急待解决的问题。
以下几个问题典型最为典型:
1) 在线消息的体验提升;
2) 离线推送的到达率;
3) 消息玩法与消息底层系统的耦合过强。
经过评估,我们认为现阶段离线推送的到达率问题最为关键,对用户体验影响较大。
本文将要分享的是闲鱼IM消息在解决离线推送的到达率方面的技术实践,内容包括问题分析和技术优化思路等 ,希望能带给你启发。
(本文已同步发布于: )
本文是系列文章的第6篇,总目录如下:
《 阿里IM技术分享(一):企业级IM王者——钉钉在后端架构上的过人之处 》
《 阿里IM技术分享(二):闲鱼IM基于Flutter的移动端跨端改造实践 》
《 阿里IM技术分享(三):闲鱼亿级IM消息系统的架构演进之路 》
《 阿里IM技术分享(四):闲鱼亿级IM消息系统的可靠投递优化实践 》
《 阿里IM技术分享(五):闲鱼亿级IM消息系统的及时性优化实践 》
《 阿里IM技术分享(六):闲鱼亿级IM消息系统的离线推送到达率优化 》(* 本文)
从数据通信链接的技术角度,我们根据闲鱼客户端是否在线,将整体消息链路大致分为强感知链路和弱感知链路。
强感知链路由以下子系统或模块:
1) 发送方客户端;
2) idleapi-message(闲鱼的消息网关);
3) heracles(闲鱼的消息底层服务);
4) accs(阿里自研的长连接通道);
5) 接收方客户端组成。
整条链路的核心指标在于端到端延迟和消息到达率。
强感知链路中的双方都是在线的,消息到达客户端就可以保证接收方感知到。强感知链路的主要痛点在消息的端到端延迟。
弱感知链路与强感知链路的主要不同在于: 弱感知链路的接收方是离线的,需要依赖离线推送这样的方式送达。
因此弱感知链路的用户感知度不强,其核心指标在于消息的到达率,而非延迟。
所以当前阶段,优化弱感知链路的重点也就是提升离线消息的到达率。换句话说, 提升离线消息到达率问题,也就是优化弱感知链路本身 。
下图一张整个IM消息系统的架构图,感受下整体链路:
如上图所示,各主要组件和子系统分工如下:
1) HSF是一个远程服务框架,是dubbo的内部版本;
2) tair是阿里自研的分布式缓存框架,支持 memcached、Redis、LevelDB 等不同存储引擎;
3) agoo是阿里的离线推送中台,负责整合不同厂商的离线推送通道,向集团用户提供一个统一的离线推送服务;
4) accs是阿里自研的长连接通道,为客户端、服务端的实时双向交互提供便利;
5) lindorm是阿里自研的NoSQL产品,与HBase有异曲同工之妙;
6) 域环是闲鱼消息优化性能的核心结构,用来存储用户最新的若干条消息。
强感知链路和弱感知链路在通道选择上是不同的:
1) 强感知链路使用accs这个在线通道;
2) 弱感知链路使用agoo这个离线通道。
通俗了说,弱感知链路指的就是离线消息推送系统。
相比较于在线消息和端内推送(也就是上面说的强感知链路),离线推送难以确保被用户感知到。
典型的情况包括:
1) 未发送到用户设备:即推送未送达用户设备,这种情况可以从通道的返回分析;
2) 发送到用户设备但没有展示到系统通知栏:闲鱼曾遇到通道返回成功,但是用户未看到推送的案例;
3) 展示到通知栏,并被系统折叠:不同安卓厂商对推送的折叠策略不同,被折叠后,需用户主动展开才能看到内容,触达效果明显变差;
4) 展示到通知栏,并被用户忽略:离线推送的点击率相比于在线推送更低。
针对“1)未发送到用户设备”,原因有:
1) 离线通道的token失效;
2) 参数错误;
3) 用户关闭应用通知;
4) 用户已卸载等。
针对“3)展示到通知栏,并被系统折叠”,原因有:
1) 通知的点击率;
2) 应用在厂商处的权重;
3) 推送的数量等。
针对“4)展示到通知栏,并被用户忽略”,原因有:
1) 用户不愿意查看推送;
2) 用户看到了推送,但是对内容不感兴趣;
3) 用户在忙别的事,无暇处理。
总之: 以上这些离线消息推送场景,对于用户来说感知度不高,我们也便称之为弱感知链路。
我们的弱感知链路分为3部分,即:
1) 系统;
2) 通道;
3) 用户。
共包含了Hermes、agoo、厂商、设备、用户、承接页这几个环节。具体如下图所示。
从推送的产生到用户最终进入APP,共分为如下几个步骤:
步骤1 :Hermes是闲鱼的用户触达系统,负责人群管理、内容管理、时机把控,是整个弱感知链路的起点。;
步骤2 :agoo是阿里内部承接离线推送的中台,是闲鱼离线推送能力的基础;
步骤3 :agoo实现离线推送依靠的是厂商的推送通道(如:苹果的 apns通道 、Google的fcm通道、及 国内各厂商的自建通道 。;
步骤4 :通过厂商的通道,推送最终出现在用户的设备上,这是用户能感知到推送的前提条件;
步骤5 :如果用户刚巧看到这条推送,推送的内容也很有趣,在用户的主动点击下会唤起APP,打开承接页,进而给用户展示个性化的商品。
经过以上5个步骤,至此弱感知链路就完成了使命。
弱感知链路的核心问题在于:
1) 推送的消息是否投递给了用户;
2) 已投递到的消息用户是否有感知。
这对应推送的两个阶段:
1) 推送消息是否已到达设备;
2) 用户是否查看推送并点击。
其中: 到达设备这个阶段是最基础的,也是本次优化的核心。
我们可以将每一步的消息处理量依次平铺,展开为一张漏斗图,从而直观的查看链路的瓶颈。
漏斗图斜率最大的地方是优化的重点,差异小的地方不需要优化:
通过分析以上漏斗图,弱感知链路的优化重点在三个方面:
1) agoo受理率:是指我们发送推送请到的数量到可以通过agoo(阿里承接离线推送的中台)转发到厂商通道的数量之间的漏斗;
2) 厂商受理率:是指agoo中台受理的量到厂商返回成功的量之间的漏斗;
3) Push点击率:也就通过以上通道最终已送到到用户终端的消息,是否最终转化为用户的主动“点击”。
有了优化方向,我们来看看优化手段吧。
跟随推送的视角,顺着链路看一下我们是如何进行优化的。
用户的推送,从 Hermes 站点搭乘“班车”,驶向下一站: agoo 。
这是推送经历的第一站。到站一看,傻眼了,只有不到一半的推送到站下车了。这是咋回事嘞?
这就要先说说 agoo 了,调用 agoo 有两种方式:
1) 指定设备和客户端,agoo直接将推送投递到相应的设备;
2) 指定用户和客户端,agoo根据内部的转换表,找到用户对应的设备,再进行投递。
我们的系统不保存用户的设备信息。因此,是按照用户来调用agoo的。
同时: 由于没有用户的设备信息,并不知道用户是 iOS 客户端还是 Android 客户端。工程侧不得不向 iOS 和 Android 都发送一遍推送。虽然保证了到达,但是,一半的调用都是无效的。
为了解这个问题: 我们使用了agoo的设备信息。将用户转换设备这一阶段提前到了调用 agoo 之前,先明确用户对应的设备,再指定设备调用 agoo,从而避免无效调用。
agoo调用方式优化后,立刻剔除了无效调用,agoo受理率有了明显提升。
至此: 我们总算能对 agoo 受理失败的真正原因做一个高大上的分析了。
根据统计: 推送被 agoo 拒绝的主要原因是——用户关闭了通知权限。同时,我们对 agoo 调用数据的进一步分析发现——有部分用户找不到对应的设备。 优化到此,我们猛然发现多了两个问题。
那就继续优化呗:
1) 通知体验优化,引导打开通知权限;
2) 与agoo共建设备库,解决设备转换失败的问题。
这两个优化方向又是一片新天地,我们择日再聊。
推送到达 agoo ,分机型搭乘厂商“专列”,驶向下一站:用户设备。
这是推送经历的第二站。出站查票,发现竟然超员了。
于是乎: 我们每天有大量推送因为超过厂商设定的限额被拦截。
为什么会这样呢?
实际上: 提供推送通道的厂商(没错, 各手机厂商的自家推送通道良莠不齐 ),为了保证用户体验,会对每个应用能够推送的消息总量进行限制。
对于厂商而言,这个限制会根据推送的类型和应用的用户规模设定——推送主要分为产品类的推送和营销类的推送。
厂商推送通道对于不同类型消息的限制是:
1) 对于产品类推送,厂商会保证到达;
2) 对于营销类推送,厂商会进行额度限制;
3) 未标记的推送,默认作为营销类推送对待。
我们刚好没有对推送进行标记,因此触发了厂商的推送限制。
这对我们的用户来说,会带来困扰。闲鱼的交易,很依赖买卖家之间的消息互动。这部分消息是需要确保到达的。
同样: 订单类的消息、用户的关注,也需要保证推送给用户。
根据主流厂商的接口协议,我们将推送的消息分为以下几类,并进行相应标记:
1) 即时通讯消息;
2) 订单状态变化;
3) 用户关注内容;
4) 营销消息这几类。
同时,在业务上,我们也进行了推送的治理——将用户关注度不高的消息,取消推送,避免打扰。
经过这些优化,因为超过厂商限额而被拦截的推送实现了清零。
通过优化agoo受理率、厂商受理率,我们解决了推送到达量的瓶颈。但即使消息被最终送达,用户到底点击了没有?这才是消息推送的根本意义所在。
于是,在日常的开发测试过程中,我们发现了推送的两个体验问题:
1) 用户点击Push有开屏广告;
2) 营销Push也有权限校验,更换用户登陆后无法点击。
对于开屏广告功能,我们增加了Push点击跳过广告的能力。
针对Push的权限校验功能,闲鱼根据场景做了细分:
1) 涉及个人隐私的推送,保持权限校验不变;
2) 营销类的推送,放开权限校验。
以上是点击体验的优化,我们还需要考虑用户的点击意愿。
用户点击量与推送的曝光量、推送素材的有趣程度相关。推送的曝光量又和推送的到达量、推送的到达时机有关。
具体的优化手段是:
1) 在推送内容上:我们需要优化的是推送的时机和相应的素材;
2) 在推送时机上:算法会根据用户的偏好和个性化行为数据,计算每个用户的个性化推送时间,在用户空闲的时间推送(避免在不合适的时间打扰用户,同时也能提升用户看到推送的可能性)。
3) 在推送素材上:算法会根据素材的实时点击反馈,对素材做实时赛马。只发用户感兴趣的素材,提高用户点击意愿。
通过以上我们的分析和技术优化手段,整体弱推送链路链路有了不错的提升,离线消息的到达率相对提升了两位数。
本篇主要和大家聊的是只是IM消息系统链路中的一环——弱感知链路的优化,落地到到具体的业务也就是离线消息送达率问题。
整体IM消息系统,还是一个比较复杂的领域。
我们在消息系统的发展过程中,面临着如下问题:
1) 如何进行消息的链路追踪;
2) 如何保证IM消息的快速到达(见《 闲鱼亿级IM消息系统的及时性优化实践 》);
3) 如何将消息的玩法和底层能力分离;
4) 离线推送中如何通过用户找到对应的设备。
这些问题,我们在以前的文章中有所分享,以后也会陆续分享更多,敬请期待。
[1] Android P正式版即将到来:后台应用保活、消息推送的真正噩梦
[2] 一套高可用、易伸缩、高并发的IM群聊、单聊架构方案设计实践
[3] 一套亿级用户的IM架构技术干货(上篇):整体架构、服务拆分等
[4] 一套亿级用户的IM架构技术干货(下篇):可靠性、有序性、弱网优化等
[5] 从新手到专家:如何设计一套亿级消息量的分布式IM系统
[6] 企业微信的IM架构设计揭秘:消息模型、万人群、已读回执、消息撤回等
[7] 融云技术分享:全面揭秘亿级IM消息的可靠投递机制
[8] 移动端IM中大规模群消息的推送如何保证效率、实时性?
[9] 现代IM系统中聊天消息的同步和存储方案探讨
[10] 新手入门一篇就够:从零开发移动端IM
[11] 移动端IM开发者必读(一):通俗易懂,理解移动网络的“弱”和“慢”
[12] 移动端IM开发者必读(二):史上最全移动弱网络优化方法总结
[13] IM消息送达保证机制实现(一):保证在线实时消息的可靠投递
[14] IM消息送达保证机制实现(二):保证离线消息的可靠投递
[15] 零基础IM开发入门(一):什么是IM系统?
[16] 零基础IM开发入门(二):什么是IM系统的实时性?
[17] 零基础IM开发入门(三):什么是IM系统的可靠性?
[18] 零基础IM开发入门(四):什么是IM系统的消息时序一致性?
(本文已同步发布于: )
生活中的英语(例如:广告,产品名称,口头语,网络语)
1。如果只是遇见,不能停留,不如不遇见。 If we can only encounter each other rather than stay with each other,then I wish we had never encountered. 2。宁愿笑着流泪,也不哭着说后悔。心碎了,还需再补吗? I would like weeping with the smile rather than repenting with the cry,when my heart is broken ,is it needed to fix? 3。没有谁对不起谁,只有谁不懂得珍惜谁。 No one indebted for others,while many people don't know how to cherish others. 4。命里有时钟需有 命里无时莫强求 You will have it if it belongs to you,whereas you don't kvetch for it if it doesn't appear in your life. 5。当香烟爱上火柴时,就注定受到伤害 When a cigarette falls in love with a match,it is destined to be hurt. 6。爱情…在指缝间承诺 指缝…。在爱情下交缠。 Love ,promised between the fingers Finger rift,twisted in the love 7。没有人值得你流泪,值得让你这么做的人不会让你哭泣。 No man or woman is worth your tears, and the one who is, won’t make you cry. 8。记住该记住的,忘记该忘记的。改变能改变的,接受不能改变的。 Remember what should be remembered, and forget what should be forgotten.Alter what is changeable, and accept what is unchangeable. Love is like a butterfly. It goes where it pleases and it pleases where it goes. 爱情就像一只蝴蝶,它喜欢飞到哪里,就把欢乐带到哪里。 If I had a single flower for every time I think about you, I could walk forever in my garden. 假如每次想起你我都会得到一朵鲜花,那么我将永远在花丛中徜徉。 Within you I lose myself, without you I find myself wanting to be lost again. 有了你,我迷失了自我。失去你,我多么希望自己再度迷失。 At the touch of love everyone becomes a poet. 每一个沐浴在爱河中的人都是诗人。 Look into my eyes - you will see what you mean to me. 看看我的眼睛,你会发现你对我而言意味着什么。 Distance makes the hearts grow fonder. 距离使两颗心靠得更近。 I need him like I need the air to breathe. 我需要他,正如我需要呼吸空气。 If equal affection cannot be, let the more loving be me. 如果没有相等的爱,那就让我爱多一些吧。 Love is a vine that grows into our hearts. 爱是长在我们心里的藤蔓。 If I know what love is, it is because of you. 因为你,我懂得了爱。 Love is the greatest refreshment in life. 爱情是生活最好的提神剂。 Love never dies. 爱情永不死。 The darkness is no darkness with thee. 有了你,黑暗不再是黑暗。 We cease loving ourselves if no one loves us. 如果没有人爱我们,我们也就不会再爱自己了。 There is no remedy for love but to love more. 治疗爱的创伤唯有加倍地去爱。 When love is not madness, it is not love. 如果爱不疯狂就不是爱了。 A heart that loves is always young. 有爱的心永远年轻。 Love is blind. 爱情是盲目的。 1.A bad workman always blames his tools. 拙匠总怪工具差。 2.A contented mind is a perpetual feast. 知足长乐。 3.A good beginning is half the battle. 好的开端等于成功一半。 4.A little pot is soon hot. 壶小易热,量小易怒。 5.All lay loads on a willing horse. 好马重负。 6.A merry heart goes all the way. 心情愉快,万事顺利 。 7.Bad excuses are worse than none. 狡辩比不辩护还糟 。 8.Character is the first and last word in the success circle. 人的品格是事业成功的先决条件。 9.Cleanliness is next to godliness. 整洁近于美德 。 10.Courtesy costs nothing. 彬彬有礼,惠而不费 。 11.Doing nothing is doing ill. 无所事事,必干坏事。 12.Early to bed, early to rise, make a man healthy, wealthy, and wise. 睡得早,起得早,聪明、富裕、身体好 。 13.Empty vessels make the most noise. 满瓶子不响,半瓶子晃荡 。 14.Every man hath his weak side. 人皆有弱点 。 15.Everything ought to be beautiful in a human being: face, dress, soul and idea. 人的一切都应当是美丽的:容貌、衣着、心灵和思想。 16.Extremes are dangerous. 凡事走向极端是危险的 。 17.Good advice is harsh to the ear. 忠言逆耳 。 18.Grasp all, lose all. 欲尽得,必尽失 。 19.Great hopes make great men. 伟大的理想造就伟大的人物。 20.Handsome is he who does handsomely. 行为美者才真美。 21.Have but few friends, though many acquaintances. 结交可广,知己宜少。 22.Hear all parties.兼听则明,偏听则暗 。 23.He is a wise man who speaks little. 智者寡言。 24.He is not laughed at that laughs at himself first. 有自知之明者被人尊敬。 25.He is rich enough that wants nothing. 无欲者最富有,贪欲者最贫穷。 26.He is truly happy who makes others happy. 使他人幸福的人,是真正的幸福。 27.Honesty is the best policy. 诚实乃上策。 28.Hope for the best and prepare for the worst.?? 抱最好的希望,作最坏的准备 。 29.Idleness is the root of all evil. 懒惰是万恶之源 。 30.If we dream, everything is possible. 敢于梦想,一切都将成为可能。 31.Kind hearts are the gardens, kind thoughts are the roots, kind words are flowers and kind deeds are the fruits. 仁慈的心田是花园,崇高的思想是根茎,友善的言语是花朵,良好的行为是果实。 32.Laugh, and the world laughs with you; Weep, and you weep lone. 欢笑,整个世界伴你欢笑。哭泣,只有你独自向隅而泣 。 33.Life is measured by thought and action not by time. 衡量生命的尺度是思想和行为,而不是时间。 34.Life is not all beer and skittles. 人生并非尽是乐事 。 35.Long absent, soon forgotten. 别久情疏 。 36.Look before you leap. 三思而后行 。 37.Lookers-on see most of the game. 旁观者清,当局者迷。 38.Manners make the man.观其待人而知其人 。 39.Misfortune tests the sincerity of friends. 患难识知交。 40.No cross, no crown.没有苦难,就没有快乐 。 41.Nobody's enemy but his own. 自寻苦恼 。 42.One man's fault is another man's lesson. 前车之覆,后车之鉴 。 43.Pardon all men, but never thyself. 严以律已,宽以待人。 44.Reason is the guide and light of life. 理智是人生的灯塔 。 45.Sadness and gladness succeed one another. 乐极生悲,苦尽甘来 。 46.Still waters run deep.流静水深,人静心深 。 47.The fire is the test of gold; adversity of strong men. 烈火炼真金,逆境炼壮士 。 48.The fox may grow grey, but never good. 江山易改,本性难移 。 49.The more a man learns, the more he sees his ignorance. 知识越广博,越感已无知 。 50.Virtue is a jewel of great price. 美德是无价之宝 。 51.Weak things united become strong. 一根筷子易折断,十根筷子硬如铁 。 52.We can't judge a person by what he says but by what he does. 判断一个人,不听言语看行动 。 53.Where there is a will there is a way. 有志者,事竟成 。 54.Will is power. 意志就是力量 。 55.Wise men are silent; fools talk. 智者沉默寡言,愚者滔滔不绝 。 56.Wise men learn by others' harm, fools by their own. 智者以他人挫折为鉴,愚者必自身碰壁方知觉。 1 夏天的飞鸟,飞到我的窗前唱歌,又飞去了。 秋天的黄叶,它们没有什么可唱,只叹息一声,飞落在那里。 stray birds of summer come to my window to sing and fly away. and yellow leaves of autumn, which have no songs, flutter and fall there with a sign. 2 世界上的一队小小的漂泊者呀,请留下你们的足印在我的文字里。 o troupe of little vagrants of the world, leave your footprints in my words. 3 世界对着它的爱人,把它浩翰的面具揭下了。 它变小了,小如一首歌,小如一回永恒的接吻。 the world puts off its mask of vastness to its lover. it becomes small as one song, as one kiss of the eternal. 4 是大地的泪点,使她的微笑保持着青春不谢。 it is the tears of the earth that keep here smiles in bloom. 5 无垠的沙漠热烈追求一叶绿草的爱,她摇摇头笑着飞开了。 the mighty desert is burning for the love of a bladeof grass who shakes her head and laughs and flies away. 6 如果你因失去了太阳而流泪,那么你也将失去群星了。 if you shed tears when you miss the sun, you also miss the stars. 7 跳舞着的流水呀,在你途中的泥沙,要求你的歌声,你的流动呢。你肯挟 瘸足的泥沙而俱下么? the sands in your way beg for your song and your movement, dancing water. will you carry the burden of their lameness? 8 她的热切的脸,如夜雨似的,搅扰着我的梦魂。 her wishful face haunts my dreams like the rain at night. 9 有一次,我们梦见大家都是不相识的。 我们醒了,却知道我们原是相亲相爱的。 once we dreamt that we were strangers. we wake up to find that we were dear to each other. 10 忧思在我的心里平静下去,正如暮色降临在寂静的山林中。 sorrow is hushed into peace in my heart like the evening among the silent trees A star has 5 ends; A square has 4 ends; A triangle has 3 ends; A line has 2 ends; A life has one end. But I hope your happiness has no end. 1. 记住该记住的,忘记该忘记的。改变能改变的,接受不能改变的。 Remember what should be remembered, and forget what should be forgotten. Alter what is changeable, and accept what is mutable. 2.鱼对水说你看不到我的眼泪,因为我在水里。水说我能感觉到你的眼泪,因为你在我心里。 “You couldn’t see my tears cause I am in the water.” Fish said to water. “But I could feel your tears cause you are in my heart.” Answered water. 3.人生短短几十年,不要给自己留下了什么遗憾,想笑就笑,想哭就哭,该爱的时候就去爱,无谓压抑自己。 Your life only lasts for a few decades, so be sure that you don't leave any regrets. Laugh or cry as you like, and it‘s meaningless to oppress yourself. 4. 生命中,不断地有人进入或离开。于是,看见的,看不见了;记住的,遗忘了。生命中,不断地有得到和失落。于是,看不见的,看见了;遗忘的,记住了。然而,看不见的,是不是就等于不存在?记住的,是不是永远不会消失? There is someone that is coming or passing away in your life around the clock, so you may lose sight of those seen, and forget those remembered. There is gain and loss in your life, so you may catch sight of those unseen, and remember those forgotten. Nevertheless, doesn’t the unseen exist for sure? Will the remembered remain forever? 5. 后悔是一种耗费精神的情绪。后悔是比损失更大的损失,比错误更大的错误。所以不要后悔。 Penitence is something that enervates our spirit, causing a greater loss than the loss itself and making a bigger mistake than the mistake itself. So never regret.
怎样评价uni-app?
前端最火热的话题无法就是flutter,不管是刷哪个论坛,必定有探讨flutter的文章。没用过flutter,但是对于跨平台的技术,我一直都在研究。
为什么是uni-app
之前一直在找解决跨平台的方案,尝试了很多方案,比如滴滴的变色龙,但是最终还是选择了uni-app,这里附上uni-app的官网。为什么会选择它呢,第一,vue语法,学习成本低,上手速度快,只要之前你做过vue的项目,那么就能很快上手,其实是vue和微信小程序的结合体,一半vue,一半微信小程序。第二,长期维护,之前做微信小程序的时候,选择了美团的mpvue,但是后面发现长期不维护了,提了Issues也没人理,随之就放弃了,而uni-app长期在维护,这样看出了开发团队的用心。第三,跨平台的能力,uni-app能够跨多个终端,H5,安卓,Ios,微信小程序,百度小程序,头条小程序,支付宝小程序,真正实现了一套代码,多端运行,而且很好适应了我国的市场。第四,日益丰富的插件市场,uni的插件市场也在日益强大,能够基本上满足我们平时的开发需求。
uni-app的组件有原生调用能力,第三方的vue库在调原生接口时跟 5+runtime 不兼容。就像nativescript 有vue版和angular版,类似于react native , 都是起源于phonegap/cordova
实际开发效果遵义小红椒 做了一款app,打包了安卓,Ios,微信小程序3个平台,产出的效果都还是不错,总体还是比较满意。而且打包过程也很方便简单,配套的HBuilderX自动内置了打包功能,所以也省去了打包的烦恼。
遵义小红椒 建议
如果你现在想做一款跨平台的产品,而且有vue和微信小程序的经验,最重要的,你不想学习一门新语言,那么uni-app也许是你的一个选择。
Flutter视频播放器,简洁!
注:亮度调节和音量调节gif无法体现,功能是ok的,其次默认Icon锁的close和open实在难以分辨。
环境:Flutter 2.8.1 channel stable ;Dart 2.15.1
需要音频播放器的看这里: Flutter音乐播放器
重点说下这个工具类,因为视频播放,涉及到状态改变有很多,笔者刚开始选择使用 InheritedWidget 来在众多的widget之间共享数据。但是总感觉这样有点繁琐,且不很优雅!
这里非广告,如果是使用 GetX 就很简单了,笔者也使用了 GetX 进行封装了,一泻千里的赶脚!,但是笔者还是那句话:刚开始接触Flutter的开发者不是很建议使用 GetX ,可以先熟悉下Flutter状态管理的基础原理再行使用。而且为了尽量简洁,还是不引入其他的第三方了。
我们选择对第三方插件进行封装的目的不外乎这几个:
于是笔者就写了一个工具类 VideoPlayerUtils ,专门且只用来处理播放器的所有业务。包括暂停、播放、跳转、调节音量、调节亮度、切换视频等操作。在所有的widget中不会引用关于 video_player 或其他第三方插件的任何信息, VideoPlayerUtils 负责widget与播放器之间的所有操作交互。后续优化迭代或更换播放器插件时,只需针对这个工具类进行修改,对所有widget不会有任何的影响,大大的解耦合了。
其中 VideoPlayerState :
提供以上的公共属性,可以通过 VideoPlayerUtils 来获取对应的值,使用 get 只读,使外界不会误修改这些属性,以保证数值的安全性。开发者可根据自身需要自行添加属性。
提供以上方法来处理播放器的所有业务。同样的开发者可根据自身需要自行添加或修改。
重点说下这个方法,是整个业务的核心方法,控制视频的播放或暂停。开发者只要遇到播放或暂停是均可调用此方法,具体是播放或暂停,内部根据传入的 url 自行判断,开发者不需要关心。
切换新视频也是使用此方法,传入的 url 与上次不一致,自动切换新视频。笔者可根据 statusListener 来监听播放状态的改变,以此处理自身逻辑。
这个也需要提下,视频播放器在播放新视频时会异步初始化,一般我们的操作是在 initState() 初始化,成功后再 setState() 。这里笔者遇到一个让人蛋疼的问题:
我们看 video_player 的使用:
VideoPlayer(controller) :widget中已经持有了controller。本来笔者封装的目的就是为了让widget与controller的之间解耦合。但此时的笔者。。。。
放弃不是不可能放弃的,这辈子都不会放弃的!
于是笔者取了巧,写了一个初始化监听器 initializedListener ,包换2个参数: bool,Widget ,初始化是否成功;其中widget为初始化成功返回需要展示的播放器UI,失败默认返回 const SizedBox() 。
到这里就可以简单使用了:
没看错,视频播放就是这么简单。
如果有更多的业务功能,笔者也按照自己的需求写了一套,同样的开发者可根据自身需要自行添加或修改。
VideoPlayerGestures 主要是处理手势的,比如快进、快退等跳转播放;左侧上下滑动调节亮度;右侧上下滑动调节音量;单击是否开启沉浸式播放,所有widget的隐藏与显示;双击播放、暂停等。
哦,还有 PercentageWidget 也放到这个文件下了,就是这玩意:
因为显示的百分比与手势相关,随着手势移动而更新。开发者可自行处理。
笔者处出于简单考虑,就按照整个UI的位置命名了。瞅一眼就知道是啥玩意。
同样的开发者可根据自身需要自行添加或修改。
就是这玩意:
同样的开发者可根据自身需要自行添加或修改。话说这个锁的 Icon 的open和close是真的难分辨!
就是这玩意:
同样的开发者可根据自身需要自行添加或修改。
这玩意是自定义的,别问,问就是跟产品干一架落了下风
主要就是自定义这玩意:
同样的开发者可根据自身需要自定义。
注:这里没有添加缓冲的进度,开发可查看 video_player 中的源码 VideoProgressIndicator ,按业务自行定义。
这玩意就是整合以上的widget,再考虑下全屏的安全区域,没啥东西。开发者可自行处理!
具体的实现监听器的思路, 看这里 。
自此一个漂亮的Flutter视频播放器就已经结束了。如果您觉得对您有些许帮助的话,欢迎 Star !
Flutter:快速创建简单闪屏页
近来闲暇时间一直在做Flutter,闪屏页是一个比较常见的需求,网上的闪屏页教程大部分是那种类似于广告页,而非iOS中的 LaunchScreen 性质的闪屏页.按照原来的方案我们要配置闪屏页的话,我们需要同时配置两端的闪屏页,那么有没有比较简单的方案来配置闪屏页呢? 毋庸置疑,当然是有了,那就是Flutter的插件 - flutter_native_splash . 接下来我们就来看一下具体应该怎么使用这个插件.
首先把 flutter_native_splash 导入到工程的 pubspec.yaml 中.这里需要注意的是需要放在 dev_dependencies 下,而不是 dependencies .具体如下所示.
接下来我们就来配置 flutter_native_splash ,在配置之前我们看一下 flutter_native_splash 的可配置项.
例如,我现在只有一个logo图片,那么我想生成iOS和android两端的闪屏页,这时候我只需在 pubspec.yaml 如下设置即可.
当然了,如果你有其他配置可以自行进行添加.
配置完成了,我们该如何生成呢?这时候需要我们打开终端 cd 到我们的工程目录下.如果是Android Studio 或者 VSCode 默认就是在当前工程目录下.
然后我们需要执行下面的三个命令来生成闪屏页
每一次都敲三个命令实属麻烦,我们把上诉的三个命令整合成一个命令,如下所示.
那么,我们不想使用该插件生成的闪屏页该怎么办呢?我们只需要执行下面命令即可.
注:每一次更换图片都是需要重新执行命令重新生成.
OK,上面就是关于 flutter_native_splash 的使用全部内容,其实比较简单,如果需要定制化的,建议还是各自平台配置各自的闪屏页.如果有任何问题欢迎在评论区批评指导,感谢大家了.
一句香水的广告词“弥散在空中的唯美”,怎样译成英文且意境类似?
字译:Aesthetic feeling dispersing in the air
衍生:Let the fragrance flutter in your breath
再来:Scenting all around
想到一句成语千古流香(芳香):Fragrance that last thousands of generations
本文题目:flutter开平广告,flutter 开屏广告
网站URL:http://scpingwu.com/article/dsgdehg.html