Android面试准备 第一天 第2-4例

參考:http://blog.csdn.net/lmj623565791/article/details/24015867


2、假设有个100M大的文件。须要上传至server中。而serverform表单最大仅仅能上传2M,能够用什么方法。

个人理解:所谓表单最大仅仅能上传2M。是不是指一个表单中附件仅仅能上传最大2M。假设是的话,如今要求上传100M,为什么不直接

把附件上传大小设置为100M就能够了,除此菜鸟的我,并想不出这个题目到底要考核面试者的什么知识点。

标准答案:这个问题不是非常明白我认为,首先来说使用http协议上传数据。特别在android下,跟form没什么关系。传统的在web中,在form中写文件上传,事实上浏览器所做的就是将我们的数据进行解析组拼成字符串。以流的方式发送到server,且上传文件用的都是POST方式,POST方式对大小没什么限制。回到题目。能够说如果每次真的仅仅能上传2M。那么可能我们仅仅能把文件截断,然后分别上传了。

二次理解:由于android程序中页面都是活动和碎片,跟form没有什么关系,仅仅有一个类似form的布局tablelayout,但跟form字面上都不是一回事所以android并不存在form表单说法,应该是指htmlform表单。那么表单附件上传中首先把数据解析字符串文件以IO流的方式上传到server,并且并没有上传限制,假设有话,应该是什么原因,防止长时间占用server,减少效率。那么把文件截断上传还有意义嘛,这个问题不具备普遍性。所以不进行深究。



3、内存溢出和内存泄漏有什么差别?何时会产生内存泄漏?内存优化有哪些方法?

个人理解:不知道他们的差别。以为是同一件事情,比方内存中创建的对象过多时,导致内容空间不够,会产生该现象;及时关闭不必要的

资源,如数据库连接、IO流等,另外能够通过代码重构。尽可能降低对象的创建数量。

标准答案:

内存溢出通俗理解就是软件(应用)执行须要的内存,超出了它可用的最大内存。

内存泄漏就是我们对某一内存空间的使用,使用完毕后没有释放。

内存优化:Android中easy内存溢出的部分。就是图片的载入。我们可以使用图片的压缩加上使用LruCache缓存的目的来控制图片所可以使用的内存。还有对于比較耗资源的对象及时的关闭。比如Database Conn , 各种传感器 Service等等

     

二次理解:内存溢出指内存超出它可用的最大内容,而内存泄露则特指使用的资源没有释放。而内容中easy内存溢出的就是存储载入,使用压缩和缓存的方式,而内存泄露则须要及时关闭资源,比方数据连接、服务等。概念话的内容,并没有实际的參照价值,由于临时没有学习图片载入的使用        


 4AsyncTask使用在哪些场景?它的缺陷是什么?怎样解决?                                         

个人理解:AsyncTask也是一种多线程的通信机制,可是AsyncTask并不须要去创建新的线程。把须要的操作直接定义在该对象的方法中就能够了,详细什么方法忘记。在第一行代码由于学习的Handler机制所以并没有加深对AsyncTask的理解,至于它的缺陷和解决方案更是不清楚了。

标准答案:

AsyncTask 运用的场景就是我们须要进行一些耗时的操作,耗时操作完毕后更新主线程,或者在操作过程中对主线程的UI进行更新。

缺陷:AsyncTask中维护着一个长度为128的线程池,同一时候能够运行5个工作线程。另一个缓冲队列。当线程池中已有128个线程,缓冲队列已满时。假设此时向线程提交任务,将会抛出RejectedExecutionException

解决:由一个控制线程来处理AsyncTask的调用推断线程池是否满了,假设满了则线程睡眠否则请求AsyncTask继续处理。

二次理解:

AsyncTaskHandler机制类似,即异步消息处理机制,所谓异步,即是并行运行的意思,而多线程就是相互独立"同一时候"在运行,而AsyncTask相对与Handler机制的不同,他对Handler进行了封装。在使用时不须要走Handler的流程,直接在doInBackground()定义相关方法,系统自己主动创建新子线程去运行。

