备战金九银十:BAT 大厂最爱问的 Android 核心面试百题详细解析!
简介:定时向
HTTP服务端接口(Web Service API)获取最新消息。优点:实现简单、可控性强,部署硬件成本低。
缺点:实时性差。
9.Android 的数据存储
1. 使用SharedPreferences存储数据
它是Android提供的用来存储一些简单配置信息的一种机制,采用了XML格式将数据存储到设备中。只能在同一个包内使用,不能在不同的包之间使用。
2. 文件存储数据
文件存储方式是一种较常用的方法,在Android中读取/写入文件的方法,与Java中实现I/O的程序是完全一样的,提供了openFileInput()和openFileOutput()方法来读取设备上的文件。
3. SQLite 数据库存储数据
SQLite是Android所带的一个标准的数据库,它支持SQL语句,它是一个轻量级的嵌入式数据库。
4. 使用 ContentProvider 存储数据
主要用于应用程序之间进行数据交换,从而能够让其他的应用保存或读取此Content Provider的各种数据类型。
5. 网络存储数据
通过网络上提供给我们的存储空间来上传(存储)和下载(获取)我们存储在网络空间中的数据信息。
10.Activity 启动模式
介绍 Android启动模式之前,先介绍两个概念task和taskAffinity
task
翻译过来就是“任务”,是一组相互有关联的activity集合,可以理解为Activity是在 task 里面活动的。task 存在于一个称为back stack 的数据结构中,也就是说,task是以栈的形式去管理 activity 的,所以也叫可以称为任务栈。
taskAffinity:
官方文档解释是:The task that the activity has an affinity for.,可以翻译为 activity相关或者亲和的任务,这个参数标识了一个Activity所需要的任务栈的名字。默认情况下,所有Activity所需的任务栈的名字为应用的包名。 taskAffinity属性主要和singleTask启动模式或者 allowTaskReparenting 属性配对使用。
4 种启动模式
1. standard 标准模式
也是系统默认的启动模式。假如activity A启动了activity B ,activity B则会运行在 activity A 所在的任务栈中。而且每次启动一个 Activity,都会重新创建新的实例,不管这个实例在任务中是否已经存在。非Activity类型的context (如 ApplicationContext )启动standard模式的Activity时会报错。非 Activity类型的 context并没有所谓的任务栈,由于上面第 1 点的原因所以系统会报错。此解决办法就是为待启动Activity指定 FLAG_ACTIVITY_NEW_TASK标记位,这样启动的时候系统就会为它创建一个新的任务栈。这个时候待启动 Activity 其实是以 singleTask模式启动的。
2. singleTop 栈顶复用模式
假如activity A启动了 activity B,就会判断 A 所在的任务栈栈顶是否是 B 的实例。如果是,则不创建新的 activity B 实例而是直接引用这个栈顶实例,同时 onNewIntent 方法会被回调,通过该方法的参数可以取得当前请求的信息;如果不是,则创建新的 activity B实例。
3. singleTask 栈内复用模式
在第一次启动这个 Activity时,系统便会创建一个新的任务,并且初始化Activity的实例,放在新任务的底部。不过需要满足一定条件的。那就是需要设置taskAffinity属性。前面也说过了,taskAffinity 属性是和singleTask模式搭配使用的。
4. singleInstance 单实例模式
这个是singleTask 模式的加强版,它除了具有singleTask模式的所有特性外,它还有一点独特的特性,那就是此模式的Activity 只能单独地位于一个任务栈,不与其他 Activity共存于同一个任务栈。
11.广播注册
首先写一个类要继承BroadCastReceiver
第一种:在清单文件中声明,添加
<receive android:name=".BroadCastReceiverDemo"><intent-filter><action android:name="android.provider.Telephony.SMS_RECEIVED"></intent-filter></receiver>
第二种:使用代码进行注册如:
IntentFilter filter = new IntentFilter("android.provider.Telephony.SMS_RECEIVED");BroadCastReceiverDemo receiver = new BroadCastReceiver();registerReceiver(receiver, filter);
两种注册类型的区别是:
a.第一种是常驻型广播,也就是说当应用程序关闭后,如果有信息广播来,程序也会被系统调用自动运行。b.第二种不是常驻广播,也就是说广播跟随程序的生命周期。
12.Android 中的 ANR
ANR的全称application not responding应用程序未响应。
超出执行时间就会产生ANR。注意:ANR是系统抛出的异常,程序是捕捉不了这个异常的。
解决方法:
运行在主线程里的任何方法都尽可能少做事情。特别是,
Activity应该在它的关键生命周期方法(如onCreate()和onResume())里尽可能少的去做创建操作。可以采用重新开启子线程的方式,然后使用Handler+Message的方式做一些操作,比如更新主线程中的ui等。应用程序应该避免在·BroadcastReceiver·里做耗时的操作或计算。但不再是在子线程里做这些任务
(因为 BroadcastReceiver的生命周期短),替代的是,如果响应Intent广播需要执行一个耗时的动作的话,应用程序应该启动一个Service。
13.ListView 优化
1. convertView 重用
利用好convertView来重用View,切忌每次 getView() 都新建。ListView的核心原理就是重用View,如果重用view 不改变宽高,重用View可以减少重新分配缓存造成的内存频繁分配/回收;
2. ViewHolder 优化
使用ViewHolder的原因是findViewById方法耗时较大,如果控件个数过多,会严重影响性能,而使用ViewHolder主要是为了可以省去这个时间。通过setTag,getTag直接获取View。
3. 减少 Item View 的布局层级
这是所有Layout都必须遵循的,布局层级过深会直接导致View的测量与绘制浪费大量的时间。
4. adapter 中的 getView 方法尽量少使用逻辑
5. 图片加载采用三级缓存,避免每次都要重新加载。
6. 尝试开启硬件加速来使 ListView 的滑动更加流畅。
7. 使用 RecycleView 代替。
14.Android 数字签名
所有的应用程序都必须有数字证书,
Android系统不会安装一个没有数字证书的应用程序Android程序包使用的数字证书可以是自签名的,不需要一个权威的数字证书机构签名认证如果要正式发布一个
Android,必须使用一个合适的私钥生成的数字证书来给程序签名。数字证书都是有有效期的,
Android只是在应用程序安装的时候才会检查证书的有效期。如果程序已经安装在系统中,即使证书过期也不会影响程序的正常功能。
15.Android root 机制
root指的是你有权限可以再系统上对所有档案有 "读" "写" "执行"的权力。root机器不是真正能让你的应用程序具有root权限。它原理就跟linux下的像sudo这样的命令。在系统的bin目录下放个su程序并属主是root并有suid权限。则通过su执行的命令都具有Android root权限。当然使用临时用户权限想把su拷贝的/system/bin目录并改属性并不是一件容易的事情。这里用到2个工具跟2个命令。把busybox拷贝到你有权限访问的目录然后给他赋予4755权限,你就可以用它做很多事了。
16.View、surfaceView、GLSurfaceView
View
显示视图,内置画布,提供图形绘制函数、触屏事件、按键事件函数等,必须在 UI 主线程内更新画面,速度较慢
SurfaceView
基于view视图进行拓展的视图类,更适合2D游戏的开发,是view的子类,类似使用双缓机制,在新的线程中更新画面所以刷新界面速度比view快。
GLSurfaceView
基于SurfaceView视图再次进行拓展的视图类,专用于3D游戏开发的视图,是surfaceView的子类,openGL专用
AsyncTask
AsyncTask 的三个泛型参数说明
第一个参数:传入
doInBackground()方法的参数类型第二个参数:传入
onProgressUpdate()方法的参数类型第三个参数:传入
onPostExecute()方法的参数类型,也是doInBackground()方法返回的类型
运行在主线程的方法:
onPostExecute()onPreExecute()onProgressUpdate(Progress...)
运行在子线程的方法:
doInBackground()
控制 AsyncTask 停止的方法:
cancel(boolean mayInterruptIfRunning)
AsyncTask 的执行分为四个步骤
继承
AsyncTask。实现
AsyncTask中定义的下面一个或几个方法onPreExecute()、doInBackground(Params...)、onProgressUpdate(Progress...)、onPostExecute(Result)。调用
execute方法必须在UI thread中调用。该
task只能被执行一次,否则多次调用时将会出现异常,取消任务可调用cancel。
17.Android i18n
I18n叫做国际化。Android对i18n和L10n提供了非常好的支持。软件在res/vales 以及 其他带有语言修饰符的文件夹。如: values-zh 这些文件夹中 提供语言,样式,尺寸xml 资源。
18.NDK
NDK是一系列工具集合,NDK提供了一系列的工具,帮助开发者迅速的开发C/C++的动态库,并能自动将so和Java应用打成apk包。NDK集成了交叉编译器,并提供了相应的mk文件和隔离cpu、平台等的差异,开发人员只需要简单的修改mk文件就可以创建出so文件。
19.启动一个程序,可以主界面点击图标进入,也可以从一个程序中跳转过去,二者有什么区别?
通过主界面进入,就是设置默认启动的activity。在manifest.xml文件的activity标签中,写以下代码
<intent- filter><intent android:name=“android.intent.action.MAIN”><intent android:name=”android:intent.category.LAUNCHER”></intent-filter>
从另一个组件跳转到目标 activity ,需要通过 intent 进行跳转。具体
Intent intent=new Intent(this,activity.class),startActivity(intent)
20.内存溢出和内存泄漏有什么区别?何时会产生内存泄漏?
内存溢出:
当程序运行时所需的内存大于程序允许的最高内存,这时会出现内存溢出;
内存泄漏:
在一些比较消耗资源的操作中,如果操作中内存一直未被释放,就会出现内存泄漏。比如未关闭io,cursor。
21.sim 卡的 EF 文件有何作用
sim卡就是电话卡,sim卡内有自己的操作系统,用来与手机通讯的。Ef文件用来存储数据的。
22.Activity 的状态有几种?
主要有以下三种状态:
1.运行 2.暂停 3.停止
23.让 Activity 变成一个窗口
设置activity的style属性=”@android:style/Theme.Dialog”
<activityandroid:name=".CondorMainActivity"android:label="@string/app_name"
android:theme="@android:style/Theme.Dialog" ><intent-filter><action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity>
24.android:gravity 与 android:layout_gravity 的区别
gravity:表示组件内元素的对齐方式layout_gravity:相对于父类容器,该视图组件的对齐方式
25.如何退出 Activity
结束当前activity
Finish()killProgress()System.exit(0)
关闭应用程序时,结束所有的activity可以创建一个List集合,每新创建一个activity,将该activity的实例放进list中,程序结束时,从集合中取出循环取出activity实例,调用finish()方法结束
26.如果后台的 Activity 由于某原因被系统回收了,如何在被系统回收之前保存当前状态?
在onPuase方法中调用onSavedInstanceState()
27.Android 中的长度单位详解
Px:像素Sp与dp是长度单位,但是与屏幕的单位密度无关.
28.activity,service,intent 之间的关系
这三个都是Android应用频率非常的组件。Activity与service是四大核心组件。Activity用来加载布局,显示窗口界面,service运行后台,没有界面显示,intent是activity与service的通信使者。
29.activity 之间传递参数,除了 intent,广播接收器,contentProvider 之外,还有那些方法?
File:文件存储,推荐使用sharedPreferecnces静态变量
30.Adapter 是什么?你所接触过的 adapter 有那些?
是适配器,用来为列表提供数据适配的。经常使用的adapter有baseadapter,arrayAdapter,SimpleAdapter,cursorAdapter,SpinnerAdapter等
31.Fragment 与 activity 如何传值和交互?
Fragment对象有一个getActivity()的方法,通过该方法与activity交互使用framentmentManager.findFragmentByXX可以获取fragment对象,在activity中直接操作fragment对象。
32.如果 Listview 中的数据源发生改变,如何更新 listview 中的数据
使用adapter的notifyDataSetChanged方法
33.广播接受者的生命周期?
广播接收者的生命周期非常短。当执行onRecieve方法之后,广播就会销毁在广播接受者不能进行耗时较长的操作在广播接收者不要创建子线程。广播接收者完成操作后,所在进程会变成空进程,很容易被系统回收
34.ContentProvider 与 sqlite 有什么不一样的?
ContentProvider 会对外隐藏内部实现,只需要关注访问contentProvider的uri即可,contentProvider应用在app间共享。Sqlite操作本应用程序的数据库。ContentProiver`可以对本地文件进行增删改查操作
35.如何保存 activity 的状态?
默认情况下activity的状态系统会自动保存,有些时候需要我们手动调用保存。
当activity处于onPause,onStop之后,activity处于未活动状态,但是activity对象却仍然存在。当内存不足,onPause,onStop之后的activity可能会被系统摧毁。
当通过返回退出activity时,activity状态并不会保存。
保存activity状态需要重写onSavedInstanceState()方法,在执行onPause,onStop之前调用onSavedInstanceState方法,onSavedInstanceState需要一个Bundle类型的参数,我们可以将数据保存到bundle中,通过实参传递给onSavedInstanceState方法。
Activity被销毁后,重新启动时,在onCreate方法中,接受保存的bundle参数,并将之前的数据取出。
36.Android 中 activity,context,application 有什么不同。
Content与application都继承与contextWrapper,contextWrapper继承于Context类。
Context:表示当前上下文对象,保存的是上下文中的参数和变量,它可以让更加方便访问到一些资源。Context通常与activity的生命周期是一样的,application表示整个应用程序的对象。
对于一些生命周期较长的,不要使用context,可以使用application。
在activity中,尽量使用静态内部类,不要使用内部类。内部里作为外部类的成员存在,不是独立于activity,如果内存中还有内存继续引用到context,activity如果被销毁,context还不会结束。
37.Service 是否在 main thread 中执行, service 里面是否能执行耗时的操作?
默认情况service在main thread中执行,当service在主线程中运行,那在service中不要进行一些比较耗时的操作,比如说网络连接,文件拷贝等。
38.Service 和 Activity 在同一个线程吗
默认情况下service与activity在同一个线程,都在main Thread,或者ui线程中。
如果在清单文件中指定service的process属性,那么service就在另一个进程中运行。
39.Service 里面可以弹 Toast 么
可以。
40.在 service 的生命周期方法 onstartConmand()可不可以执行网络操作?如何在 service 中执行网络操作?
可以的,就在onstartConmand方法内执行。
41.说说 ContentProvider、ContentResolver、ContentObserver 之间的关系
ContentProvider:内容提供者,对外提供数据的操作,contentProvider.notifyChanged(uir):可以更新数据contentResolver:内容解析者,解析ContentProvider返回的数据ContentObServer:内容监听者,监听数据的改变,contentResolver.registerContentObServer()
42.请介绍下 ContentProvider 是如何实现数据共享的
ContentProvider是一个对外提供数据的接口,首先需要实现ContentProvider这个接口,然后重写query,insert,getType,delete,update方法,最后在清单文件定义contentProvider的访问uri。
43.Intent 传递数据时,可以传递哪些类型数据?
1.基本数据类型以及对应的数组类型
2.可以传递bundle类型,但是bundle类型的数据需要实现Serializable或者parcelable接口
44.Serializable 和 Parcelable 的区别?
如果存储在内存中,推荐使用parcelable,使用serialiable在序列化的时候会产生大量的临时变量,会引起频繁的GC
如果存储在硬盘上,推荐使用Serializable,虽然serializable效率较低
Serializable的实现:只需要实现Serializable接口,就会自动生成一个序列化id
Parcelable的实现:需要实现Parcelable接口,还需要Parcelable.CREATER变量
45.请描述一下Intent 和 IntentFilter
Intent是组件的通讯使者,可以在组件间传递消息和数据。IntentFilter是intent的筛选器,可以对intent的action,data,catgory,uri这些属性进行筛选,确定符合的目标组件。
46.什么是 IntentService?有何优点?
IntentService是Service的子类,比普通的 Service增加了额外的功能。先看Service本身存在两个问题:
1.Service 不会专门启动一条单独的进程,Service与它所在应用位于同一个进程中;2.Service 也不是专门一条新线程,因此不应该在 Service中直接处理耗时的任务;
特征
会创建独立的 worker线程来处理所有的Intent请求;会创建独立的worker线程来处理 onHandleIntent()方法实现的代码,无需处理多线程问题;所有请求处理完成后,IntentService会自动停止,无需调用 stopSelf()方法停止 Service;为 Service的 onBind()提供默认实现,返回 null;为 Service的 onStartCommand 提供默认实现,将请求 Intent 添加到队列中
使用
让 service类继承IntentService,重写onStartCommand和onHandleIntent实现
47.Android 引入广播机制的用意
从 MVC 的角度考虑(应用程序内) 其实回答这个问题的时候还可以这样问,android为什么要有那 4 大组件,现在的移动开发模型基本上也是照搬的 web那一套 MVC架构,只不过稍微做了修改。android的四大组件本质上就是为了实现移动或者说嵌入式设备上的 MVC架构,它们之间有时候是一种相互依存的关系,有时候又是一种补充关系,引入广播机制可以方便几大组件的信息和数据交互。
程序间互通消息(例如在自己的应用程序内监听系统来电)
效率上(参考UDP的广播协议在局域网的方便性)
设计模式上(反转控制的一种应用,类似监听者模式)
48.ListView 如何提高其效率?
当 convertView为空时,用setTag()方法为每个 View 绑定一个存放控件的 ViewHolder对象。当convertView不为空, 重复利用已经创建的view 的时候, 使用 getTag()方法获取绑定的 ViewHolder对象,这样就避免了findViewById对控件的层层查询,而是快速定位到控件。 复用 ConvertView,使用历史的view,提升效率 200%
自定义静态类 ViewHolder,减少 findViewById 的次数。提升效率 50%
异步加载数据,分页加载数据。
使用WeakRefrence 引用ImageView 对象
49.ListView 如何实现分页加载
设置 ListView的滚动监听器:setOnScrollListener(new OnScrollListener{….})在监听器中有两个方法: 滚动状态发生变化的方法(onScrollStateChanged)和listView被滚动时调用的方法(onScroll)
在滚动状态发生改变的方法中,有三种状态:手指按下移动的状态:SCROLL_STATE_TOUCH_SCROLL惯性滚动(滑翔(flgin)状态):SCROLL_STATE_FLING:静止状态:SCROLL_STATE_IDLE:
分批加载数据,只关心静止状态:关心最后一个可见的条目,如果最后一个可见条目就是数据适配器(集合)里的最后一个,此时可加载更多的数据。在每次加载的时候,计算出滚动的数量,当滚动的数量大于等于总数量的时候,可以提示用户无更多数据了。
50.ListView 可以显示多种类型的条目吗
这个当然可以的,ListView 显示的每个条目都是通过 baseAdapter 的 getView(int position,View convertView, ViewGroup parent)来展示的,理论上我们完全可以让每个条目都是不同类型的 view。
比如:从服务器拿回一个标识为id=1,那么当id=1的时候,我们就加载类型一的条目,当 id=2的时候,加载类型二的条目。常见布局在资讯类客户端中可以经常看到。
除此之外adapter 还提供了 getViewTypeCount()和 getItemViewType(int position)两个方法。在 getView方法中我们可以根据不同的 viewtype加载不同的布局文件。
51.ListView 如何定位到指定位置
可以通过 ListView提供的 lv.setSelection(listView.getPosition())方法。
52.如何在 ScrollView 中如何嵌入 ListView
通常情况下我们不会在 ScrollView中嵌套 ListView。
在 ScrollView 添加一个 ListView会导致listview 控件显示不全,通常只会显示一条,这是因为两个控件的滚动事件冲突导致。所以需要通过 listview 中的item 数量去计算listview的显示高度,从而使其完整展示。
现阶段最好的处理的方式是: 自定义 ListView,重载 onMeasure()方法,设置全部显示。
53.Manifest.xml 文件中主要包括哪些信息?
manifest:根节点,描述了package中所有的内容。uses-permission:请求你的package正常运作所需赋予的安全许可。permission:声明了安全许可来限制哪些程序能你package中的组件和功能。instrumentation:声明了用来测试此package或其他package指令组件的代码。application:包含package中application级别组件声明的根节点。activity:Activity 是用来与用户交互的主要工具。receiver:IntentReceiver能使的application获得数据的改变或者发生的操作,即使它当前不在运行。service:Service是能在后台运行任意时间的组件。provider:ContentProvider是用来管理持久化数据并发布给其他应用程序使用的组件。
54.ListView 中图片错位的问题是如何产生的
图片错位问题的本质源于我们的 listview使用了缓存convertView, 假设一种场景, 一个 listview一屏显示九个 item,那么在拉出第十个item 的时候,事实上该item是重复使用了第一个 item,也就是说在第一个item 从网络中下载图片并最终要显示的时候,其实该 item已经不在当前显示区域内了,此时显示的后果将可能在第十个item上输出图像,这就导致了图片错位的问题。所以解决办法就是可见则显示,不可见则不显示。
55.Fragment 的 replace 和 add 方法的区别
Fragment本身并没有 replace 和 add方法,FragmentManager才有replace和add方法。我们经常使用的一个架构就是通过RadioGroup切换Fragment,每个Fragment 就是一个功能模块。
Fragment的容器一个FrameLayout,add的时候是把所有的 Fragment一层一层的叠加到了。FrameLayout上了,而 replace的话首先将该容器中的其他Fragment去除掉然后将当前Fragment添加到容器中。
一个Fragment 容器中只能添加一个Fragment 种类,如果多次添加则会报异常,导致程序终止,而replace 则无所谓,随便切换。因为通过 add的方法添加的 Fragment,每个 Fragment只能添加一次,因此如果要想达到切换效果需要通过Fragment 的的hide 和 show方法结合者使用。将要显示的show 出来,将其他hide起来。这个过程 Fragment的生命周期没有变化。
通过 replace 切换Fragment,每次都会执行上一个Fragment的 onDestroyView,新 Fragment的 onCreateView、onStart、onResume方法。基于以上不同的特点我们在使用的使用一定要结合着生命周期操作我们的视图和数据。
56.Fragment 如何实现类似 Activity 栈的压栈和出栈效果的?
Fragment的事物管理器内部维持了一个双向链表结构,该结构可以记录我们每次 add 的Fragment 和 replace的Fragment,然后当我们点击 back 按钮的时候会自动帮我们实现退栈操作。
57.Fragment 在你们项目中的使用
Fragment是android3.0以后引入的的概念,做局部内容更新更方便,原来为了到达这一点要把多个布局放到一个 activity里面,现在可以用多 Fragment 来代替,只有在需要的时候才加载Fragment,提高性能。
Fragment 的好处:
Fragment可以使你能够将 activity分离成多个可重用的组件,每个都有它自己的生命周期和UI。Fragment可以轻松得创建动态灵活的UI 设计,可以适应于不同的屏幕尺寸。从手机到平板电脑。Fragment是一个独立的模块,紧紧地与 activity 绑定在一起。可以运行中动态地移除、加入、交换等。Fragment提供一个新的方式让你在不同的安卓设备上统一你的 UI。Fragment解决 Activity间的切换不流畅,轻量切换。Fragment 替代TabActivity做导航,性能更好。Fragment 在 4.2.版本中新增嵌套 fragment使用方法,能够生成更好的界面效果。
58.如何切换 fragement,不重新实例化
翻看了Android官方Doc,和一些组件的源代码,发现 replace()这个方法只是在上一个 Fragment不再需要时采用的简便方法.
正确的切换方式是 add(),切换时hide(),add()另一个 Fragment;再次切换时,只需 hide()当前,show()另一个。
这样就能做到多个 Fragment 切换不重新实例化:
59.如何对 Android 应用进行性能分析
如果不考虑使用其他第三方性能分析工具的话,我们可以直接使用ddms 中的工具,其实 ddms 工具已经非常的强大了。ddms中有 traceview、heap、allocation tracker等工具都可以帮助我们分析应用的方法执行时间效率和内存使用情况。
Traceview是 Android平台特有的数据采集和分析工具,它主要用于分析 Android中应用程序的 hotspot(瓶颈)。Traceview本身只是一个数据分析工具,而数据的采集则需要使用 AndroidSDK 中的Debug类或者利用 DDMS 工具。
heap工具可以帮助我们检查代码中是否存在会造成内存泄漏的地方。
allocation tracker是内存分配跟踪工具
60.Android 中如何捕获未捕获的异常
UncaughtExceptionHandler
自 定 义 一 个 Application, 比 如 叫MyApplication 继 承 Application实 现UncaughtExceptionHandler。覆写 UncaughtExceptionHandler 的onCreate和 uncaughtException方法。注意:上面的代码只是简单的将异常打印出来。在onCreate 方法中我们给Thread类设置默认异常处理 handler,如果这句代码不执行则一切都是白搭。在uncaughtException方法中我们必须新开辟个线程进行我们异常的收集工作,然后将系统给杀死。在 AndroidManifest中配置该 Application:<application android:name="com.example.uncatchexception.MyApplication"
Bug 收集工具 Crashlytics
Crashlytics是专门为移动应用开发者提供的保存和分析应用崩溃的工具。国内主要使用的是友盟做数据统计。Crashlytics 的好处:1.Crashlytics不会漏掉任何应用崩溃信息。2.Crashlytics 可以像Bug管理工具那样,管理这些崩溃日志。3.Crashlytics 可以每天和每周将崩溃信息汇总发到你的邮箱,所有信息一目了然。
61.如何将 SQLite 数据库(dictionary.db 文件)与 apk 文件一起发布
把这个文件放在/res/raw目录下即可。res\raw目录中的文件不会被压缩,这样可以直接提取该目录中的文件,会生成资源id。
62.什么是 IntentService?有何优点?
IntentService是 Service 的子类,比普通的 Service增加了额外的功能。先看 Service 本身存在两个问题:
Service 不会专门启动一条单独的进程,Service 与它所在应用位于同一个进程中;Service 也不是专门一条新线程,因此不应该在Service 中直接处理耗时的任务;
IntentService 特征
会创建独立的 worker线程来处理所有的Intent请求;会创建独立的 worker 线程来处理onHandleIntent()方法实现的代码,无需处理多线程问题;所有请求处理完成后,IntentService会自动停止,无需调用 stopSelf()方法停止 Service;为Service 的 onBind()提供默认实现,返回 null;为 Service的 onStartCommand 提供默认实现,将请求Intent添加到队列中;
63.谈谈对 Android NDK 的理解
NDK是一系列工具的集合.NDK提供了一系列的工具,帮助开发者快速开发C或C++的动态库,并能自动将so和java应用一起打包成apk.这些工具对开发者的帮助是巨大的.NDK集成了交叉编译器,并提供了相应的mk文件隔离CPU,平台,ABI等差异,开发人员只需要简单修改 mk文件(指出"哪些文件需要编译","编译特性要求"等),就可以创建出so.
NDK可以自动地将so和Java应用一起打包,极大地减轻了开发人员的打包工作.NDK提供了一份稳定,功能有限的API头文件声明.
Google明确声明该API是稳定的,在后续所有版本中都稳定支持当前发布的API.从该版本的NDK中看出,这些 API支持的功能非常有限,包含有:C标准库(libc),标准数学库(libm ),压缩库(libz),Log库(liblog).
64.AsyncTask 使用在哪些场景?它的缺陷是什么?如何解决?
AsyncTask 运用的场景就是我们需要进行一些耗时的操作,耗时操作完成后更新主线程,或者在操作过程中对主线程的UI进行更新。
缺陷:
AsyncTask中维护着一个长度为128的线程池,同时可以执行5个工作线程,还有一个缓冲队列,当线程池中已有128个线程,缓冲队列已满时,如果 此时向线程提交任务,将会抛出RejectedExecutionException。
解决:
由一个控制线程来处理AsyncTask的调用判断线程池是否满了,如果满了则线程睡眠否则请求AsyncTask继续处理。
65.Android 线程间通信有哪几种方式(重要)
1.共享内存(变量)2.文件,数据库 3.Handler4.Java 里的 wait(),notify(),notifyAll()
66.请解释下 Android 程序运行时权限与文件系统权限的区别?
apk 程序是运行在虚拟机上的,对应的是Android 独特的权限机制,只有体现到文件系统上时才
使用 linux 的权限设置。
linux文件系统上的权限-rwxr-x--x system system 4156 2010-04-30 16:13 test.apk代表的是相应的用户/用户组及其他人对此文件的访问权限,与此文件运行起来具有的权限完全不相关。比如上面的例子只能说明 system 用户拥有对此文件的读写执行权限;system 组的用户对此文件拥有读、执行权限;其他人对此文件只具有执行权限。而 test.apk运行起来后可以干哪些事情,跟这个就不相关了。千万不要看apk 文件系统上属于system/system 用户及用户组,或者root/root 用户及用户组,就认为apk 具有system 或 root权限
Android 的权限规则
Android中的apk必须签名基于 UserID 的进程级别的安全机制默认 apk生成的数据对外是不可见的AndroidManifest.xml 中的显式权限声明
67.Framework 工作方式及原理,Activity 是如何生成一个 view 的,机制是什么?
所有的框架都是基于反射 和 配置文件(manifest)的。
普通的情况:
Activity创建一个 view 是通过 ondraw画出来的, 画这个view之前呢,还会调用 onmeasure方法来计算显示的大小.
特殊情况:
Surfaceview是直接操作硬件的,因为 或者视频播放对帧数有要求,onDraw 效率太低,不够使,Surfaceview 直接把数据写到显存。
68.什么是 AIDL?如何使用?
aidl是 Android interface definition Language 的英文缩写,意思 Android 接口定义语言。
使用aidl可以帮助我们发布以及调用远程服务,实现跨进程通信。
将服务的 aidl 放到对应的 src目录,工程的 gen目录会生成相应的接口类我们通过 bindService(Intent,ServiceConnect,int)方法绑定远程服务,在 bindService中 有 一 个 ServiceConnect 接 口 , 我 们 需 要 覆 写 该 类 的onServiceConnected(ComponentName,IBinder)方法,这个方法的第二个参数IBinder对象其实就是已经在 aidl中定义的接口,因此我们可以将IBinder 对象强制转换为aidl中的接口类。我们通过IBinder 获取到的对象(也就是 aidl文件生成的接口)其实是系统产生的代理对象,该代理对象既可以跟我们的进程通信, 又可以跟远程进程通信, 作为一个中间的角色实现了进程间通信。
69.AIDL 的全称是什么?如何工作?能处理哪些类型的数据?
AIDL全称 Android Interface Definition Language(AndRoid 接口描述语言) 是一种接口描述语言; 编译器可以通过 aidl文件生成一段代码,通过预先定义的接口达到两个进程内部通信进程跨界对象访问的目的。需要完成两件事情:
1.引入AIDL 的相关类.;2.调用aidl产生的 class
理论上, 参数可以传递基本数据类型和 String, 还有就是 Bundle的派生类, 不过在Eclipse中,目前的 ADT 不支持Bundle 做为参数。
70.Android 判断 SD 卡是否存在
/**
判断 SD 是否挂载*/public static boolean isSDCardMount() {return Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED);}
71.Android 中任务栈的分配
Task实际上是一个Activity栈,通常用户感受的一个Application就是一个Task。从这个定义来看,Task跟Service或者其他Components是没有任何联系的,它只是针对Activity而言的。
Activity有不同的启动模式, 可以影响到task的分配
72.SQLite 支持事务吗? 添加删除如何提高性能?
在sqlite插入数据的时候默认一条语句就是一个事务,有多少条数据就有多少次磁盘操作 比如5000条记录也就是要5000次读写磁盘操作。
添加事务处理,把多条记录的插入或者删除作为一个事务
73.Android 中 touch 事件的传递机制是怎样的?
1.Touch事件传递的相关API有dispatchTouchEvent、onTouchEvent、onInterceptTouchEvent2.Touch事件相关的类有View、ViewGroup、Activity3.Touch事件会被封装成MotionEvent对象,该对象封装了手势按下、移动、松开等动作 4.Touch事件通常从Activity#dispatchTouchEvent发出,只要没有被消费,会一直往下传递,到最底层的View。5.如果Touch事件传递到的每个View都不消费事件,那么Touch事件会反向向上传递,最终交由Activity#onTouchEvent处理.6.onInterceptTouchEvent为ViewGroup特有,可以拦截事件.7.Down事件到来时,如果一个View没有消费该事件,那么后续的MOVE/UP事件都不会再给它
74.描述下 Handler 机制
1)Looper:
一个线程可以产生一个Looper对象,由它来管理此线程里的MessageQueue(消息队列)。
2)Handler:
你可以构造Handler对象来与Looper沟通,以便push新消息到MessageQueue里;或者接收Looper从Message Queue取出所送来的消息。
Message Queue(消息队列):











评论