黎活明8天快速掌握android视频教程25_网络通信之资讯客户端

1 该项目的主要功能是:后台通过xml或者json格式返回后台的视频资讯,然后Android客户端界面显示出来

首先后台新建立一个java web后台

采用mvc的框架

所以的servlet都放在servlet中,servlet想当于android中的controller层,android中的控制层下面可以在分为activity 、fragment和adapter 、serverce服务和broadcast广播

Service是业务层,按照面向接口编程的原则,控制层的数据不能直接和业务层的实现类打交道,只能和业务层的具体实现类打交道,这里service就是业务层的业务接口,定义了业务操作的接口

Service.impl

这里就是业务接口的具体实现类,在这里通过调研数据库model模型层的数据接口访问数据库,在这里业务实现类也不能直接访问数据量,而是通过model层提供的数据库接口才能操作数据库。

Model数据库模型层,里面又可以分为db、dao,model层主要通过单例;提供一个数据库访问的接口提供给业务层的实现类调用。

例如:在这里通过数据库的接口操作数据库 Model.getInstace().getVideoNewsDao(),通过这种方式通过model层提供一个接口让业务库实现类调用,最好不要使用直接new VideoNewsDao()的方式

bean

就是存储具体的java对象

我们来看下后台的代码:

package com.videonews.bean;

public class VideoNewsBean {
    private Integer videoId ;//电影id
    private String videoTitle ;// 电影标题
    private Integer timeLength ;//电影时间长度
    
    
    public VideoNewsBean() {
        super();
    }
    public VideoNewsBean(Integer videoId, String videoTitle, Integer timeLength) {
        this.videoId = videoId;
        this.videoTitle = videoTitle;
        this.timeLength = timeLength;
    }
    public Integer getVideoId() {
        return videoId;
    }
    public void setVideoId(Integer videoId) {
        this.videoId = videoId;
    }
    public String getVideoTitle() {
        return videoTitle;
    }
    public void setVideoTitle(String videoTitle) {
        this.videoTitle = videoTitle;
    }
    public Integer getTimeLength() {
        return timeLength;
    }
    public void setTimeLength(Integer timeLength) {
        this.timeLength = timeLength;
    }
    @Override
    public String toString() {
        return "VideoNewsBean [videoId=" + videoId + ", videoTitle="
                + videoTitle + ", timeLength=" + timeLength + "]";
    }
    
    
}

我们来看业务的实现接口:

package com.videonews.service;

import java.util.List;

import com.videonews.bean.VideoNewsBean;

public interface VideoNewsService {

    public List<VideoNewsBean> getAllNews();
}

我们来看业务接口的实现类:

public class VideoNewsServiceImp  implements VideoNewsService {

    @Override
    public List<VideoNewsBean> getAllNews() throws Exception{
        // TODO Auto-generated method stub
        
        //在这里通过数据库的接口操作数据库 Model.getInstace().getVideoNewsDao()
        
        //这里我们操作数据库,我们直接模拟数据
         List<VideoNewsBean> datas = new ArrayList<VideoNewsBean>();
         datas.add(new VideoNewsBean(1, "人民的利益", 100));
         datas.add(new VideoNewsBean(2, "三生三世", 100));
         datas.add(new VideoNewsBean(1, "光头强", 100));
        return datas;
    }

正常的业务操作中:业务实现类应该通过数据库接口操作数据库获得对象的数据,这里我们模拟该过程直接返回数据

我们来看看

ListServlet:

public class ListServlet extends HttpServlet {
     VideoNewsService service = new VideoNewsServiceImp(); //采用面向接口的编程方式,控制层只能和业务类的接口打交道,最好不要写成

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException

