终端开发

Android开发之FileProvider相关知识 Android 权限 及设置描述信息 局域网开启 Android ADB 无线调试 安卓开发那些事之开发规范 安卓组件化之组件通信及拦截 安卓开发那些事之版本管理 安卓组件化之持续集成与自动构建 安卓组件化之组件调试和运行 安卓组件化之组件管理(导入、依赖、升级) 安卓组件化之框架设计 安卓开发那些事总纲 Android Textview 对HTML 的支持 appium 爬坑之基于 Chromedriver 测试 Android webview Linux 搭建 Android 编译构建环境 appium爬坑之iMac上基于appium多设备并发测试 appium爬坑之iMac上appium环境搭建及使用真机测试Android项目简介 H5页面通过指定Schema拉起Android应用 Android JSBridge简介 Gradle插件开发系列之发布gradle插件到开源库 Gradle插件开发系列之开发第一个gradle插件 Gradle插件开发系列之gradle插件调试方法 Gradle插件开发系列之总纲 一款检查Gradle依赖配置是否冲突的插件 记一次升级Android Gradle Plugin后databing编译报错填坑经历 iMac上 Xcode 相关设置及常见问题 iOS开发 -- 首次使用Xcode运行iOS项目代码 使用Android Studio开发可独立运行(runnable)混淆过的Jar程序 Android安装包精简系列之资源精简 Android安装包精简系列之图片优化 Android安装包精简系列之为什么要优化精简安装包 Android安装包精简系列(总纲) Android安装包精简系列之图标转字体 Gradle 相关资料汇总 Android编译常见错误解决 Android编译编译速度提升 终端基于gradle的开源项目运行环境配置指引 制作终端产品演示的gif 一个关于APK Signature Scheme v2签名的神奇bug定位经历 如何随apk一起打包并使用SQLite SDK热更之gradle插件(如何在SDK代码中自动插桩及如何生成补丁包) 关于Android的APK Signature Scheme v2签名相关的资料汇总 封装HttpURLConnection实现的简单的网络请求库 一款基于Java环境的读取应用包名、签名、是否V2签名等基本信息的工具 Android的APK Signature Scheme v2签名及一款基于Java环境的校验工具介绍 如何使用Eclipse开发可执行Jar程序,并生成混淆过的jar程序 Android 相关的学习资料整理(持续更新) macOS(Sierra 10.12)上Android源码(AOSP)的下载、编译与导入到Android Studio Android开发常用命令备忘 Google也看不下去被玩坏的悬浮窗了么? Android开发常用工具资源 SDK热更系列之概述(持续整理编辑中~) SDK热更系列之SDKHotfix待优化点 Android 终端开发相关的一些神图(持续更新) SDK热更系列之Demo项目介绍概述 SDK热更系列之Demo体验方法 SDK热更系列之如何获取应用在当前设备上的so对应的指令集 Gradle Android插件使用的中那些特别注意的点 Experimental Plugin User Guide(From Android Tools Project Site) 基于Android Studio使用gradle构建包含jni以及so的构建实例 基于Instrumentation框架的自动化测试 - Android自动化测试系列(四) Instrumentation框架介绍-Android自动化测试系列(三) 关于终端设备的设备唯一性的那些事之MAC地址 关于终端设备的设备唯一性的那些事之IMEI Android 检查应用是否有root权限 ant常见错误解决方案 Gradle介绍 iMac上Android Studio 相关设置及常见问题 ADB命令系列之再说ADB 再看Android官方文档之分享 再看Android官方文档之Fragment&数据保存 再看Android官方文档之Activity&Intent 再看Android官方文档之ActionBar和兼容性 ADB命令系列之 adb shell input(Android模拟输入)简单总结 再看Android官方文档之建立第一个APP Android开发调试常用工具 ANR(网络资料整理) Java参数引用传递引发的惨案(又一次Java的String的“非对象”特性的踩坑经历) android.view.WindowManager$BadTokenException,Unable to add window Android签名校验机制(数字证书)及命令行获取 keystore 公钥等信息 Robotium二三事-Android自动化测试系列(二) Robotium介绍-Android自动化测试系列(一) Android开发中遇到的那些坑 Eclipse使用中部分经验总结 Android中关于Nativa编译(NDK、JNI)的一些问题 Android简单实现的多线程下载模块 Android内存耗用之VSS/RSS/PSS/USS ADB命令系列之 Advanced Command URL编码中的空格(编码以后变为+) Android MD5后 bye数组转化为Hex字符串的坑(记一次为女神排忧解难的经历) Android学习之路 ADB命令系列之 Base Command Android Log的那些坑…………

