溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

怎么Android應(yīng)用中實(shí)現(xiàn)一個(gè)APP顯示界面滑動(dòng)優(yōu)化

發(fā)布時(shí)間:2020-11-30 16:43:23 來源:億速云 閱讀:241 作者:Leah 欄目:移動(dòng)開發(fā)

這篇文章將為大家詳細(xì)講解有關(guān)怎么Android應(yīng)用中實(shí)現(xiàn)一個(gè)APP顯示界面滑動(dòng)優(yōu)化,文章內(nèi)容質(zhì)量較高,因此小編分享給大家做個(gè)參考,希望大家閱讀完這篇文章后對相關(guān)知識有一定的了解。

1、界面布局

<?xml version="1.0" encoding="utf-8"?> 
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 
 android:orientation="horizontal" android:layout_width="match_parent" 
 android:layout_height="wrap_content" 
 android:gravity="center"> 
 <ImageView 
  android:id="@+id/image" 
  android:src="@mipmap/ic_launcher" 
  android:layout_width="60dp" 
  android:layout_height="60dp" /> 
 <LinearLayout 
  android:orientation="vertical" 
  android:layout_marginLeft="10dp" 
  android:layout_width="match_parent" 
  android:layout_height="wrap_content"> 
  <TextView 
   android:id="@+id/title_tv" 
   android:text="TITLE" 
   android:textSize="15dp" 
   android:maxLines="1" 
   android:layout_width="match_parent" 
   android:layout_height="wrap_content" /> 
  <TextView 
   android:id="@+id/content_tv" 
   android:text="CONTENT" 
   android:textSize="10dp" 
   android:maxLines="3" 
   android:layout_width="match_parent" 
   android:layout_height="wrap_content" /> 
 </LinearLayout> 
</LinearLayout>
<?xml version="1.0" encoding="utf-8"?> 
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" 
 xmlns:tools="http://schemas.android.com/tools" 
 android:id="@+id/activity_main" 
 android:layout_width="match_parent" 
 android:layout_height="match_parent" 
 android:paddingBottom="@dimen/activity_vertical_margin" 
android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" 
tools:context="com.example.leixiansheng.news.MainActivity"> 
 <ListView 
  android:id="@+id/list_view" 
  android:layout_width="match_parent" 
  android:layout_height="match_parent"> 
 </ListView> 
</RelativeLayout>

2、開啟異步解析數(shù)據(jù)

package com.example.leixiansheng.news; 
 
/** 
 * Created by Leixiansheng on 2017/3/21. 
 */ 
