android 消息机制与仿新闻客户端

效果图如下:

具体步骤如下:

1 布局文件中控件的设计

2 访问远程服务器的资源xml文件,该文件包含新闻的内容等信息

3 访问到内容后把访问内容显示到页面上

具体代码如下:

1 MainActivity

package com.yuanlp.newsclient;

import android.graphics.Color;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.Toast;

import com.yuanlp.newsclient.bean.NewsBean;
import com.yuanlp.newsclient.utils.XMLToBean;
import com.yuanlp.newsclient.view.SmartImageView;

import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.List;

public class MainActivity extends AppCompatActivity {

    private static final int LOAD_ERROR =2 ;
    private static final int LOAD_SUCCESS =1 ;
    private ListView mLv_news;
    private LinearLayout mLoading;
    //new 一个Handler来做android的消息机制,处理子线程数据
    private Handler handler=new Handler(){
        @Override
        public void handleMessage(Message msg) {
            //不管加载成功失败,进度条应该隐藏
            mLoading.setVisibility(View.INVISIBLE);
            switch (msg.what){
                case LOAD_ERROR:
                    Toast.makeText(MainActivity.this, "加载失败", Toast.LENGTH_SHORT).show();
                    break;
                case LOAD_SUCCESS:
                    //显示listView的内容
                    mLv_news.setAdapter(new MyNewsAdapter());
                    break;
            }
        }
    };
    private List<NewsBean> mList;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        /**
         * 初始化获取界面元素
         */
        initView();

        /**
         * 加载远程资源到本地文件
         */
        loadMessage();

        //readMessage();
    }

    /**
     * 初始化界面
     */
    private void initView() {
        setContentView(R.layout.activity_main);
        mLv_news = (ListView) findViewById(R.id.lv_news);
        mLoading = (LinearLayout) findViewById(R.id.ll_loading);
        //打开客户端后,设置ll_loading可见
        mLoading.setVisibility(View.VISIBLE);
    }

    /**
     * 加载远程资源到本地
     */
    private void loadMessage() {
        NewsBean newsBean=null;

        //访问网络不能再主线程中进行,需要一个新线程
        new Thread(){
            @Override
            public void run() {

                try {
                    Thread.sleep(5000);
                    NewsBean newsBean=null;
                    URL url = new URL("http://192.168.1.107:8080/WebServer/news.xml");
                    HttpURLConnection conn= (HttpURLConnection) url.openConnection();
                    conn.setRequestMethod("GET");
                    int code = conn.getResponseCode();
                    if (code==200){
                        //获取到资源
                        InputStream is = conn.getInputStream();
                        //调用utils方法,将xml文件转为list类型返回
                        mList = XMLToBean.readStream(newsBean,is);

                        //利用android的循环消息机制,放到UI线程中执行结果
                        Message msg = Message.obtain();
                        msg.what=LOAD_SUCCESS;
                        handler.sendMessage(msg);

                    }else{
                        Message msg = Message.obtain();
                        msg.what=LOAD_ERROR;
                        handler.sendMessage(msg);
                    }

                } catch (Exception e) {
                    e.printStackTrace();
                    Message msg = Message.obtain();
                    msg.what=LOAD_ERROR;
                    handler.sendMessage(msg);
                }
            }
        }.start();
    }

    /**
     * 显示listView中的数据
     */
    private class MyNewsAdapter extends BaseAdapter {
        @Override
        public int getCount() {
            return mList.size();
        }

        @Override
        public View getView(int position, View convertView, ViewGroup parent) {
            //布局打气筒,把xml资源文件转变为view     即加载另外的一个布局文件
            View view = View.inflate(MainActivity.this, R.layout.news, null);
            //获取子布局文件中的各个控件
            SmartImageView img = (SmartImageView) view.findViewById(R.id.iv_img);
            TextView tv_title= (TextView) view.findViewById(R.id.tv_title);
            TextView tv_desc= (TextView) view.findViewById(R.id.tv_desc);
            TextView tv_type= (TextView) view.findViewById(R.id.tv_type);

            //获取当前位置的对象
            NewsBean item = getItem(position);

            //设置图片显示
            img.showImgByPath(item.getImg());

            tv_title.setText(item.getTitle());
            tv_desc.setText(item.getDescription());

            String type=item.getType();
            if ("1".equals(type)){ //当类型为1时,表示有评论,而且显示评论数
                tv_type.setText("评论:"+item.getComment());
                tv_type.setTextColor(Color.BLACK);
                tv_type.setBackgroundColor(Color.TRANSPARENT);
            }else if ("2".equals(type)){
                tv_type.setText("专题");
                tv_type.setBackgroundColor(Color.RED);
                tv_type.setTextColor(Color.WHITE);
            }else if ("3".equals(type)){
                tv_type.setText("直播");
                tv_type.setBackgroundColor(Color.RED);
                tv_type.setTextColor(Color.WHITE);
            }

            return view;
        }

        @Override
        public NewsBean getItem(int position) {
            return mList.get(position);
        }

        @Override
        public long getItemId(int position) {
            return position;
        }

    }
}