        doPost(request, response);
    }

    
    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {
        //判断是返回json,还是返回xml格式的数据
        String format = (String) request.getAttribute("format");
        if(format == null || "json".equals(format)){
              //进行业务操作获得数据
            try {
                List<VideoNewsBean> news = service.getAllNews();
                
                //封装json字符串数据
                //1 方法1 使用第三方的gson框架,这里使用Gson,需要gson的jar包
                Gson gson = new Gson();  
                String jsonString = gson.toJson(news);
                //方法2自己封装json字符串数据,在实际的开发过程中当然使用gson框架
                //[{"videoId":1,"videoTitle":"人民的利益","timeLength":100},{"videoId":2,"videoTitle":"三生三世","timeLength":100},
                //{"videoId":1,"videoTitle":"光头强","timeLength":100}]
                // "使用转义写成\",{直接写成"{"
                StringBuilder builder = new StringBuilder();
                builder.append("[");
                for(VideoNewsBean videonew:news){
                    builder.append("{").append("\"");
                    builder.append("videoId").append("\"").append(":").append(""+videonew.getVideoId()).append(";");
                    builder.append("videoTitle").append("\"").append(":").append(""+videonew.getVideoTitle()).append(";");
                    builder.append("timeLength").append("\"").append(":").append(""+videonew.getTimeLength()).append("}");
                    builder.append(",");
                
                }
                //遍历之后最后多了一个builder.append(",");要记得去掉
                builder.deleteCharAt(builder.length()-1);
                builder.append("]");
                String jsonString2 = builder.toString();
                
                request.setAttribute("json", jsonString);
                //重定向到jsp页面
                request.getRequestDispatcher("WEB-INF/page/json.jsp").forward(request, response);
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
                //业务操作失败,需要在界面jsp上面提示用户,或者对业务操作失败做出其他的处理
            }
        }else{
              //进行业务操作获得数据
            try {
                List<VideoNewsBean> news = service.getAllNews();
                request.setAttribute("videos", news);
                //重定向到jsp页面
                request.getRequestDispatcher("WEB-INF/page/videonews.jsp").forward(request, response);
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
                //业务操作失败,需要在界面jsp上面提示用户,或者对业务操作失败做出其他的处理
            }
            
        }
        
    }

}

上面有几个地方需要注意的:

控制层和业务层采用面向接口的编程方式,控制层只和业务的接口类打交道

VideoNewsService service = new VideoNewsServiceImp();

而不是具体和业务的实现类打交道,最好不需写成 VideoNewsServiceImp service = new VideoNewsServiceImp();

2、依据客户端请求的客户觉得是以xml的格式还是json的格式返回数据给客户端

3、同理业务库实现类也不能直接操作数据库,必须通过数据库的接口操作数据库Model.getInstance.getVideoNewsDao();通过model模型层提供一个操作数据库的接口访问数据库,这就是面向接口的编程的思想。这里模拟操作数据库的过程直接将数据返回业务类

4、在进行业务操作的时候getAllNews()如果这个业务操作有异常,应该将异常抛出,不应该内存try catch进行处理,而应该抛出给控制层servlet。Servlet已经异常才知道该业务操作是成功还是失败,在jsp或者html中进行显示。

 

5、我们来看看web-inf/page/videonews.jsp的代码:如果以xml的方法返回给客户端


<%@ page language="java" contentType="text/xml; charset=UTF-8" pageEncoding="UTF-8"%><%@taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %><?xml version="1.0" encoding="UTF-8"?>
<videonews><c:forEach items="${videos}" var ="video">
     <news id = "${video.videoId}">
      <title>${video.videoTitle}</title>
      <timelength>${video.timeLength}</timelength>
     </news></c:forEach>
</videonews>
 

上面使用了jsp的c标签进行遍历,需要在java web工程的libs目录下添加下面两个jar包,并且

contentType="text/xml;必须是xml格式

我们在浏览器上访问:

现在我们在浏览器访问http://localhost:8080/lihuoming_25/ListServlet

如果是以json格式返回,videonews.jsp的代码如下:

<%@ page language="java" contentType="text/plain; charset=utf-8"
    pageEncoding="utf-8"%>
${json}

我们来看看整个java web工程的程序框架是:

我们来看看Android端客户端的代码:

客户端也是按照后台一样采用mvc的框架

也是采用控制层、模型层、业务层、工具类类型的四种方式

控制层包括activity 、fargment 、receiver等

业务层:主要进行业务操作,例如从服务器上或者接送数据,业务层的异常最好不要使用try catch内部处理,而已经将异常抛出,在控制层处理异常,如果控制层可以通过异常知道业务操作是否成功,在界面上做出相应的显示

业务层最好写成业务接口类类和业务实现类:接口主要定义业务的具有方法,实现类主要实现该业务的具体操作

Model:主要操作数据库,封装数据库操作的接口提供给业务层调用例如Model.getInstance.getDbDao()采用单例的模式提供一个数据库接口给业务实现类调用,不清楚的可以看尚硅谷硅谷社交项目。

Bean:封装Javabean对象

Utils工具类主要工具类。

 我们首先来看业务的接口类:

package application.weiyuan.com.lihuoming_25.bussiness;

import org.json.JSONException;
import org.xmlpull.v1.XmlPullParserException;

import java.io.IOException;
import java.util.List;

import application.weiyuan.com.lihuoming_25.model.bean.VideoNewsBean;

/**
 * Created by wei.yuan on 2017/4/24.
 */

public interface VideoNewsService {
    public  List<VideoNewsBean> getAllNews(String path) throws IOException, XmlPullParserException, JSONException;
}

我们来看业务的实现类:

import android.util.Log;
import android.util.Xml;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLConnection;
import java.util.ArrayList;
import java.util.List;

import javax.xml.datatype.XMLGregorianCalendar;

import application.weiyuan.com.lihuoming_25.model.bean.VideoNewsBean;

/**
 * Created by Administrator on 2017/4/18.
 *
 XmlPullParser.START_DOCUMENT对应文档开始
 XmlPullParser.END_DOCUMENT
 <videnews>是根元素、有三个<News>子元素
 <标记名称 属性名1="属性值1" 属性名1="属性值1" ……>内容</标记名称>
 起始标签(外国语:starttag)表示一个特定区域的开始,例如<起始>;
 结束标签(外国语:end tag)定义了一个区域的结束,除了在小于号之后紧跟着一个斜线(/)外,和起始标签基本一样,例如</结束>;
 其中标记名称对应的事件是XmlPullParser.START_TAG ,<news>或者<title>都是一个开始tag
 </news>对应一个XmlPullParser.END_TAG
 <news id =”2”>其中id就是这个tag的属性
 <title>光头强</title>这个tag标签没有属性,但是存在内容,内容就是光头强
 注意根元素<videnews>也是一个tag标签,我们通过程序的代码日志打印就可以看出来

 04-19 00:19:06.508 10100-10138/? D/123456 start_tag: videonews
 04-19 00:19:06.508 10100-10138/? D/123456 start_tag: news
 04-19 00:19:06.508 10100-10138/? D/123456 start_tag: title
 04-19 00:19:06.509 10100-10138/? D/123456 start_tag: timelength
 04-19 00:19:06.509 10100-10138/? D/123456 end_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: title
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: timelength
 04-19 00:19:06.510 10100-10138/? D/123456 end_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: title
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: timelength
 04-19 00:19:06.510 10100-10138/? D/123456 end_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 end_tag: videonews
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: videonews
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: title
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: timelength
 04-19 00:19:11.492 10370-10387/? D/123456 end_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: title
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: timelength
 04-19 00:19:11.492 10370-10387/? D/123456 end_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: title
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: timelength
 04-19 00:19:11.492 10370-10387/? D/123456 end_tag: news
 04-19 00:19:11.493 10370-10387/? D/123456 end_tag: videonews



 */
public class VideoNewsServiceImp  implements VideoNewsService{

    public   List<VideoNewsBean> getAllNews(String path) throws IOException, XmlPullParserException, JSONException {
        List<VideoNewsBean> news = new ArrayList<>();
        URL url = new URL(path);
        HttpURLConnection connection = (HttpURLConnection)url.openConnection();
        connection.setRequestMethod("GET");
        connection.setConnectTimeout(5000);
        connection.connect();
        if(connection.getResponseCode() == 200){
            InputStream inputStream = connection.getInputStream();
            //news = pullXML(inputStream);//服务器返回的xml格式
            news = pullJson(inputStream); //返回的json字符串
        }
        return  news;
    }