public class NewsBean { 
 public String viewUrl; 
 public String title; 
 public String content; 
} 
package com.example.leixiansheng.news; 
import android.os.AsyncTask; 
import android.support.v7.app.AppCompatActivity; 
import android.os.Bundle; 
import android.util.Log; 
import android.widget.ListView; 
import org.json.JSONArray; 
import org.json.JSONException; 
import org.json.JSONObject; 
import java.io.BufferedReader; 
import java.io.IOException; 
import java.io.InputStream; 
import java.io.InputStreamReader; 
import java.io.UnsupportedEncodingException; 
import java.net.URL; 
import java.util.ArrayList; 
import java.util.List; 
public class MainActivity extends AppCompatActivity { 
 private ListView listView; 
 private static String URL = "http://www.imooc.com/api/teacher?type=4&num=30"; 
 @Override 
 protected void onCreate(Bundle savedInstanceState) { 
  super.onCreate(savedInstanceState); 
  setContentView(R.layout.activity_main); 
  listView = (ListView) findViewById(R.id.list_view); 
  new NewsAsyncTask().execute(URL); 
 } 
 //*&*異步加載,處理耗時(shí)任務(wù),UI更新 
 class NewsAsyncTask extends AsyncTask<String, Void, List<NewsBean>> { 
  @Override 
  protected List<NewsBean> doInBackground(String... strings) { 
   return getJsonData(strings[0]); 
  } 
  @Override 
  protected void onPostExecute(List<NewsBean> newsBeen) { 
   super.onPostExecute(newsBeen); 
   NewsAdapter adapter = new NewsAdapter(MainActivity.this, newsBeen,listView); 
   listView.setAdapter(adapter); 
  } 
 } 
 //*&*JSON解析網(wǎng)頁獲取數(shù)據(jù) 
 private List<NewsBean> getJsonData(String url) { 
  List<NewsBean> newsBeanList = new ArrayList<>(); 
  try { 
   String jsonString = readSteam(new URL(url).openStream()); 
   Log.i("DATA", jsonString); 
   JSONObject jsonObject; 
   NewsBean newsBean; 
   try { 
    jsonObject = new JSONObject(jsonString); 
    JSONArray jsonArray = jsonObject.getJSONArray("data"); 
    for (int i = 0; i < jsonArray.length(); i++) { 
     jsonObject = jsonArray.getJSONObject(i); 
     newsBean = new NewsBean(); 
     newsBean.content = jsonObject.getString("description"); 
     newsBean.title = jsonObject.getString("name"); 
     newsBean.viewUrl = jsonObject.getString("picSmall"); 
     newsBeanList.add(newsBean); 
    } 
   } catch (JSONException e) { 
    e.printStackTrace(); 
   } 
  } catch (IOException e) { 
   e.printStackTrace(); 
  } 
  return newsBeanList; 
 } 
 //*&*讀取數(shù)據(jù)流 
 private String readSteam(InputStream is) { 
  InputStreamReader isr; 
  String result = ""; 
  try { 
   String line = ""; 
   isr = new InputStreamReader(is, "utf-8"); 
   BufferedReader br = new BufferedReader(isr); 
   try { 
    while ((line = br.readLine()) != null) { 
     result += line; 
    } 
   } catch (IOException e) { 
    e.printStackTrace(); 
   } 
  } catch (UnsupportedEncodingException e) { 
   e.printStackTrace(); 
  } 
  return result; 
 } 
}

3、自定義適配器(在此處設(shè)置滑動(dòng)監(jiān)聽,以此來判斷什么時(shí)候加載資源)

package com.example.leixiansheng.news; 
import android.content.Context; 
import android.view.LayoutInflater; 
import android.view.View; 
import android.view.ViewGroup; 
import android.widget.AbsListView; 
import android.widget.BaseAdapter; 
import android.widget.ImageView; 
import android.widget.ListView; 
import android.widget.TextView; 
import java.util.List; 
/** 
 * Created by Leixiansheng on 2017/3/21. 
 */ 