2 utils 中的将xml转为list,可以参考前面几天写的博客,android解析xml。

package com.yuanlp.newsclient.utils;

import android.util.Xml;

import com.yuanlp.newsclient.bean.NewsBean;

import org.xmlpull.v1.XmlPullParser;

import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;

/**
 * Created by 原立鹏 on 2017/6/21.
 */

public class XMLToBean {

    public static List<NewsBean> readStream(NewsBean newsBean, InputStream is){
        List<NewsBean> list=null;
        try {

            XmlPullParser xmlPullParser = Xml.newPullParser();
            xmlPullParser.setInput(is,"utf-8");

            int event=xmlPullParser.getEventType();
            while(event!=XmlPullParser.END_DOCUMENT){
                switch (event){  //根据eventType来区分,分为START_DOCUMENT,START_TAG,END_TAG,END_DOCUMENT
                    case XmlPullParser.START_DOCUMENT:
                        list=new ArrayList<NewsBean>();
                        break;
                    case XmlPullParser.START_TAG:
                        String tagName=xmlPullParser.getName();  //获取标签名称
                        if (tagName.equalsIgnoreCase("item")){
                            newsBean=new NewsBean();
                        }else if (tagName.equalsIgnoreCase("title")){
                            newsBean.setTitle(xmlPullParser.nextText());
                        }else if (tagName.equalsIgnoreCase("description")){
                            newsBean.setDescription(xmlPullParser.nextText());
                        }else if (tagName.equalsIgnoreCase("image")){
                            newsBean.setImg(xmlPullParser.nextText());
                        }else if (tagName.equalsIgnoreCase("type")){
                            newsBean.setType(xmlPullParser.nextText());
                        }else if (tagName.equalsIgnoreCase("comment")){
                            newsBean.setComment(xmlPullParser.nextText());
                        }
                        break;
                    case XmlPullParser.END_TAG:
                        if (xmlPullParser.getName().equalsIgnoreCase("item")&&newsBean!=null){
                            list.add(newsBean);
                            newsBean=null;

                        }

                }
                event=xmlPullParser.next();
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        return list;
    }
}

3 获取远程新闻的方法

package com.yuanlp.newsclient.view;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Handler;
import android.os.Message;
import android.support.annotation.Nullable;
import android.util.AttributeSet;
import android.widget.ImageView;

import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;

/**
 * Created by 原立鹏 on 2017/6/21.
 */

/**
 * 根据url地址去获取远程图片,并显示到新闻标题的左侧
 */
public class SmartImageView extends ImageView {
    private Handler handler=new Handler(){
        @Override
        public void handleMessage(Message msg) {
            Bitmap bitmap= (Bitmap) msg.obj;
            setImageBitmap(bitmap);
        }
    };

    public SmartImageView(Context context) {
        super(context);
    }

    public SmartImageView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public SmartImageView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    public  void showImgByPath(final String imgURL){
        new Thread(){
            @Override
            public void run() {
                try {
                    URL url = new URL(imgURL);
                    HttpURLConnection conn = (HttpURLConnection) url.openConnection();
                    conn.setRequestMethod("GET");
                    int code = conn.getResponseCode();
                    if (code==200){
                        InputStream is = conn.getInputStream();
                        Bitmap bitmap = BitmapFactory.decodeStream(is);
                        Message msg = Message.obtain();
                        msg.what=1;
                        msg.obj=bitmap;
                        handler.sendMessage(msg);
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }

            }
        }.start();
    }
}

4 activity_main.xml

<?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:layout_width="match_parent"
                android:layout_height="match_parent">

    <LinearLayout
        android:id="@+id/ll_loading"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:gravity="center"
        android:orientation="vertical"
        android:visibility="invisible">

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:text="TextView"
            tools:text="正在拼命加载"/>

        <ProgressBar
            android:id="@+id/progressBar"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"/>
    </LinearLayout>

    <ListView
        android:id="@+id/lv_news"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>
</RelativeLayout>

5 news.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
                xmlns:app="http://schemas.android.com/apk/res-auto"
                xmlns:tools="http://schemas.android.com/tools"
                android:layout_width="match_parent"
                android:layout_height="match_parent">

    <com.yuanlp.newsclient.view.SmartImageView
        android:id="@+id/iv_img"
        android:layout_width="72dp"
        android:layout_height="50dp"
        android:layout_alignParentLeft="true"
        android:layout_alignParentStart="true"
        android:layout_alignParentTop="true"
        android:layout_marginLeft="3dp"

        app:srcCompat="@mipmap/ic_launcher"/>

    <TextView
        android:id="@+id/tv_title"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"

        android:layout_marginLeft="3dp"
        android:layout_marginTop="5dp"
        android:layout_toRightOf="@+id/iv_img"
        android:text="新闻标题"/>

    <TextView
        android:id="@+id/tv_desc"
        android:layout_width="300dp"
        android:layout_height="wrap_content"
        android:layout_below="@+id/tv_title"
        android:layout_toRightOf="@+id/iv_img"
        android:ellipsize="end"
        android:lines="2"
        android:text="TextView"
        android:textColor="#99000000"
        android:textSize="12sp"
        tools:text="新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述新闻描述"
        />

    <TextView
        android:id="@+id/tv_type"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="新闻类型"
        android:layout_below="@+id/tv_desc"
        android:layout_alignParentRight="true"
        android:textSize="10sp"
        android:textColor="#99000000"/>

</RelativeLayout>
时间: 2024-11-04 20:10:15

android 消息机制与仿新闻客户端的相关文章

Android消息机制Handler的实现原理解析

Android的主线程为什么可以一直存在? 线程是一个动态执行的过程,从产生到死亡包括五个状态:新建.就绪.运行.死亡和堵塞.只要线程没有执行完毕或者没有被其它线程杀死,线程就不会进入死亡状态.Android中的主线程一直存在是因为主线程中一直在监听消息,从而使线程无法被执行完毕. 线程的五种状态: 新建new Thread 当创建Thread类的一个实例对象时,此线程进入新建状态未被启动. 就绪runnable 线程已经被启动,正在等待被分配给CPU时间片,也就是说此时线程正在就绪队列中排队等

【转】Android 消息机制

Android 消息机制 本文首先从整体架构分析了Android整个线程间消息传递机制,然后从源码角度介绍了各个组件的作用和完成的任务.文中并未对基础概念进行介绍,关于threadLacal和垃圾回收等等机制请自行研究. 基础架构 首先,我们需要从整体架构上了解一下Android线程通信都做了哪些工作.我们都知道,进程是操作系统分配资源的最小单位,一个进程中可以启动多个线程来执行任务,这些线程可以共享进程的资源但不分配资源,这里讲的资源主要是只内存资源.Android的线程间消息传递机制其实和我

Android消息机制字典型探

Android消息机制字典型探究(一) Android消息机制字典型探究(二) 带着这篇去通关所有Handler的提问(三)

Android消息机制:Looper,MessageQueue,Message与handler

Android消息机制好多人都讲过,但是自己去翻源码的时候才能明白. 今天试着讲一下,因为目标是讲清楚整体逻辑,所以不追究细节. Message是消息机制的核心,所以从Message讲起. 1.Message是什么? 看一个从消息池中取出一个msg的方法: public static Message obtain(Handler h, int what, int arg1, int arg2, Object obj) { Message m = obtain(); m.target = h; m

Android消息机制探索(Handler,Looper,Message,MessageQueue)

概览 Android消息机制是Android操作系统中比较重要的一块.具体使用方法在这里不再阐述,可以参考Android的官方开发文档. 消息机制的主要用途有两方面: 1.线程之间的通信.比如在子线程中想更新UI,就通过发送更新消息到UI线程中来实现. 2.任务延迟执行.比如30秒后执行刷新任务等. 消息机制运行的大概示意图如下: 一个线程中只能有一个Looper对象,一个Looper对象中持有一个消息队列,一个消息队列中维护多个消息对象,用一个Looper可以创建多个Handler对象,Han

Android消息机制1-Handler(Java层)(转)

转自:http://gityuan.com/2015/12/26/handler-message-framework/ 相关源码 framework/base/core/java/andorid/os/Handler.java framework/base/core/java/andorid/os/Looper.java framework/base/core/java/andorid/os/Message.java framework/base/core/java/andorid/os/Mes

Android 消息机制 (Handler、Message、Looper)

综合:http://blog.csdn.net/dadoneo/article/details/7667726 与 http://android.tgbus.com/Android/androidnews/201204/421642.shtml 一. 消息机制常用类的介绍和使用 在Android程序运行中,线程之间或者线程内部进行信息交互时经常会使用到消息,如果我们熟悉这些基础的东西及其内部的原理,将会使我们的Android开发变的容易.可以更好地架构系统.在学习Android消息机制之前,我们

Android消息机制Handler、Looper、MessageQueue源码分析

1. Handler Looper MessageQueue的关系 2.源码分析 下图表示了Handler.Looper.MessageQueue.Message这四个类之间的关系. Handler必须与一个Looper关联,相关Looper决定了该Handler会向哪个MessageQueue发送Message 每一个Looper中都包含一个MessageQueue Handler中的mQueue引用的就是与之关联的Looper的MessageQueue 不管Handler在哪个线程发送Mes

android消息机制原理详解

android消息机制原理详解 因为之前使用的是CSDN默认的文本编辑器,而且也因为懒得学用MarkDown来写博客,所以排版上有一些问题.就上一篇写的设计模式之抽象工厂模式提出了这个问题(一个android群的群友提出来的,没有在评论里评论),所以以后的文章都用MarkDown来写了. 好了,言归正传,这篇文章我来介绍一下android消息机制的原理 Android消息机制概述 说到Android的消息机制,Android初级工程师(不包括那些初学者)肯定会想到Handler.是的,Andro