    //对xml文件进行解析
    public  List<VideoNewsBean> pullXML(InputStream in) throws XmlPullParserException, IOException {
        List<VideoNewsBean> datas = null;
        VideoNewsBean newsBean = null;
        XmlPullParser pullParser = Xml.newPullParser(); //获得一个xml的pull解析器
        pullParser.setInput(in, "utf-8"); //设置解析的编码格式,必须和服务器段的jsp中的xml编码格式对象
        int eventType = pullParser.getEventType(); //开启一个解析事件
        while (eventType != XmlPullParser.END_DOCUMENT){ //只要事件不是xml的文档结束事件,就一直解析,
            //XmlPullParser.END_DOCUMENT表示xml已经解析完成了
            switch (eventType){
                case XmlPullParser.START_DOCUMENT: //开启文档解析事件
                    datas = new ArrayList<>();
                    break;
                case  XmlPullParser.START_TAG: // 开始一个标签的解析
                    Log.d("123456 start_tag",pullParser.getName());
                    if(pullParser.getName().equals("news")){
                        newsBean = new VideoNewsBean();
                        //获得标签的属性值
                        int id = Integer.parseInt(pullParser.getAttributeValue(0));//0表示第一个属性,如果是1表示第二个属性值
                        newsBean.setVideoId(id);
                        break;

                    }else if(pullParser.getName().equals("title")){
                        //获得titile的内容
                       // <title>人民的利益</title>,解析分为三个事件<title>是标签开始事件 人民的利益是标签的内容为第二个事件,
                       // </title>为第三个事件
                        //现在解析到了<title>,pullParser.next()就是到下一个事件 pullParser.nextText()下一个事件的文本值
                        String title = pullParser.nextText();
                        newsBean.setVideoTitle(title);
                        break;

                    }else if(pullParser.getName().equals("timelength")){
                        int timeLength = Integer.parseInt(pullParser.nextText());
                        newsBean.setTimeLength(timeLength);
                        break;

                    }else {
                       break;
                    }
                case  XmlPullParser.END_TAG: //一个标签解析完成
                    Log.d("123456 end_tag",pullParser.getName());
                    if(pullParser.getName().equals("news")){
                        datas.add(newsBean);
                        newsBean = null;
                    }
                    break;
                case XmlPullParser.END_DOCUMENT: //结束文档解析事件
                    return datas;
            }
            //解析下一个事件
            eventType = pullParser.next();

        }

        return  datas;
    }


    //对服务器返回的json数据进行解析
    public    List<VideoNewsBean> pullJson(InputStream in) throws IOException, JSONException {
        List<VideoNewsBean> news = new ArrayList<>();
        //第一步将InputStream转化成byte[]
         byte[] buffer= new byte[1024];
        ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
         int len = -1;
         while ((len = in.read(buffer))!= -1){
             outputStream.write(buffer,0,len);
         }
        byte[] bytes = outputStream.toByteArray();

        String jsonString = new String(bytes,"utf-8");//编码必须和服务器的一样
        //第二步将byte转化成json数组
        JSONArray jsonArray = new JSONArray(jsonString);
        //获得json数组中的jsonObject对象
        for(int i = 0 ;i < jsonArray.length();i++){
            JSONObject jsonObject = jsonArray.getJSONObject(i);
            VideoNewsBean bean = new VideoNewsBean();
            bean.setVideoId(jsonObject.getInt("videoId"));
            bean.setVideoTitle(jsonObject.getString("videoTitle"));
            bean.setTimeLength(jsonObject.getInt("timeLength"));
            news.add(bean);
        }

        return  news;
    }
}

对xml解析比较复杂:

XML元素与HTML元素的格式基本相同,其格式如下:

<标记名称 属性名1="属性值1" 属性名1="属性值1" ……>内容</标记名称>

所有的数据内容都必须在某个标记的开始和结束标记内,而每个标记又必须包含在另一个标记的开始与结束标记内,形成嵌套式的分布,只有最外层的标记不必被其他的标记所包含。最外层的是根元素(Root),又称文件(Document)元素,所有的元素都包含在根元素内。在前面的Flowers.xml文件中,根元素就是<Flowers>,根元素必须而且只能有一个,在该文件有三个<Flower>子元素,这样的元素可以有多个。

XmlPullParser.START_DOCUMENT对应文档开始

XmlPullParser.END_DOCUMENT

<videnews>是根元素、有三个<News>子元素

<标记名称 属性名1="属性值1" 属性名1="属性值1" ……>内容</标记名称>

起始标签(外国语:starttag)表示一个特定区域的开始,例如<起始>;

结束标签(外国语:end tag)定义了一个区域的结束,除了在小于号之后紧跟着一个斜线(/)外,和起始标签基本一样,例如</结束>;

其中标记名称对应的事件是XmlPullParser.START_TAG ,<news>或者<title>都是一个开始tag

</news>对应一个XmlPullParser.END_TAG

<news id =”2”>其中id就是这个tag的属性

<title>光头强</title>这个tag标签没有属性,但是存在内容,内容就是光头强

注意根元素<videnews>也是一个tag标签,我们通过程序的代码日志打印就可以看出来