开发工具

Linux JDK 安装、卸载、路径查看 替换Jcenter,发布开源代码到 Sonatype Maven Central 使用 gitstats 分析代码仓库 使用 Jekyll 搭建技术博客并部署到 CentOS 服务器 开发中常用的文档管理、云端笔记等效率工具介绍 iMac(OS X)常用图片处理工具介绍 使用 Octopress 搭建技术博客并使用腾讯云静态网站托管 使用腾讯云静态网站托管个人博客 使用 Jekyll 搭建技术博客 iMac上安装Python2.7 和 Python3 iMac 使用自带的 Apache 启用本地web服务 发布开源代码到jcenter Gradle环境变量那些事 iMac搭建jekyll本地环境 iMac(OS X)开发和使用中经验汇总(持续更新) iMac配置基于crontab的定时任务 iMac配置支持读取NTFS硬盘 新iMac机器装机及开发环境搭建攻略 git常用命令整理(已包括branch、tag等持续更新~) iMac使用过程中的简单故障解决 iMac上RubyGems相关的问题汇总 开发中常用的一些Chrome插件介绍 iMac(OS X)日常开发中各种代理设置方法汇总(shell、Android Studio、gem、npm) Markdown格式优化及使用技巧 iMac下制作含透明度图片及判断图片透明度 iMac(OS X)中设置大小写敏感的分区并切换 Linux & MacOS中一些常用命令备忘 iMac(OS X)搭建私有maven仓库,提供Nexus Responsitory镜像 iMac(OS X)El Capitan 更新遇到的那些坑 vi常用命令 iMac(OS X)常用开发工具介绍 iMac(OS X)不可或缺的套件管理器 —— Homebrew 开发环境通用设置 windows中一些常用命令备忘 问题定位之快速模拟请求 Ant中的SVN 使用 Markdown语法简介 SVN 常用命令

标签

android 50

ADB快捷输入法 - AAF 开发中图片图标处理常用工具资源 Linux 搭建 Android 编译构建环境 H5页面通过指定Schema拉起Android应用 Android编译常见错误解决 一个关于APK Signature Scheme v2签名的神奇bug定位经历 关于Android的APK Signature Scheme v2签名相关的资料汇总 封装HttpURLConnection实现的简单的网络请求库 一款基于Java环境的读取应用包名、签名、是否V2签名等基本信息的工具 Android的APK Signature Scheme v2签名及一款基于Java环境的校验工具介绍 如何使用Eclipse开发可执行Jar程序,并生成混淆过的jar程序 Android 相关的学习资料整理(持续更新) macOS(Sierra 10.12)上Android源码(AOSP)的下载、编译与导入到Android Studio Android开发常用命令备忘 Google也看不下去被玩坏的悬浮窗了么? Android开发常用工具资源 Android 终端开发相关的一些神图(持续更新) Gradle Android插件使用的中那些特别注意的点 Experimental Plugin User Guide(From Android Tools Project Site) iMac(OS X)搭建私有maven仓库,提供Nexus Responsitory镜像 基于Android Studio使用gradle构建包含jni以及so的构建实例 基于Instrumentation框架的自动化测试 - Android自动化测试系列(四) Instrumentation框架介绍-Android自动化测试系列(三) 关于终端设备的设备唯一性的那些事之MAC地址 关于终端设备的设备唯一性的那些事之IMEI Android 检查应用是否有root权限 iMac(OS X)El Capitan 更新遇到的那些坑 ant常见错误解决方案 Gradle介绍 iMac上Android Studio 相关设置及常见问题 ADB命令系列之再说ADB 再看Android官方文档之分享 再看Android官方文档之Fragment&数据保存 再看Android官方文档之Activity&Intent 再看Android官方文档之ActionBar和兼容性 ADB命令系列之 adb shell input(Android模拟输入)简单总结 再看Android官方文档之建立第一个APP Android开发调试常用工具 ANR(网络资料整理) Java参数引用传递引发的惨案(又一次Java的String的“非对象”特性的踩坑经历) android.view.WindowManager$BadTokenException,Unable to add window Android签名校验机制(数字证书)及命令行获取 keystore 公钥等信息 Eclipse使用中部分经验总结 Android内存耗用之VSS/RSS/PSS/USS ADB命令系列之 Advanced Command URL编码中的空格(编码以后变为+) Android MD5后 bye数组转化为Hex字符串的坑(记一次为女神排忧解难的经历) Android学习之路 ADB命令系列之 Base Command Android Log的那些坑…………

