以下内容为原创,转载时请注明链接地址:http://www.cnblogs.com/tiantianbyconan/p/3311658.html

这是我现在碰到的一个问题,如果需要在TextView中加载大文本的时候,比如几M的txt文件时,TextView载入的时候会出现卡死的现象,甚至会出现异常等待退出出现。

解决办法之一就是通过“分段”或“分页”来显示数据,在TextView(嵌入在ScrollView之中实现了TextView的滚动)中滚动到底部的时候,再去加载下一部分的数据,依次类推,这样每次加载的数据相对来说都比较小,不会出现卡顿的现象。

遇到的问题是,如何监听ScrollView滚动的位置(滚动到顶部还是底部?)。

如下,通过自定义ScrollView类(BorderScrollView):

 package com.wangjie.bigtextloadtest;

 import android.content.Context;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.widget.ScrollView; /**
* Created with IntelliJ IDEA.
* Author: wangjie email:tiantian.china.2@gmail.com
* Date: 13-9-6
* Time: 下午2:06
*/
public class BorderScrollView extends ScrollView{
private long millis;
// 滚动监听者
private OnScrollChangedListener onScrollChangedListener; public BorderScrollView(Context context) {
super(context);
} public BorderScrollView(Context context, AttributeSet attrs) {
super(context, attrs);
} public BorderScrollView(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
} @Override
protected void onScrollChanged(int l, int t, int oldl, int oldt) {
super.onScrollChanged(l, t, oldl, oldt); if(null == onScrollChangedListener){
return;
} long now = System.currentTimeMillis(); // 通知监听者当前滚动的具体信息
onScrollChangedListener.onScrollChanged(l, t, oldl, oldt); if(now - millis > 1000l){
// 滚动到底部(前提:从不是底部滚动到底部)
if((this.getHeight() + oldt) != this.getTotalVerticalScrollRange()
&& (this.getHeight() + t) == this.getTotalVerticalScrollRange()){ onScrollChangedListener.onScrollBottom(); // 通知监听者滚动到底部 millis = now;
}
} // 滚动到顶部(前提:从不是顶部滚动到顶部)
if(oldt != 0 && t == 0){
onScrollChangedListener.onScrollTop(); // 通知监听者滚动到顶部
} } public OnScrollChangedListener getOnScrollChangedListener() {
return onScrollChangedListener;
} public void setOnScrollChangedListener(OnScrollChangedListener onScrollChangedListener) {
this.onScrollChangedListener = onScrollChangedListener;
} /**
* 获得滚动总长度
* @return
*/
public int getTotalVerticalScrollRange() {
return this.computeVerticalScrollRange();
} @Override
protected int computeScrollDeltaToGetChildRectOnScreen(Rect rect) {
return 0; // 禁止ScrollView在子控件的布局改变时自动滚动
} }

滚动监听器接口OnScrollChangedListener:

 package com.wangjie.bigtextloadtest;

 /**
* Created with IntelliJ IDEA.
* Author: wangjie email:tiantian.china.2@gmail.com
* Date: 13-9-6
* Time: 下午2:53
*/
public interface OnScrollChangedListener {
/**
* 监听滚动变化
* @param l
* @param t
* @param oldl
* @param oldt
*/
public void onScrollChanged(int l, int t, int oldl, int oldt); /**
* 监听滚动到顶部
*/
public void onScrollTop(); /**
* 监听滚动到底部
*/
public void onScrollBottom(); }

滚动监听器的空实现OnScrollChangedListenerSimple(简洁真正用时候的代码):

 package com.wangjie.bigtextloadtest;

 /**
* Created with IntelliJ IDEA.
* Author: wangjie email:tiantian.china.2@gmail.com
* Date: 13-9-9
* Time: 下午2:39
*/
public class OnScrollChangedListenerSimple implements OnScrollChangedListener{
@Override
public void onScrollChanged(int l, int t, int oldl, int oldt) {} @Override
public void onScrollTop() {} @Override
public void onScrollBottom() {}
}

异步加载分段文本(这里每次加载50行):

 package com.wangjie.bigtextloadtest;

 import android.content.Context;