04-19 00:19:06.508 10100-10138/? D/123456 start_tag: videonews
 04-19 00:19:06.508 10100-10138/? D/123456 start_tag: news
 04-19 00:19:06.508 10100-10138/? D/123456 start_tag: title
 04-19 00:19:06.509 10100-10138/? D/123456 start_tag: timelength
 04-19 00:19:06.509 10100-10138/? D/123456 end_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: title
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: timelength
 04-19 00:19:06.510 10100-10138/? D/123456 end_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: title
 04-19 00:19:06.510 10100-10138/? D/123456 start_tag: timelength
 04-19 00:19:06.510 10100-10138/? D/123456 end_tag: news
 04-19 00:19:06.510 10100-10138/? D/123456 end_tag: videonews
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: videonews
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: title
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: timelength
 04-19 00:19:11.492 10370-10387/? D/123456 end_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: title
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: timelength
 04-19 00:19:11.492 10370-10387/? D/123456 end_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: news
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: title
 04-19 00:19:11.492 10370-10387/? D/123456 start_tag: timelength
 04-19 00:19:11.492 10370-10387/? D/123456 end_tag: news
 04-19 00:19:11.493 10370-10387/? D/123456 end_tag: videonews


我们来看activity的代码:

public class MainActivity extends Activity {

    private ListView lv_main;
    private VideoNewsService service = null;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initView();
        initData();
    }

    private void initData() {
        service = new VideoNewsServiceImp();
        final String path = "http://192.168.1.103:8080/lihuoming_25/ListServlet";
        ExecutorService executorService = Executors.newCachedThreadPool();
        executorService.execute(new Runnable() {
            @Override
            public void run() {
                try {
                    final List<VideoNewsBean> news = service.getAllNews(path);
                    runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            Toast.makeText(MainActivity.this,"" +
                                    "获得数据成功"+news.size(),Toast.LENGTH_SHORT).show();
                            List<Map<String, String>> listems = new ArrayList<Map<String, String>>();
                            for (int i = 0; i < news.size(); i++) {
                                VideoNewsBean newsBean = news.get(i);
                                Map<String, String> listem = new HashMap<String, String>();
                                listem.put("id", newsBean.getVideoId()+"");
                                listem.put("title", newsBean.getVideoTitle()+"");
                                listem.put("timeLength", newsBean.getTimeLength()+"");
                                listems.add(listem);
                            }
                            SimpleAdapter adapter = new SimpleAdapter(MainActivity.this, listems,
                                    R.layout.item_news, new String[] {"title", "timeLength" },
                                    new int[] {R.id.name,R.id.desc});
                            lv_main.setAdapter(adapter);

                            /*SimpleAdapter的参数说明
         * 第一个参数 表示访问整个android应用程序接口,基本上所有的组件都需要
         * 第二个参数表示生成一个Map(String ,Object)列表选项
         * 第三个参数表示界面布局的id  表示该文件作为列表项的组件
         * 第四个参数表示该Map对象的哪些key对应value来生成列表项
         * 第五个参数表示来填充的组件 Map对象key对应的资源一依次填充组件 顺序有对应关系
         * 注意的是map对象可以key可以找不到 但组件的必须要有资源填充  因为 找不到key也会返回null 其实就相当于给了一个null资源
         * 下面的程序中如果 new String[] { "name", "head", "desc","name" } new int[] {R.id.name,R.id.head,R.id.desc,R.id.head}
         * 这个head的组件会被name资源覆盖
         * */

                        }
                    });
                } catch (final Exception e) {
                    e.printStackTrace();
                    runOnUiThread(new Runnable() {
                        @Override
                        public void run() {
                            Toast.makeText(MainActivity.this,"" +
                                    "获得数据失败"+e.toString(),Toast.LENGTH_SHORT).show();
                        }
                    });
                }
            }
        });

    }

    private void initView() {
        lv_main = (ListView) findViewById(R.id.lv_main);
    }
}

我们来看xml布局文件:

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"
    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="application.weiyuan.com.lihuoming_25.MainActivity">
<ListView
    android:id="@+id/lv_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

</ListView>

</RelativeLayout>

item_news.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="horizontal" >


        <TextView
            android:id="@+id/name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="20dp"
            android:textColor="#f0f"
            android:paddingLeft="10dp"/>


        <TextView
            android:id="@+id/desc"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textSize="14dp"
            android:paddingLeft="10dp"/>


</LinearLayout>
原文地址:https://www.cnblogs.com/kebibuluan/p/6755960.html