AsyncTaskHandler机制是功能是相似的,能够用来在子线程中更新UI。或者下载耗时比較久的操作,比方下载文件由于主线程页面响应,即一个操作点击后,页面针对该操作做出反应的时间过长,那么Android系统会觉得是不友好的行为。该程序会直接崩溃。关于标准答案中AsyncTask能够维护一个长度为128的线程池,到底是什么概念,我能够理解为能够容纳128线程嘛?可是我能想象到的就是手机App眼下都是client本地的模式。即C/S模式。基本不存在一个用户在一个应用下打开128个下载内容,但不排除这个情况,然后运行5个工作线程的又是什么意思。与128是否冲突。还是概念上有什么区别(解答:所谓能够运行5个工作线程,长度为128的线程池,好比迅雷下载。你能够开启128个下载任务,可是它仅仅支持5个任务正在下载,超出的任务将进入缓冲区域,仅仅有当前5个任务完结时。才会从缓冲区域获取并运行新的任务,而超出128个任务时。则提示拒绝运行异常)。关于标准答案开辟一个子线程去监控线程缓冲区是否已满,假设满了。则让请求的线程的睡眠,可是睡眠的时间怎么设定?怎么实现线程缓冲区空暇时。睡眠停止的工作。

临时还没有找到更加具体资料。


使用介绍:因为本人对AsyncTask的使用没有很多其它的了解,所以这里会讲AsyncTask的简介和使用办法。

特点1AsyncTask,是一个抽象类,所以要使用必需要创建一个子类就继承它。

特点2AsyncTask,必须指定三个泛型參数,所谓类的泛型參数,能够觉得是java的一种指定类中所关联对象的详细类型做法,这样的做法的意义就是能够限制或者是明白了对象类型的范围,能够让编译器依据定义的泛型更加明白判定到内部代码的正确性。这是解决程序运行时多态转型异常的一种手段。參数列表为<Params,Progress,Result>,分别指传入的參数,能够在后台任务中使用;能够选择在前台显示当前进度。这里指定进度单位。任务运行完成后。假设须要对结果进行返回,则这里指定返回指的类型。

       最简单的形式:class DownloadTask extendsAsyncTask<Void,Integer,Boolean>,见第一行代码。Void表示不传入參数。

特点3AsyncTask方法介绍:

  1. onPreExecute:顾明思义,就是在运行子线程任务之前调用,不如我们要下载一个文件,能够在这里显示一个进度框。由于这是在主线程中运行的。

  2. doInBackground(Params):该方法就是后台操作的内容。系统会自己主动生成一个子线程去运行该方法中的内容,一些耗时的内容,比方下载一个文件的代码。该方法的參数和返回值类型和泛型中第二、三个相相应。由于这里是在子线程中运行的,假设须要对主线程中进度框更新,则须要调用publicProgress(Progress),该方法的调用自己主动引入第三个方法onProgressUpdate(Progress)的调用。
  3. onProgressUpdate(Progress),它的调用取决于doInBackground(Params)中是否调用了publicProgress(Progress)的方法,它的Progress參数也来源于publicProgress(Progress)中的參数,该方法事实上就是在主线中去更新进度。
  4. onPostExcuteResult(Result),该方法将在doInBackground(Params)运行完毕后运行,当中Result參数也来源于doInBackground方法的返回值,该方法能够运行对主线程的中UI的操作,比方关闭onPreExecute中创建的进度框,提示文件下载完毕等。

  5. excute(Params),在主线程创建这个AsyncTask子类的对象后,调用改方法,将启动异步运行的操作。


代码演示样例1

UIActivity.xml

<pre name="code" class="java">package com.noodles.uipractice;

import android.os.AsyncTask;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button;
import android.widget.ProgressBar;
import android.widget.TextView;
import android.widget.Toast;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.URL;
import java.net.URLConnection;

public class UIActivity extends AppCompatActivity {

    private TextView tv;

    private Button btn;