tags 53

替换Jcenter,发布开源代码到 Sonatype Maven Central Android开发之FileProvider相关知识 子勰开源项目展示 照片整理系之视频归档整理方案 使用 gitstats 分析代码仓库 Android 权限 及设置描述信息 局域网开启 Android ADB 无线调试 安卓开发那些事之开发规范 安卓组件化之组件通信及拦截 安卓开发那些事之版本管理 安卓组件化之持续集成与自动构建 安卓组件化之组件调试和运行 安卓组件化之组件管理(导入、依赖、升级) 安卓组件化之框架设计 安卓开发那些事总纲 Android Textview 对HTML 的支持 使用 Jekyll 搭建技术博客并部署到 CentOS 服务器 2018年春节仙本那海岛游行程规划 照片整理系列之单次整理流程 照片整理系列之整理及归档的总体方案 使用 Octopress 搭建技术博客并使用腾讯云静态网站托管 使用腾讯云静态网站托管个人博客 使用 Jekyll 搭建技术博客 颜色透明度对照表 Android JSBridge简介 Gradle插件开发系列之发布gradle插件到开源库 Gradle插件开发系列之开发第一个gradle插件 Gradle插件开发系列之gradle插件调试方法 发布开源代码到jcenter Gradle环境变量那些事 Gradle插件开发系列之总纲 一款检查Gradle依赖配置是否冲突的插件 记一次升级Android Gradle Plugin后databing编译报错填坑经历 服务器 迁移到 腾讯云 记录 2018年春节沙巴仙本那海岛游 聚会桌游助手 nodejs入门之后台服务的几种启动方式 nginx下多域名配置示例 nodejs入门之连接mysql mysql迁移之新建用户、备份还原数据库 iMac配置基于crontab的定时任务 iOS开发 -- 首次使用Xcode运行iOS项目代码 新iMac机器装机及开发环境搭建攻略 旅行中不可或缺的软件 iMac上RubyGems相关的问题汇总 接口设计六大原则 Canvas上更高效的画SVG 微信小程序开发之SVG的使用 旅行出行前准备列表 使用Android Studio开发可独立运行(runnable)混淆过的Jar程序 2017年端午川西滇北香格里拉环线(近200张图慎点) 照片整理系列之基于命令行的照片整理及查看工具 Gradle 相关资料汇总

SDK开发经验之版本

「 经验总结 」 ——  2015年03月31日

本来想在周末写完,结果周末打扰,到现在才开始。要加班了,不然写不完。……

任何产品都有迭代,有周期,有版本,版本是一个阶段工作的里程碑和产出。 ​ 对于应用开发者,更新版本可能是件很开心的事,因为说明在不断的优化和改进。然而对于SDK开发者来说,对于版本更新的感觉却是复杂的,开心又心塞。

开心是因为又修复了不少问题,或者出了新的功能,做了新的优化。但是更多的是多版本维护带来的压力和成本。SDK一个版本的生命周期并不像应用那么短,而且他带来的问题的影响也会更久远。

本来想和关于版本号的那篇一起写完,发现有点多,先发了,写的过程思路有偏差,把他们两篇的耦合关系降低了,所以可以独立开来了。

需要有几个版本

这个其实不用考虑,一个还算合格的SDK肯定都有两个版本:一个稳定版用于大面积的推广;一个开发版用于一些急于使用新功能的游戏试用。

稳定版本和开发版本的存在是为了提高SDK的版本质量,同时结合版本发布的一些策略,降低SDK版本质量对使用者的影响以及SDK的bug的影响范围。

这里对开发版再增加一点说明。这里的开发版不是说开发临时出的版本。任何只要对外提供的版本,都必须要经过标准的版本发布流程,无论是开发版还是稳定版都是对外正式发布的版本。而开发通过本地环境打的包或者自己临时出的包是不可以包含在版本计划中,更是绝对不能提供给游戏。任何流程,只要是人完成的就总会有问题,而且这里也被坑过,因此才有这个也算严格的规范。这样即可以杜绝游戏使用有问题的包上线的风险,而且不会让别人觉得很不专业(PS:接过的某SDK,经常随意给版本,甚至给过100%crash的版本………………)。

