Android面试题问答整理
概述
找工作挺难,为更好的应对面试,我将一些面试题收集整理起来,好让自己随时复习。
1.请谈一下android系统的架构。
答:android系统采用了分层架构,从高层到低层分别是应用程序层、应用程序框架层、系统运行库层和linux核心层。
2.谈谈android大众常用的五种布局。
答:在android中,共有五种布局方式,分别是:framelayout(框架布局),linearlayout (线性布局),absolutelayout(绝对布局),relativelayout(相对布局),tablelayout(表格布局)。
- (1)framelayout 框架布局,放入其中的所有元素都被放置在最左上的区域,而且无法为这些元素指定一个确切的位置,下一个子元素会重叠覆盖上一个子元素,适合浏览单张图片。
- (2)linearlayout 线性布局,是应用程序中最常用的布局方式,主要提供控件水平或者垂直排列的模型,每个子组件都是以垂直或水平的方式来定位.(默认是垂直)
- (3)absolutelayout 绝对定位布局,采用坐标轴的方式定位组件,左上角是(0,0)点,往右x轴递增,往下y轴递增,组件定位属性为android:layout_x 和 android:layout_y来确定坐标。
- (4)relativelayout 相对布局,根据另外一个组件或是顶层父组件来确定下一个组件的位置。和css里面的类似。
- (5)tablelayout 表格布局,类似html里的table.使用tablerow来布局,其中tablerow代表一行,tablerow的每一个视图组件代表一个单元格。
3.谈谈android数据存储方式。
答:android提供了5种方式存储数据:
- (1)使用sharedpreferences存储数据;它是android提供的用来存储一些简单配置信息的一种机制,采用了xml格式将数据存储到设备中。只能在同一个包内使用,不能在不同的包之间使用。
- (2)文件存储数据;文件存储方式是一种较常用的方法,在android中读取/写入文件的方法,与java中实现i/o的程序是完全一样的,提供了openfileinput()和openfileoutput()方法来读取设备上的文件。
- (3)sqlite数据库存储数据;sqlite是android所带的一个标准的数据库,它支持sql语句,它是一个轻量级的嵌入式数据库。
- (4)使用contentprovider存储数据;主要用于应用程序之间进行数据交换,从而能够让其他的应用保存或读取此content provider的各种数据类型。
- (5)网络存储数据;通过网络上提供给我们的存储空间来上传(存储)和下载(获取)我们存储在网络空间中的数据信息。
4.android中activity, intent, content provider, service各有什么区别。
答:
- activity: 活动,是最基本的android应用程序组件。一个活动就是一个单独的屏幕,每一个活动都被实现为一个独立的类,并且从活动基类继承而来。
- intent: 意图,描述应用想干什么。最重要的部分是动作和动作对应的数据。
- content provider:内容提供器,android应用程序能够将它们的数据保存到文件、sqlite数据库中,甚至是任何有效的设备中。当你想将你的应用数据和其他应用共享时,内容提供器就可以发挥作用了。
- service:服务,具有一段较长生命周期且没有用户界面的程序。
5.view, surfaceview, glsurfaceview有什么区别。
答:
- view是最基础的,必须在ui主线程内更新画面,速度较慢。
- surfaceview 是view的子类,类似使用双缓机制,在新的线程中更新画面所以刷新界面速度比view快
- glsurfaceview 是surfaceview的子类,opengl 专用的
6.adapter有什么作用?常见的adapter有哪些?
答:
adapter是连接后端数据和前端显示的适配器接口。常见的adapter有arrayadapter, baseadapter, cursoradapter, headerviewlistadapter, listadapter, resourcecursoradapter, simpleadapter, simplecursoradapter, spinneradapter, wrapperlistadapter等
7.manifest.xml文件中主要包括哪些信息?
答:
- manifest:根节点,描述了package中所有的内容。
- uses-permission:请求你的package正常运作所需赋予的安全许可。
- permission: 声明了安全许可来限制哪些程序能你package中的组件和功能。
- instrumentation:声明了用来测试此package或其他package指令组件的代码。
- application:包含package中application级别组件声明的根节点。
- activity:activity是用来与用户交互的主要工具。
- receiver:intentreceiver能使的application获得数据的改变或者发生的操作,即使它当前不在运行。
- service:service是能在后台运行任意时间的组件。
- provider:contentprovider是用来管理持久化数据并发布给其他应用程序使用的组件。
8.请写一段代码(sax, dom, 或者pull )来解析xml文档。
答:下面是要解析的xml文件:
张三
22 李四
23 定义一个名为person的javabean用于存放上面解析出来的xml内容
public class person { private integer id; private string name; private short age; public integer getid() { return id; } public void setid(integer id) { this.id = id; } public string getname() { return name; } public void setname(string name) { this.name = name; } public short getage() { return age; } public void setage(short age) { this.age = age; } } (1)使用sax读取xml文件;它采用的是事件驱动,并不需要解析完整个文档,速度快并且占用内存少。需要为sax提供实现contenthandler接口的类。 persondefaulthandler.java import java.util.arraylist; import java.util.list; import org.xml.sax.attributes; import org.xml.sax.saxexception; import org.xml.sax.helpers.defaulthandler; import com.sinber.domain.person; public class persondefaulthandler extends defaulthandler { private list persons; private person person ; //记录当前person private string pertag; //记录前一个标签的名称 /** * 重写父类的开始文档方法。用于初始化 */ @override public void startdocument() throws saxexception { persons = new arraylist(); } @override public void startelement(string uri, string localname, string qname, attributes attributes) throws saxexception { if(“person”.equals(localname)){ integer id = new integer(attributes.getvalue(0)); //取id person = new person(); person.setid(id); } pertag = localname; } /**参数: * ch 整个xml字符串 * start 节点值在整个xml字符串中的索引位置 * length 节点值的长度 */ @override public void characters(char[] ch, int start, int length) throws saxexception { if(pertag!=null){ string data = new string(ch,start,length); if(“name”.equals(pertag)){ person.setname(data); }else if(“age”.equals(pertag)){ person.setage(new short(data)); } } } @override public void endelement(string uri, string localname, string qname) throws saxexception { if(“person”.equals(localname)){ persons.add(person); person = null; } pertag = null; } public list getpersons() { return persons; } } saxperson.java import java.io.inputstream; import java.util.list; import javax.xml.parsers.saxparser; import javax.xml.parsers.saxparserfactory; import com.sinber.domain.person; public class saxperson{ public static list getperson() throws exception{ //通过类装载器获取文件 inputstream instream = saxpersonservice.class.getclassloader().getresourceasstream(“person.xml”); saxparserfactory factory = saxparserfactory.newinstance(); saxparser saxparser = factory.newsaxparser(); persondefaulthandler handler = new persondefaulthandler(); saxparser.parse(instream, handler); instream.close(); return handler.getpersons(); } } (2)dom解析xml文件时,会将xml文件的所有内容读取到内存中,然后允许您使用dom api遍历xml树、检索所需的数据。 domperson.java import java.io.inputstream; import java.util.arraylist; import java.util.list; import javax.xml.parsers.documentbuilder; import javax.xml.parsers.documentbuilderfactory; import org.w3c.dom.document; import org.w3c.dom.element; import org.w3c.dom.node; import org.w3c.dom.nodelist; import com.sinber.domain.person; public class domperson { public static list getperson() throws exception{ list pers = new arraylist(); inputstream instream = saxpersonservice.class.getclassloader().getresourceasstream(“person.xml”); documentbuilderfactory factory =documentbuilderfactory.newinstance(); documentbuilder builder = factory.newdocumentbuilder(); document dom = builder.parse(instream); element root = dom.getdocumentelement(); nodelist persons = root.getelementsbytagname(“person”); for(int i=0;i<persons.getlength();i++){ element personnode =(element)persons.item(i); person person = new person(); person.setid(new integer(personnode.getattribute("id"))); nodelist childnodes = personnode.getchildnodes(); for(int j=0;j<childnodes.getlength();j++){ node childnode = childnodes.item(j); if(childnode.getnodetype()==node.element_node){ element element = (element)childnode; if("name".equals(childnode.getnodename())){ person.setname(new string(element.getfirstchild().getnodevalue())); }else if("age".equals(childnode.getnodename())){ person.setage(new short(element.getfirstchild().getnodevalue())); } } } pers.add(person); } instream.close(); return pers; } } (3)使用pull解析器读取xml文件 pullperson.java import java.io.file; import java.io.fileoutputstream; import java.io.inputstream; import java.util.arraylist; import java.util.list; import org.xmlpull.v1.xmlpullparser; import org.xmlpull.v1.xmlserializer; import android.os.environment; import android.util.xml; import com.sinber.domain.person; public class pullperson { public static void save(list persons) throws exception{ xmlserializer serializer = xml.newserializer(); file file = new file(environment.getexternalstoragedirectory(),”person.xml”); fileoutputstream outstream = new fileoutputstream(file); serializer.setoutput(outstream,”utf-8″); serializer.startdocument(“utf-8″, true); serializer.starttag(“”, “persons”); for(person person:persons){ serializer.starttag(“”, “person”); //person serializer.attribute(“”, “id”, “”+person.getid()); serializer.starttag(“”, “name”); //name serializer.text(person.getname()); serializer.endtag(“”, “name”); //name serializer.starttag(“”, “age”); //age serializer.text(person.getage().tostring()); serializer.endtag(“”, “age”);//age serializer.endtag(“”, “person”); //person } serializer.endtag(“”, “persons”); serializer.enddocument(); outstream.close(); } public static list getpersons() throws exception{ list persons = null; person person = null; xmlpullparser parser= xml.newpullparser(); inputstream instream = pullpersonservice.class.getclassloader().getresourceasstream(“person.xml”); parser.setinput(instream, “utf-8″); int eventtype = parser.geteventtype(); //触发第一个事件 while(eventtype!=xmlpullparser.end_document){ switch(eventtype){ case xmlpullparser.start_document: persons = new arraylist(); break; case xmlpullparser.start_tag: //开始元素事件 if(“person”.equals(parser.getname())){ person = new person(); person.setid(new integer(parser.getattributevalue(0))); }else if(person!=null){ if(“name”.equals(parser.getname())){ person.setname(parser.nexttext()); }else if(“age”.equals(parser.getname())){ person.setage(new short(parser.nexttext())); } } break; case xmlpullparser.end_tag: //结束元素事件 if(“person”.equals(parser.getname())){ persons.add(person); person = null; } break; default: break; } eventtype = parser.next(); } return persons; } } 以上三种方式任选其一即可。
9.根据自己的理解描述下android数字签名。
答:
- (1)所有的应用程序都必须有数字证书,android系统不会安装一个没有数字证书的应用程序
- (2)android程序包使用的数字证书可以是自签名的,不需要一个权威的数字证书机构签名认证
- (3)如果要正式发布一个android ,必须使用一个合适的私钥生成的数字证书来给程序签名,而不能使用adt插件或者ant工具生成的调试证书来发布。
- (4)数字证书都是有有效期的,android只是在应用程序安装的时候才会检查证书的有效期。如果程序已经安装在系统中,即使证书过期也不会影响程序的正常功能。
10.已知单链表的头结构head,写一个函数把这个链表逆序。
答: 如下所示
node.java public class node { private integer count; private node nextnode; public node(){ } public node(int count){ this.count = new integer(count); } public integer getcount() { return count; } public void setcount(integer count) { this.count = count; } public node getnextnode() { return nextnode; } public void setnextnode(node nextnode) { this.nextnode = nextnode; } } reversesinglelink.java public class reversesinglelink { public static node revsinglelink(node head){ if(head == null){ //链表为空不能逆序 return head; } if(head.getnextnode()==null){ //如果只有一个结点,当然逆过来也是同一个 return head; } node rhead = revsinglelink(head.getnextnode()); head.getnextnode().setnextnode(head); head.setnextnode(null); return rhead; } public static void main(string[] args){ node head = new node(0); node temp1 = null,temp2 = null; for(int i=1;i<100;i++){ temp1 = new node(i); if(i==1){ head.setnextnode(temp1); }else{ temp2.setnextnode(temp1); } temp2 = temp1; } head = revsinglelink(head); while(head!=null){ head = head.getnextnode(); } } }
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。如果你想了解更多相关内容请查看下面相关链接