import android.os.AsyncTask;
import android.os.Handler;
import android.view.View; import java.io.*; /**
* Created with IntelliJ IDEA.
* Author: wangjie email:tiantian.china.2@gmail.com
* Date: 13-9-6
* Time: 上午11:34
*/
public class AsyncTextLoadTask extends AsyncTask<Object, String, String> {
private Context context;
private MainActivity activity;
private BufferedReader br; public AsyncTextLoadTask(Context context, BufferedReader br) {
this.context = context;
this.br = br;
activity = (MainActivity)context;
} @Override
protected String doInBackground(Object... params) {
StringBuilder paragraph = new StringBuilder();
try { String line = ""; int index = 0;
while(index < 50 && (line = br.readLine()) != null){
paragraph.append(line + "\n");
index++;
} } catch (IOException e) {
e.printStackTrace();
} return paragraph.toString();
} @Override
protected void onPreExecute() {
super.onPreExecute();
} @Override
protected void onPostExecute(String result) {
super.onPostExecute(result);
activity.contentTv.setText(result);
new Handler().postDelayed(new Runnable() { @Override
public void run() {
activity.contentScroll.scrollTo(0, 0); // 记载完新数据后滚动到顶部
}
}, 100);
activity.isLoading = false;
} }

真正使用分段加载数据Activity(这里加载一本小说《百年孤独》):

 package com.wangjie.bigtextloadtest;

 import android.app.Activity;