版本的周期(或者版本发布的策略)

对于我们目前的SDK的发布的策略或者版本的周期,我觉得还是不错的。这里也做一个简单的说明吧。

推荐做法:

SDK从开发完成到最后对外提供,建议包括以下过程:开发自测、专项测试、发布开发版(或者叫内测、小范围灰度)、转为稳定版、下线版本这样几个周期。

测试我会在下面继续说。这里说下其余的几个环节。发布开发版,也就是灰度我觉得是至关重要的一环。我们的使用者除了外面的开发商还会有自研的游戏。因此当我们的新版本发布以后,我们会优先提供内网的下载地址供内部游戏接入,同时我们从后台观察具体的数据;当有游戏通过新版本上线而且没有问题,我们就会正式推出这个版本。

对于转为稳定版,一般是在我们某个有重大功能更新的版本在很多游戏已经验证过没有问题以后。一些重大功能版本之间的小版本,一般即使他很稳定,我们也不会用它替换我们的稳定版,因此会发现我们的开发版是不停的变,但是稳定版每次变化都会很大,版本也是跳跃的。

最后一个环节不是非必须的,当我们推出新版的时候,我们会下线旧的版本。我们是为了限制推动游戏升级和更新版本,所有我们不会提供游戏老版本的下载链接,这个因实际需求不同而不同。

关于新功能的增加

就一个原则:所有新功能的开发只能排期到最新版去开发,绝不能为了尽快的推某个版本而简单的在稳定版拉个分支去支持新功能。(当然老大非让你上也没办法,说明厉害关系)

就说说我们的故事吧,我们在稳定版1.7,开发版1.9的时候,某平台想推出新功能。然后有个游戏想急用,同时平台也很着急。迫于多方压力,我们在1.7增加了一个新功能版本加入这个功能。由于1.7以上的版本太多了,我们不可能在所有版本都假如新功能,所以只在目前外发的稳定版和开发版的最新迭代以及正在开发的主干版本同时加入了这个坑爹的功能。然后噩梦就开始了,他们的版本和接口中间都有变动,为了向上兼容每一次我们就要把对外发布的1.7和1.9以及我们正在开发中的2.0对应的地方同步改动,同时出三个版本……这个问题的后遗症一直持续了半年,有段时间每天疲于应付各种版本……

其实因为类似的问题,我们遇到过一个更坑更坑的问题,这次低版本调整涉及到了本地数据库的变更,我会接下来讲故事说明,这一次在某个老版本插入新功能直接影响了我们超过5个大的版本。那段时间游戏问的最多的是,我们是x.x.x,我们现在要升级,应该升级到哪个版本呢?让你苦死,苦逼到极致的经历……

因此一定要和产品、测试、老大约定好,不要随意在低版本增加新功能,尤其是变动交大的功能,这个对接下来的生活来说是致命的。

关于DB版本

​对于android的向下兼容,专门聊聊本地数据库的版本的问题吧。这个问题让我们心碎了不久。问题的起因是没有规划的在低版本中增加新功能,结果引来一系列悲剧的问题就。这里我随便用三个版本来描述吧。我们当时稳定版1.1.1的DB版本号是14,开发版1.3.3的是17(中间还有别的版本是15,16);然后1.1.1版本临时要增加新功能,而且这次功能调整了DB,这个时候就要升级DB的版本号了,那改到多少呢?

  • 如果高于17,万一游戏从稳定版升级到中间版本(DB是15、16),这个时候会遇到DB降级的问题
  • 如果低于17,那问题更严重,一样的版本号,不一样的DB就彻底噶屁了

最终选择了升级到17以上,然后重载了SQLiteOpenHelper的onDowngrade函数,当DB降级的时候,就直接删除了DB,重建一个。这样变为:1.1.2是18,新功能也没加入1.3.3,所以他的版本保持17,我们开发中的1.4.0升级到19。

你以为这样就完了,太天真。这次以后有很长时间我们没有更新过DB,所以就忘了这件事。然后有一天更新的时候,开发版本想变为稳定版,就把这个功能也加上,此时

  • 我们的主干DB已经到了20多
  • 然后把1.3.3更新到1.3.4,改了DB,然后就升级了个版本号18……