public class NewsAdapter extends BaseAdapter implements AbsListView.OnScrollListener{ 
 private List<NewsBean> newsBeanList; 
 private LayoutInflater inflater; 
 private ImageLoader imageLoader; //圖片加載 
 private int start;  //第一個(gè)元素 
 private int end;  //最后一個(gè)元素 
 private boolean isFirstIn;  //是否第一次進(jìn)入 
 public static String[] URLS;  //所有資源 
 public NewsAdapter(Context context, List<NewsBean> newsBeanList, ListView listView) { 
  this.newsBeanList = newsBeanList; 
  inflater = LayoutInflater.from(context); 
  imageLoader = new ImageLoader(listView); 
  URLS = new String[newsBeanList.size()]; 
  for (int i = 0; i < newsBeanList.size(); i++) { 
   URLS[i] = newsBeanList.get(i).viewUrl; 
  } 
  isFirstIn = true; 
  listView.setOnScrollListener(this); 
 } 
 @Override 
 public int getCount() { 
  return newsBeanList.size(); 
 } 
 @Override 
 public Object getItem(int i) { 
  return newsBeanList.get(i); 
 } 
 @Override 
 public long getItemId(int i) { 
  return i; 
 } 
 @Override 
 public View getView(int i, View view, ViewGroup viewGroup) { 
  ViewHolder viewHolder = null; 
  if (view == null) { 
   viewHolder = new ViewHolder(); 
   view = inflater.inflate(R.layout.item, null); 
   viewHolder.imageView = (ImageView) view.findViewById(R.id.image); 
   viewHolder.title = (TextView) view.findViewById(R.id.title_tv); 
   viewHolder.content = (TextView) view.findViewById(R.id.content_tv); 
   view.setTag(viewHolder); 
  } else { 
   viewHolder = (ViewHolder) view.getTag(); 
  } 
  String url = newsBeanList.get(i).viewUrl; 
  viewHolder.imageView.setImageResource(R.mipmap.ic_launcher); 
  //*&*設(shè)置標(biāo)簽,避免快速滑動(dòng)listview出現(xiàn)位置誤差 
  viewHolder.imageView.setTag(url); 
//  new ImageLoader().showImageByThread(viewHolder.imageView, url); 
  imageLoader.showImageViewByAsyncTask(viewHolder.imageView,url); 
  viewHolder.title.setText(newsBeanList.get(i).title); 
  viewHolder.content.setText(newsBeanList.get(i).content); 
  return view; 
 } 
 //*&*優(yōu)化 
 class ViewHolder { 
  public TextView title; 
  public TextView content; 
  private ImageView imageView; 
 } 
 //滑動(dòng)監(jiān)聽 
 @Override 
 public void onScrollStateChanged(AbsListView absListView, int i) { 
  if (i == SCROLL_STATE_IDLE) { 
   //停止?fàn)顟B(tài):加載圖片 
   imageLoader.loadImages(start, end); 
  } else { 
   //滑動(dòng)狀態(tài):停止加載 
   imageLoader.cancelAllTasks(); 
  } 
 } 
 /** 
  * 
  * @param absListView 
  * @param i  第一個(gè)元素 
  * @param i1 元素?cái)?shù)量 
  * @param i2 
  */ 
 @Override 
 public void onScroll(AbsListView absListView, int i, int i1, int i2) { 
  start = i; 
  end = i + i1; 
  //第一次進(jìn)入需要手動(dòng)加載 
  if (isFirstIn && i1 > 0) { 
   imageLoader.loadImages(start, end); 
   isFirstIn = false; 
  } 
 } 
}
package com.example.leixiansheng.news; 
import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.os.AsyncTask; 
import android.os.Handler; 
import android.os.Message; 
import android.util.LruCache; 
import android.widget.ImageView; 
import android.widget.ListView; 
import java.io.BufferedInputStream; 
import java.io.IOException; 
import java.io.InputStream; 
import java.net.HttpURLConnection; 
import java.net.MalformedURLException; 
import java.net.URL; 
import java.util.HashSet; 
import java.util.Set; 
/** 
 * Created by Leixiansheng on 2017/3/21. 
 */ 