    private ProgressBar progressBar;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_ui);

        tv = (TextView) findViewById(R.id.tv);

        progressBar = (ProgressBar) findViewById(R.id.progress_bar);

        btn = (Button) findViewById(R.id.send_text);
        /**
         * 4、给Button定义个点击事件。就是启动AsyncTask
         * 当中execute的參数Params内容会自己主动传递给doInBackground的字符串数组类型的參数
         */
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                new MyAsyncTask().execute("http://www.baidu.com");
            }
        });
    }
    /**
     * 1、在主活动中中定义一个内部类MyAsyncTask继承AsyncTask
     * 当然也能够不以内部类的形式。比方单独定义一个类或者用匿名内部类
     * 不觉哪种定义方式更具有优势。由于单独定义分离后而代码结构更加清晰。

* 而匿名内部类和内部类更加贴近业务一体性。

*/ class MyAsyncTask extends AsyncTask<String, Integer, String> { /** * 2、复写doInBackground方法。该方法会开辟一个子线程去运行当中的代码 * 一般都是进行一些耗时的操作,这里能够去获取百度首页的内容 * <p/> * String... params形式的參数列表,能够看做String[] params */ @Override protected String doInBackground(String... params) { BufferedReader br = null; try { //首先把百度的首页地址www.baidu.com。封装成URL,就比方在浏览器中用键盘输入www.baidu.com URL url = new URL(params[0]); //调用URL的openConnnection方法。就好比在浏览器的点击了回车 URLConnection connection = url.openConnection(); //假设没有以下这行话,则默认是调用GZIP的压缩技术。getContentLength()返回的就是-1 connection.setRequestProperty("Accept-Encoding", "identity"); //connection.connect(); long Total = connection.getContentLength(); /* 建立连接以后,形成了socket通道,以IO流的方式形成数据的流转,所谓IO流能够看作是计算机 * 全部媒介进行数据流转的一种固定形式,仅仅是刚好起了一个名字叫IO流。而我们建立连接后 * 默认是以get的方式想百度server请求数据,能够请求返回一个字节输入流,即InputStream,而我们知道,返回 * 的内容都是字符串,全部能够通过InputStreamReader,把字节流转换成字符流,由于java为字符流提供的方式 * 更加易用,同一时候java提供一个缓存区的BufferReader,本质就是数组,能够把单独的字符用大的容器装载后 * 再进行传输,这样降低了传输的回合。提高效率 * */ //调用connection的getInputStream能够获取server返回的字符流 InputStream inputStream = connection.getInputStream(); //调用InputStreamReader对象,能够把字节流转换为字符流 InputStreamReader isr = new InputStreamReader(inputStream); //用BufferReader去装载字符流 br = new BufferedReader(isr); //调用BufferedReader的方法,更便捷的获取请求的内容 //定义一个字符串变量用来接收BufferedReader的readLine方法读取的每一行数据 String line; StringBuffer stringBuffer = new StringBuffer(); int count = 0; while ((line = br.readLine()) != null) { //读取的每一行数据都放在stringBuffer中 stringBuffer.append(line); /** * 5、对UI中已经定义的进度条进行更新,直接调用publishProgress方法 * 可是里面的參数类型必须和AsyncTask中声明的泛型一致,即Float */ count += line.length(); Integer value = (int) (count / (float) Total * 100); publishProgress(value); Thread.sleep(1000); } return stringBuffer.toString(); } catch (Exception e) { e.printStackTrace(); } finally { try { if (br != null) { br.close(); } } catch (IOException e) { e.printStackTrace(); } } //假设运行期间发生异常,则返回null return null; } /** * 3、复写onPostExecute,当中它的參数值来源于doInBackground的返回值 * 该方法能够更新主线程的UI,把返回的内容直接显示在主线程的TextView上 */ @Override protected void onPostExecute(String s) { tv.setText(s); Toast.makeText(UIActivity.this, "数据下载结束", Toast.LENGTH_SHORT).show(); } @Override protected void onProgressUpdate(Integer... values) { progressBar.setProgress(values[0]); } } }




AndroidManifest.xml

<?

xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.noodles.uipractice" > <uses-permission android:name="android.permission.INTERNET" /> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:supportsRtl="true" android:theme="@style/AppTheme" > <activity android:name=".UIActivity" > <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> </manifest>



点击前:

 

点击后:





总结:AsyncTask是Android提供还有一种异步信息处理机制,明白的说就是多线程间的通讯,有别于Handler机制地方,它是Handler的封装所以在使用上面

更加的便捷,首先我们须要继承该类,复写里面的方法,必须复写就是doInBackground(Params)。该方法会创建一个子线程去运行方法中代码,所以一般用来

运行耗时的操作。比方下载文件。然后我们须要创建该子类的对象,并调用execute(Params)方法,当中參数Params由外部传入,该參数会自己主动传入oInBackground(Params)的參数,而且在AsyncTask的第一个泛型參数Params指定它的类型。第二泛型參数progress是指AsyncTask在doInBackground(Params)方法中能够调用publishProgress(progress),须要指定详细类型的值progress。系统会转而调用onProgressUpdate(progress)同一时候传入该数值,这种方法能够对主线程的UI进行更新,通常是用作更新进度条。

第三个參数Result,是指doInBackground(Params)方法运行后须要返回一个值,接着系统会调用onPostExecute(Result)。并把返回指传入这种方法,这种方法是运行一些收尾工作,把信息反馈给用户,比方更新主线程的UI,告诉用户下载已经完毕等。



參考内容:

见:【极客学院】-【网络通讯】-【异步任务处理】-【AsyncTask的用法】。

见:http://blog.csdn.net/liuhe688/article/details/6532519


 



posted @ 2017-07-05 12:46  gccbuaa  阅读(229)  评论(0编辑  收藏  举报