然后当他变成稳定版以后,原来1.1.2的大批游戏升级到1.3.4以后就发现又跪了,而且是必现的crash。当时都要疯了,因为忘了1.1.2的那回事,用了一样的DB版本号,虽然新增的表改了,但是1.3和1.1本来就有差异的表没变…………

最终只能是彻底把所有版本的DB版本号都梳理了一遍,有问题的版本全部升级并把一批版本列为不可用于正式上线的版本以后才解决。导致的结果就是我们中间的5个版本是不能用于上线的,游戏等了很久才换了一个可用的版本……

这里废话太多了,而且估计还看不懂。总结下:

  • 在SQLiteOpenHelper中,一定要覆写其onUpgrade和onDowngrade函数。做到对应用DB升级和降级的处理。
  • 慎重修改表字段,修改以后一定要记得整理DB版本号。
  • 不要把高版本的内容往低版本加!!!

关于版本发布

这里主要说从版本排期到最后发布过程中个人认为比较重要,我们做的也还不错(那是因为坑了太多次)的要点吧。直接用标题列举吧。

要有一个全面规范的提测流程

对于一个版本来说,每次提交测试前需要确认的问题肯定很多,例如功能验证、开发过程中一些临时增加的一些断点、为了配合开发做的一些调整等等。每次提测的时候这些问题很多总会被遗漏,或者某一类问题的某一个点总会被遗漏。因此,我们对于一个版本提测,要有一个专门的提测流程。对于一些重点项还该有具体的说明(结合自身实际制定一个提测前的版本checkList)这部分内容我会再SDK开发经验之测试(点击阅读)中来详细说明我们的流程是什么样,踩过什么坑等。

最终提供的版本包一定要自动生成

只要人参与的总会有问题,编译环境,配置文件等等的差异有时候很容易被忽视。另外,只要能用工具、能自动的就绝不手动。

拿我们的SDK来说,之前因为是手动出包,有时候会经常遇到遗漏文件、忘记修改配置中的版本号、SVN填写错等问题。带来的结果就是:

  • 版本质量很烂,被各种鄙视
  • 一旦遇到问题,回烁到问题版本会发现难度很大,很吃力,有时候甚至找不到对应版本准备的代码版本

对于android应用来说,通过ant和批处理(win)或者shell脚本,我们可以完成代码编辑、jar包生成、自动化测试等所有的工作,大大降低人为工作引起出错的成本。

关于向下兼容

本来,客户端的向下兼容是最轻松的,因为客户端开发的时候其实是不用考虑向下兼容的,因为他是一个独立的个体。如果非要考虑,可能也只是考虑安装、签名等这些,开发中本来是最轻松的。但是!!!当你是一个SDK的客户端开发的时候,一切就都变了。

对于SDK的客户端开发,虽然你可以任性的在不同版本随意的优化和调整而不会招致什么问题。但是这对于接入你SDK的开发来说,却绝对是噩梦,你绝对是他们认为神一样的存在。尤其是你老的版本的覆盖范围已经非常大的时候。

我们的SDK目前被吐槽的最多的很多方面其实我们也知道。而且我们比来吐槽的人更想改掉。但是……目前接入我们MSDK的游戏的量太大。我们的一次改动,可能意味着增加了无数游戏的版本更新的难度,这势必会影响游戏的开发周期和版本进度。所以很多问题尽管我们知道他们的存在,而且也一直想改,但是一直没有动。我们会在问题暴漏的足够多的时候一次改变了,而不是一点点去调整,然游戏一直阵痛,每次更新都来骂一次。

SDK也是产品,只是用户比较特殊。如果有些时候不能拿定注意,就和你的用户做一次比较大范围的访谈,另外做一些测试。目前我们遇到的回调太多、模块太多、常量和错误码定义太混乱等问题其实不但困扰游戏也困扰我们自己,在2.0的时候我们本想修复这些问题,但最终被否定了。后来在和各种类型游戏开发进行了多次交流沟通以后,我们才下定决心会在3.0解决。同时我们用另一个使用范围比较小、更新比较简单的SDK对我们的想法做了一些验证,最总确定了修改的方案。

SDK存在很重要的一个价值就是方便开发的接入。当SDK接入、更新的成本超过甚至逼近开发直接对接的成本时,这个SDK其实是失败的,而且也失去了应有的意义。


赞赏

取消
微信扫一扫,赞赏子勰
扫码支持
屌丝程序猿,鸡血攻城狮!努力学技术,潜心做精品!