import android.content.Context;
import android.os.Bundle;
import android.widget.TextView; import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader; public class MainActivity extends Activity {
public BorderScrollView contentScroll;
public TextView contentTv; private BufferedReader br; private Context context; public boolean isLoading; /**
* Called when the activity is first created.
*/
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
context = this; contentTv = (TextView) findViewById(R.id.content);
contentScroll = (BorderScrollView) findViewById(R.id.contentScroll); // 注册刚写的滚动监听器
contentScroll.setOnScrollChangedListener(new OnScrollChangedListenerSimple(){
@Override
public void onScrollBottom() {
synchronized (MainActivity.class){
if(!isLoading){
isLoading = true;
new AsyncTextLoadTask(context, br).execute();
}
}
}
}); try{
InputStream is = context.getAssets().open("bngd.txt");
br = new BufferedReader(new InputStreamReader(is)); new AsyncTextLoadTask(context, br).execute(); }catch(Exception ex){
ex.printStackTrace();
} } @Override
protected void onDestroy() {
super.onDestroy();
if(null != br){
try {
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
} }

贴上布局:

 <?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<com.wangjie.bigtextloadtest.BorderScrollView
android:id="@+id/contentScroll"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<LinearLayout
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="asdfasdf"
/>
<TextView
android:id="@+id/content"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="Hello World, MainActivity"
/>
</LinearLayout> </com.wangjie.bigtextloadtest.BorderScrollView> </LinearLayout>

Android自定义ScrollView分段加载大文本数据到TextView的更多相关文章

  1. Android 自定义ScrollView ListView 体验各种纵向滑动的需求

      分类: [android 进阶之路]2014-08-31 12:59 6190人阅读 评论(10) 收藏 举报 Android自定义ScrollView纵向拖动     转载请标明出处:http: ...

  2. android 自定义ScrollView实现背景图片伸缩(阻尼效果)

    android 自定义ScrollView实现强调内容背景图片伸缩(仿多米,qq空间背景的刷新) 看到一篇文章,自己更改了一下bug: 原文地址:http://www.aiuxian.com/arti ...

  3. JDBC 关于大文本数据

    大文本数据Clob,在不同的数据库中类型名不一致,有的是text格式,有的是clob,还有其他一些格式   package test; import java.io.BufferedReader; i ...

  4. android 自定义scrollview 仿QQ空间效果 下拉伸缩顶部图片,上拉回弹 上拉滚动顶部title 颜色渐变

    首先要知道  自定义scrollview 仿QQ效果 下拉伸缩放大顶部图片 的原理是监听ontouch事件,在MotionEvent.ACTION_MOVE事件时候,使用不同倍数的系数,重置布局位置[ ...

  5. Android自定义ScrollView实现一键置顶功能

    效果图如下: (ps:动态图有太大了,上传不了,就给大家口述一下要实现的功能吧) 要实现的功能:当ScrollView向上滑动超过一定距离后,就渐变的出现一个置顶的按钮,当滑动距离小于我们指定的距离时 ...

  6. pandas处理大文本数据

    当数据文件是百万级数据时,设置chunksize来分批次处理数据 案例:美国总统竞选时的数据分析 读取数据 import numpy as np import pandas as pdfrom pan ...

  7. Android 自定义ScrollView的滑动监听事件

    项目结构: 1.LazyScrollView类(自定义ScrollView) package android.zhh.com.myapplicationscrollview; /** * Create ...

  8. Android -- 自定义ScrollView实现放大回弹效果

    1,刚刚在别人开源的项目中看到了一个挺不错的用户体验,效果图如下: 2,那下面我们就来实现一下,首先看一下布局,由于一般只是我们包含头像的那部分方法,所以这里我们要把布局分成两部分,对应的布局文件效果 ...

  9. Android 自定义ScrollView(具有反弹效果的ScrollView,能够兼容横向的滑动)

    package com.itau.jingdong.widgets; import android.content.Context; import android.graphics.Rect; imp ...

随机推荐

  1. javascript学习—理解addLoadEvent函数

    onload事件是HTML DOM Event 对象的一个属性,又叫事件句柄(Event Handlers),它会在页面或图像加载完成后(注意是加载完成后)立即发生. window.onload = ...

  2. Azure China (12) 域名备案问题

    <Windows Azure Platform 系列文章目录> (1) 默认情况下,我们在创建的Azure 服务,默认使用的DNS地址为: http://xxx.chinacloudapi ...

  3. Windows Azure Cloud Service (39) 如何将现有Web应用迁移到Azure PaaS平台

    <Windows Azure Platform 系列文章目录> 本文将简单介绍,如何将企业内现有的ASP.NET应用程序迁移到Azure PaaS平台. 因为在迁移过程中,可能需要对现有的 ...

  4. 实现在GET请求下调用WCF服务时传递对象(复合类型)参数

    WCF实现RESETFUL架构很容易,说白了,就是使WCF能够响应HTTP请求并返回所需的资源,如果有人不知道如何实现WCF支持HTTP请求的,可参见我之前的文章<实现jquery.ajax及原 ...

  5. Data URL简介及Data URL的利弊

    之前写过一篇“漫谈前端优化”的文章,里面提到过DataUrl,粗鲁的描述了下,感觉不甚详焉,所以这几天也总结了这方面的知识,参考一些资料,补充一篇文章在这里,对这方面的资料来说,也是一种强化记忆应用: ...

  6. SQL--存储过程

    声明和调用有返回值的存储过程 分页存储过程 转账的存储过程:

  7. css异常

    1.0 样式的引用顺序不对会导致样式显示不正常.个人推荐:把别人写好的样式引用在最前面,自己写的放在后面. 2.0 UC手机浏览器 UC手机浏览器在识别到页面文字很多的情况下会自动放大字体优化阅读体验 ...

  8. ASP.NET Core 开发-Entity Framework (EF) Core 1.0 Database First

    ASP.NET Core 开发-Entity Framework Core 1.0 Database First,ASP.NET Core 1.0 EF Core操作数据库. Entity Frame ...

  9. 为什么我不建议你做APP?

    最近迷上了新产品的可行性分析和推演. 有几个朋友也准备跳入创业火坑了,找到我说帮忙做做产品分析和可行性讨论,欣然应允. 我一向厌恶纯凭感觉拍脑袋的方式,所以对于我不了解的行业,都会从行业背景.现状痛点 ...

  10. Web.config配置文件详解

    整理了一下ASP.NET Web.config配置文件的基本使用方法.很适合新手参看,由于Web.config在使用很灵活,可以自定义一些节点.所以这里只介绍一些比较常用的节点. <?xml v ...