本文出自:Philosophical Hacker,译文出自:开发技术前线,译者:dengshiwei
我的第一个应用非常糟糕。事实上,它糟糕得以致于我从应用市场上删除它,同时我甚至都不会在我的简历上罗列出它。如果我在开发之前能够知道一些Android开发的事情,也不会糟糕到这步田地。
本文中所罗列的事情是你在开发第一个Android应用的时候需要牢记在大脑中的。我接下来将展示的实际错误均来自于我的第一个应用程序代码中。把这些错误经验牢记心头能够帮助你开发一个让你引以为豪的应用。
当然,正如Code Standards所说:如果你所做的工作和你作为学生开发的Android应用类似,你很有可能会讨厌你的应用。
——Code Standards 2015.5.21
如果你是一位经验丰富的Java开发者,第1、2、5条很有可能对你没有吸引力。另一方面,即使你从来没有犯过这些例子中的错误,第3、4条也可能向你展示一些很酷的事物,你可以利用一款也许你不知道的软件——Android Studio去实现这些事物。
1. 不要持有Context的静态引用
- public class MainActivity extends LocationManagingActivity implements ActionBar.OnNavigationListener,
- GooglePlayServicesClient.ConnectionCallbacks,
- GooglePlayServicesClient.OnConnectionFailedListener {
- //...
- private static MeTrackerStore mMeTrackerStore;
- //...
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- //...
- mMeTrackerStore = new MeTrackerStore(this);
- }
- }
这对于每个人来说看似是一个不可能犯的错误。但事实却并非如此,我犯了这个错误,我也看到过别人犯这个错误,同时我也采访过那些不能很快指出为什么这是放在第一位的错误的人。不要这样做,它是会变的。
如果MeTrackerStore通过它的构造函数保持一个指向Activity的引用,这个Activity将不会被垃圾回收(GC),除非静态变量被从新分配到不同的Activity。这是因为mMeTrackerStore是静态变量,而静态变量的内存是不会被回收,直到应用程序退出才回收。如果你正在试图做这样的事情,你的代码很有可能有严重的错误。寻找帮助吧,可能看看Google的Udacity课程“Android Development for Beginners”能够帮助你。
2. 注意那些你无法控制生命周期的对象的隐式引用
- public class DefineGeofenceFragment extends Fragment {
- public class GetLatAndLongAndUpdateMapCameraAsyncTask extends AsyncTask<String, Void, LatLng> {
- @Override
- protected LatLng doInBackground(String... params) {
- //...
- try {
- //Here we make the http request for the place search suggestions
- httpResponse = httpClient.execute(httpPost);
- HttpEntity entity = httpResponse.getEntity();
- inputStream = entity.getContent();
- //..
- }
- }
- }
- }
这段代码有很多问题,但我现在只会把重点问题放在“隐式引用”那些问题上。在Java中,(非静态)内部类有个对外部类实例有个隐式引用。
在这个例子中,任何GetLatAndLongAndUpdateCameraAsyncTask都有一个外部类DefineGeofenceFragment的引用。对于匿名类是同样的,它们也有一个对包含它们的类的实例的一个隐式引用。
GetLatAndLongAndUpdateCameraAsyncTask对生命周期我们无法控制的Fragment对象有一个隐式引用。Android SDK负责创建和销毁Fragment,如果GetLatAndLongAndUpdateCameraAsyncTask 因为正在运行而不能被垃圾回收,那么DefineGeofenceFragment也将因为具有隐式引用而保留不能被垃圾回收。
这里有一个很棒的Google视频,解释它为什么会发生这种事情(友情提示:需自备梯子哈)。
3.使用Android Studio进行工作
- public ViewPager getmViewPager() {
- return mViewPager;
- }
这段代码是我使用“Generate Getter”在Android Studio中进行生成的。这些getter保持了’m’前缀的实例变量,同样通过它也能为一个方法产生相同的效果,这已经不是空想。
(如果你想知道为什么’m’是实例变量的名称的第一个字母,’m’往往是实例变量的公认约定。它代表了’member'(成员)的意思)。
不管你是否认为实例变量的前缀’m’是一个好注意,在这有一个知识,Android Studio能够帮助你编写任何你想要实现的公认约定。例如,在你为实例变量生成getters、setters和connstructor参数时,你可以使用Android Studio代码风格对话框的设置使Android Studio在你的实例变量前自动添加’m’和移除’m’。
Android Studio能够做的远不止于此。学习Android Studio从学习快捷键和模版是不错的开始。
4. 一个函数只做一件事
在我写的众多类中,有一个类的一个方法我便写了有100多行。这类的方法是非常难以读懂、修改和重用,努力让一个方法只做一件事情。显然,这意味着你应该对超过20行的方法持有怀疑态度。这里,你可以使用Android Studio来帮助你发现有问题的方法:
5. 向聪明和有经验的人学习
这可能听起来微不足道,但是这是我开发第一个应用时候犯下的错误。
当你开发一个应用的时候,你会犯别人已经犯过的错误。向别人学习,你可以避免犯别人犯过的错误来节约你的时间。我在我的第一个应用中浪费了大量的时间去犯错,这些错误如果我花点时间向有经验的软件开发工程师学习就可以避免。
阅读Pragmatic Programmer,然后阅读Effective Java。这两本书会帮助你避免开发新手常犯的错误。在你学习了这两本书后,不停地寻找聪明的人并向他们学习。
6.使用类库
当你开发应用的时候,你可能会遇到一些聪明人和有经验人已经解决过的问题。而且,许多这些问题的解决方案是可以作为开源库的,充分利用它们。
在我的第一个应用中,我写了一些类库已经提供的功能代码。其中一些是Java标准库,还有一些是第三方类库,如Retrofit和Picasso。如果你不确定你使用什么样的类库,你可以做下面3件事情:
- 听Google IO Fragmented广播。在这期间,询问这些开发者什么第三方类库类库对Android很重要。
- 订阅Android周刊。这里包含了一部分最新的类库,时刻注意哪些对自己有用。
- 寻找那些能够解决与你在开发应用中遇到问题类似的开源应用。你可能发现某个应用使用的第三方类库就是你想要的,或者你会发现一个你所不知道的Java类库。
总结
开发优秀的Android应用是非常困难的,不要用重蹈覆辙来难为自己。如果你发现我写的代码中的错误请在评论中告诉我。(误导性评论比没有评论更糟糕)。如果你认为这篇文章对于新手开发者有用,请分享它,以解决他们的一些令人头疼的难题。
相关推荐
1. NDK(Native Development Kit)的定义:NDK是Android平台的一个开发工具包,它允许开发者使用C或C++语言来编写应用的某些部分。这些部分通常被称为“原生模块”,它们可以被封装进一个.so(共享库)文件,并最终...
它指导读者如何安装Android开发环境,包括必要的工具和组件,并且引导用户通过实际操作来学习如何开发第一个Android程序。此外,书中也涵盖了对Android模拟器的操作,使读者能在不实际操作真实设备的情况下进行应用...
集成微信SDK的第一步是将.jar和.so文件添加到项目中。将.jar文件放入项目的`libs`目录,并在Gradle配置中添加如下代码以确保其被正确引入: ```groovy dependencies { implementation fileTree(dir: 'libs', ...
总之,"android好用的相册选择框架"是一个实用的工具,它简化了Android应用中图片选择的实现过程,提高了用户体验,并解决了常见的开发问题,对于任何需要处理图片选择的Android应用都是一个值得考虑的解决方案。
《深入淺出 Android》是一本专门针对Android应用开发的电子教程,旨在帮助初学者和有一定经验的开发者深入了解Android平台的程序设计。这本书采用创新的CC BY-NC-ND授权方式,鼓励读者分享链接,但禁止未经许可的...
1. **2_filebrowser.rar**:这是一个文件浏览器的源码,它展示了如何实现Android应用中对设备文件系统的浏览和管理功能。通过这个源码,你可以学习到如何使用`java.io`和`java.nio`包进行文件操作,以及如何构建用户...
第一步,定义一个 layout,实现按钮内部的布局。 第二步,写一个类继承 LinearLayout,导入刚刚的布局,并且设置需要的方法,从而使的能在代码中控制这个自定义控件内容的显示。 第三步,在程序中应用自定义的 ...
【极客班第一期Android专业考核细则1】的目的是为了评估学员在Android开发领域的学习成果,确保他们具备必要的技能和知识。考核分为三个主要部分,包括课程测验(作业)、课堂优秀表现和项目实践或期末综合测试题。...
这篇标题为“Android 小项目之--消息、线程、动画显示图片(附源码)”的文章,显然聚焦于Android应用开发中的几个关键概念:消息处理、线程管理和使用动画来显示图片。这是一篇教程性质的文章,作者提供了实际的源...
在我们从事APP开发行业以来,客户通常咨询的第一个问题是“开发一个App要多少钱?”而市面上APP的开发报价则是从几千到上百万价格不等,为什么会出现这么大的差价?具体又有什么差别?其实出现这样的情况是因为采用...
标题中的“android http server”指的是在Android平台上搭建一个HTTP服务器,这是为了让Android设备能够作为服务器端,接收并处理HTTP请求。这种技术常用于测试、本地数据共享或开发自定义网络服务。Android HTTP...
根据提供的文件内容,我们可以提取出以下知识点: ...通过对文档内容的梳理,以上知识点对于了解i.MX6处理器在Android软件开发和应用中的重要信息具有指导性意义,对从事相关嵌入式系统开发的技术人员尤其有帮助。
Android Studio是Google开发的一款集成开发环境,专为Android应用开发设计,广泛用于开发Android应用。随着使用Android Studio进行开发的用户越来越多,其对硬盘空间的需求也随之增加。本文档主要讨论如何通过配置来...
Android程序打包为APK是Android应用程序开发中的一个关键步骤,它将Android程序转换为可以在Android设备上安装和运行的安装包文件(APK)。下面将详细介绍Android程序打包为APK的方法详解。 一、生成未签名的安装包...
【标题解析】:“6_安卓学生管理源码_”这个标题表明这是一个关于Android平台的学生管理系统源代码,可能是第六个版本或者是某个系列的第六部分。它暗示着开发者或分享者已经对该系统进行了多次迭代或者改进。 ...
本教程系列包括了从Qt Creator安装、编写第一个“Hello World”程序开始,到更高级的话题,如多窗口程序开发、菜单和图标添加、布局管理、文本编辑、事件处理、自定义鼠标指针、定时器和随机数生成。教程还覆盖了Qt...
"Midp20"则表示这个版本的Java VM支持MIDP(Mobile Information Device Profile)2.0规范,这是Java ME(Micro Edition)的一个子集,专为移动设备设计,提供了开发移动应用的基本框架和API。 通过这个Java VM,...
:检测应用内存泄漏问题,这个都知道吧 :帮助Android控件和回调的进行依赖注入,JakeWharton大神的力作 :Android和Java依赖注入库 :一个实现异步操作的库,现在非常火 :用于Android的Rxjava绑定库 :配合Rxjava...
目的是建立一个大的资讯平台,为学生提供第一手校园资讯 v1.6.3 ===== src/1.6.3/Shop 该版本是与小小菜团队其他成员项目讨论以后做的调整,目标进一步缩小,从开发平台转到开发订餐 v1.9.2 ===== src/1.9.2/Shop 1....
Vue.js是一个流行的JavaScript框架,它主要用于构建用户界面和单页应用程序。在Vue中实现定位功能,通常需要借助浏览器提供的地理位置API(Geolocation API)以及第三方地图服务商提供的JS API。 ### 实现步骤 1. ...