public class ImageLoader { 
 private ImageView mImageView; 
 private String mUrl; 
 //*&*創(chuàng)建緩存 
 private LruCache<String, Bitmap> lruCache; 
 private ListView listview; 
 private Set<NewsAsyncTask> mTask; 
 public ImageLoader(ListView listview) { 
  this.listview = listview; 
  mTask = new HashSet<>(); 
  //*&*獲取最大內(nèi)存 
  int maxMemory = (int) Runtime.getRuntime().maxMemory(); 
  //設(shè)置緩存大小 
  int lruCacheSize = maxMemory / 4; 
  lruCache = new LruCache<String, Bitmap>(lruCacheSize) { 
   @Override 
   protected int sizeOf(String key, Bitmap value) { 
    //獲取每個(gè)數(shù)據(jù)大小 
    return value.getByteCount(); 
   } 
  }; 
 } 
 //添加數(shù)據(jù)到緩存 
 public void addBitmapToLruCache(String url, Bitmap bitmap) { 
  if (getBitmapFromLruCache(url) == null) { 
   lruCache.put(url, bitmap); 
  } 
 } 
 //從緩存中獲取數(shù)據(jù) 
 public Bitmap getBitmapFromLruCache(String url) { 
  return lruCache.get(url); 
 } 
 private Handler handler = new Handler() { 
  @Override 
  public void handleMessage(Message msg) { 
   super.handleMessage(msg); 
   if (mImageView.getTag().equals(mUrl)) { 
    mImageView.setImageBitmap((Bitmap) msg.obj); 
   } 
  } 
 }; 
 public void showImageByThread(ImageView imageView, final String url) { 
  mImageView = imageView; 
  mUrl = url; 
  new Thread() { 
   @Override 
   public void run() { 
    super.run(); 
    Bitmap bitmap = getBitmapFromURL(url); 
    Message message = Message.obtain(); 
    message.obj = bitmap; 
    handler.sendMessage(message); 
   } 
  }.start(); 
 } 
 public Bitmap getBitmapFromURL(String urlString) { 
  Bitmap bitmap; 
  InputStream is = null; 
  try { 
   URL url = new URL(urlString); 
   HttpURLConnection connection = (HttpURLConnection) url.openConnection(); 
   is = new BufferedInputStream(connection.getInputStream()); 
   bitmap = BitmapFactory.decodeStream(is); 
   connection.disconnect(); 
   //模擬網(wǎng)速卡頓時(shí) 
//   try { 
//    Thread.sleep(1000); 
//   } catch (InterruptedException e) { 
//    e.printStackTrace(); 
//   } 
   return bitmap; 
  } catch (MalformedURLException e) { 
   e.printStackTrace(); 
  } catch (IOException e) { 
   e.printStackTrace(); 
  } finally { 
   try { 
    is.close(); 
   } catch (IOException e) { 
    e.printStackTrace(); 
   } 
  } 
  return null; 
 } 
 public void showImageViewByAsyncTask(ImageView imageView, String url) { 
  //判斷是否已經(jīng)緩存 
  Bitmap bitmap = getBitmapFromLruCache(url); 
  //沒有緩存則從新下載 
  if (bitmap == null) { 
   imageView.setImageResource(R.mipmap.ic_launcher); 
  } else { 
   imageView.setImageBitmap(bitmap); 
  } 
 } 
 //加載從start到end的所有圖片 
 public void loadImages(int start, int end) { 
  for (int i = start; i < end; i++) { 
   String url = NewsAdapter.URLS[i]; 
   //判斷是否已經(jīng)緩存 
   Bitmap bitmap = getBitmapFromLruCache(url); 
   //沒有緩存則從新下載 
   if (bitmap == null) { 
    NewsAsyncTask task = new NewsAsyncTask(url); 
    task.execute(url); 
    mTask.add(task); 
   } else { 
    ImageView imageView = (ImageView) listview.findViewWithTag(url); 
    imageView.setImageBitmap(bitmap); 
   } 
  } 
 } 
 public void cancelAllTasks() { 
  if (mTask != null) { 
   for (NewsAsyncTask task : mTask) { 
    task.cancel(false); 
   } 
  } 
 } 
 private class NewsAsyncTask extends AsyncTask<String, Void, Bitmap> { 
//  private ImageView imageView; 
  private String url; 
  public NewsAsyncTask(String url) { 
//   this.imageView = imageView; 
   this.url = url; 
  } 
  @Override 
  protected Bitmap doInBackground(String... strings) { 
   String url = strings[0]; 
   //從網(wǎng)絡(luò)獲取圖片 
   Bitmap bitmap = getBitmapFromURL(url); 
   if (bitmap != null) { 
    //將不在緩存中的圖片加入到緩存 
    addBitmapToLruCache(url, bitmap); 
   } 
   return bitmap; 
  } 
  @Override 
  protected void onPostExecute(Bitmap bitmap) { 
   super.onPostExecute(bitmap); 
   ImageView imageView = (ImageView) listview.findViewWithTag(url); 
   if (imageView != null && bitmap != null) { 
    imageView.setImageBitmap(bitmap); 
   } 
   mTask.remove(this); 
  } 
 } 
}

4、注冊聲明權(quán)限

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
 package="com.example.leixiansheng.news"> 
 <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=".MainActivity"> 
   <intent-filter> 
    <action android:name="android.intent.action.MAIN" /> 
    <category android:name="android.intent.category.LAUNCHER" /> 
   </intent-filter> 
  </activity> 
 </application> 
</manifest>

關(guān)于怎么Android應(yīng)用中實(shí)現(xiàn)一個(gè)APP顯示界面滑動(dòng)優(yōu)化就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI