第二章 布局技巧与列表

1 布局与优化

1.1 五大布局

  • LinearLayout:线性布局
  • orientation:vertical:垂直的,horizontal:水平的
  • weight:设置比重时,layout_width="0"。weightSum总比重。例如:微信底部四个tab平分宽带,可以使其weight 1:1:1:1
  • RelativeLayout:相对布局
  • xmlns:xml的命名空间
  • layout_alignParentRight:与布局容器的右边对齐
  • layout_alignParentBottom:位于父容器的底部
  • layout_alignLeft:与给出id组件的左边对齐
  • layout_above/toRightOf:给出id组件的上方/右边
  • layout_marginTop/toRightOf:外边距
  • layout_paddingRight:内边距
  • FrameLayout:帧布局,一层一层叠加起来
  • AbsoluteLayout:绝对布局
  • TableLayout:表格布局
  • TableRow

1.2 布局优化

  • 减少布局层次的重要性层次过多会影响到性能
  • 官方建议布局层次--最多10层,所以要减少层次,删除无用布局
  • 学会使用相对布局。布局结构要清晰,选择合适的布局
  • 组合控件
  • 一些有用的属性
  • 布局如何引用相同的部分,可以提取出来然后</include>重用布局文件
  • <merge/>减少视图层级
  • <ViewStub/>需要是才加载,介绍ViewStub代码 ,介绍ViewStub文字,两篇结合来看。ViewStub指向的布局文件在多数情况下并不会被显示出来,只会在某种特殊的情况下对ViewStub进行inflate操作,而且这种操作只能进行一次,此时会将ViewStub替换掉,变成显示所指向的布局文件。,一旦替换后,此时原来的布局文件中就没有ViewStub控件了,这就好像ViewStub只是为其所指向的布局占了一个位子,需要时才显示
  • 小技巧
  • 不要嵌套多个使用layout_weight属性的LinearLayout
  • Android lint :删除无用的资源,布局,类,引用
  • HierarchyViewer:分析性能及其优化

2 无比重要的ListView

这里写图片描述
  • AdapterView是一个容器,把很多的列表项以合适的形式显示出来,而列表项的内容和格式可以用Adapter指定,调用AdapterView的setAdapter(Adapter)方法设置Adapter即可。

2.1四种设置Adapter的方法:(以ListView为例子)

  • 使用ArrayAdapter设置
    1. 告诉大家我要创建列表项了,初始化这个控件
    2. 定义了一个数组,该数组是为多个列表项提供了数据
    3. 创建ArrayAdapter(Context,textViewResource,数组或者List)对象。其中Context代表访问应用的接口,一般是this。textViewResource代表一个资源的ID,该资源分ID是指定了每个列表项的组件,等于为每个列表项指定了格式。后面的数据就按照组件的格式传入。
    4. 最后为该ListView设置Adapter
public class ArrayAdapterActivity extends Activity{
    ListView mListView1;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_arrayadapter);
        //1
        mListView1 = (ListView) findViewById(R.id.listview1);
        //2
        String[] arr1 = {"王俊凯" ,"王源","易烊千玺"};
        //3
        ArrayAdapter<String> adapter1 =
                new ArrayAdapter<String>(this,R.layout.array_item,arr1);
        //4
        mListView1.setAdapter(adapter1);    
    }
}

  • 让Activity继承ListActivity来实现
  1. 创建ArrayAdapter对象之后,直接使用setListActivity(adapter)设置该窗口显示列表
public class MainActivity extends ListActivity{
    @override
    public void onCreate(Bundle saveIdnstanceState){
        supert.onCreate(savedInstanceState);
        String[] arr = {"刘诗诗", "赵丽颖", "杨幂"};
        ArrayAdapter<String> adapter1 = new ArrayAdapter<String>(this, R.layout.item, arr);
        setListAdapter(adapter);
         
    }
}
  • 使用SimpleAdapter
  1. 前面都是创建提供列表内容的数组
  2. 创建一个List集合,集合元素是Map,把前面的内容数组封装成一个List集合
  3. 创建SimpleAdapter对象。后面有5个参数,第一个是Context。第二个是一个元素是Map的List集合,每个Map对象生成一个列表项。第三个参数是指定界面布局的ID。第四个是一个String[]对象的参数,决定提取Map中那些key对应的value来生成列表项。第五个参数是对应于value填充的组件,该组件是来源与第三个参数的布局。是int[]类型的参数
  4. 设置Adapter
public class MainActivity extends AppCompatActivity {
    //1
    private  String[] names = new String[]{"lisa" , "linda" , "july" , "anna"};

    private String[] describe = new String[]{"a cute child" , "a beautiful girl" , "a smart student" , "a handsome boy"};

    private int[] imageId = new int[]{R.drawable.libai,R.drawable.nongyu,R.drawable.qingzhao,R.drawable.tiger};

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //2
        List<Map<String,Object>> listItems = new ArrayList<Map<String, Object>>();

        for (int i = 0 ; i < names.length ; i++){
           Map<String , Object> listItem = new HashMap<String, Object>();
            listItem.put("header" ,imageId[i]);
            listItem.put("name" , names[i]);
            listItem.put("describe" , describe[i]);
            listItems.add(listItem);
        }
        //3
        SimpleAdapter simpleAdapter = new SimpleAdapter(this ,
                listItems ,R.layout.simple_item ,
                new String[]{"header","name","describe"} ,
                new int[]{R.id.header,R.id.name,R.id.desc});

        ListView listView =(ListView) findViewById(R.id.mylist);
        //4
        listView.setAdapter(simpleAdapter);

        Intent intent = getIntent();
        if (intent != null) {
            String string = intent.getStringExtra(SplashActivity.TITLE);
            setTitle(string);
        }
    }
}
  • 扩展BaseAdapter实现 ,需要重写4个方法
  1. int getCount():返回值控制Adapter会有多少个列表项
  2. Object getItem(int i):返回第i处的列表项的内容
  3. getItemId(int i):返回第i除列表项的id
  4. View getView(int i, View view, ViewGroup viewGroup):返回第i除列表项的组件
 BaseAdapter baseAdapter = new BaseAdapter() {
            @Override
            public int getCount() {
                //指定40个选项
                return 40;
            }

            @Override
            public Object getItem(int i) {
                return null;
            }

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

            @Override
            public View getView(int i, View view, ViewGroup viewGroup) {
                LinearLayout linearLayout = new LinearLayout(MainActivity.this);
                linearLayout.setOrientation(LinearLayout.VERTICAL);
                ImageView imageView = new ImageView(MainActivity.this);
                imageView.setImageResource(R.mipmap.ic_launcher);
                TextView text = new TextView(MainActivity.this);
                text.setText("this is" + i + "listview");
                text.setTextSize(20);
                linearLayout.addView(imageView);
                linearLayout.addView(text);
                return linearLayout;
            }
        };

2.2 优化

  • 不必每次都要加载视图,只需要第一次的时候加载,所以可以加个判断
@Override
    public View getView(int i, View view, ViewGroup viewGroup) {
        ViewHolder viewHolder;
        //优化
        if(view == null){
            view = mLayoutInflater.inflate(R.layout.activity_items,null);
            viewHolder = new ViewHolder();
            //获取控件
            viewHolder.phone_name = (TextView) view.findViewById(R.id.item_name);
            viewHolder.phone_age = (TextView) view.findViewById(R.id.item_age);
            viewHolder.phone_image = (ImageView) view.findViewById(R.id.item_image);
            view.setTag(viewHolder);
        }else {
            viewHolder = (ViewHolder) view.getTag();
        }
        //数据绑定,mInfoList是一个类型为用户的List,该用户有Age和Name属性
        viewHolder.phone_name.setText(mInfoList.get(i).getmName());
        viewHolder.phone_age.setText(String.valueOf(mInfoList.get(i).getmAge()));
        viewHolder.phone_image.setImageResource(R.mipmap.ic_launcher);
        return view;
    }
  • 设置类来保存所加载视图的控件
class ViewHolder{
    TextView phone_name;
    TextView phone_age ;
    ImageView phone_image ;
}
  • 所以只有第一次加载的时候,会去加载视图获取控件,当前视图不为空时,只会去获取视图里控件,节约内存。

2.3 item不同怎么办,getItemViewType()

2.4 风格不同的分割线

  • ListView的 android:divider属性来设置,所以可以用自定义的Drawable来设置分割线的风格

3 GridView和ScrollView

3.1 GridView与ListView的相似之处

  • 需要设置Adapter
  • 它们俩的parent都实现了AdapterView

3.2 GridView的常用实现

  • 常用属性:
  • android:numColums:设置列数
  • android:horizontalSpacing:设置各元素之间的水平间距
  • android:verticalSpacing:设置各元素之间的垂直间距
  • android:gravity:设置对齐方式
  • 使用了SimpleAdapter,ImageSwitcher是图片切换器,在切换ImgeView时使用动画
  • activity_main.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="vertical"
    android:gravity="center_horizontal">

    <GridView
        android:id="@+id/grid01"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:horizontalSpacing="2dp"
        android:verticalSpacing="2dp"
        android:numColumns="4"
        android:gravity="center"/>
    <ImageSwitcher
        android:id="@+id/switcher"
        android:layout_width="300dp"
        android:layout_height="300dp"
        android:layout_gravity="center_horizontal"
        android:inAnimation="@android:anim/fade_in"
        android:outAnimation="@android:anim/fade_out"/>
</LinearLayout>

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

    <ImageView
        android:id="@+id/imageView"
        android:layout_width="50dp"
        android:layout_height="50dp"
        />

</LinearLayout>
  • java源代码
package com.example.w.myapplication;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.AdapterView;
import android.widget.GridView;
import android.widget.ImageSwitcher;
import android.widget.ImageView;
import android.widget.SimpleAdapter;
import android.widget.Toast;
import android.widget.ViewSwitcher;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class MainActivity extends AppCompatActivity {

    GridView gridView;
    ImageSwitcher imageSwitcher;

    int[] imageIds = new int[] {
            R.drawable.bomb5,R.drawable.bomb6,R.drawable.bomb7,R.drawable.bomb8,
            R.drawable.bomb9,R.drawable.bomb10,R.drawable.bomb11,R.drawable.bomb12,
            R.drawable.bomb13,R.drawable.bomb14,R.drawable.bomb15,R.drawable.bomb16,
    };

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

        gridView = (GridView)findViewById(R.id.grid01);

        //创建一个List对象,List对象的元素是Map
        List<Map<String,Object>> listItems = new ArrayList<Map<String, Object>>();
        for (int i = 0 ; i < imageIds.length ; i++){
            Map<String,Object> listItem = new HashMap<String, Object>();
            listItem.put("image" , imageIds[i]);
            listItems.add(listItem);
        }
        //为ImageSwitcher提供一个ViewFactory,由其生成View
        imageSwitcher = (ImageSwitcher)findViewById(R.id.switcher);
        imageSwitcher.setFactory(new ViewSwitcher.ViewFactory() {
            @Override
            public View makeView() {
                //创建ImageView对象
                ImageView imageView = new ImageView(MainActivity.this);
                imageView.setScaleType(ImageView.ScaleType.FIT_CENTER);
                imageView.setLayoutParams(new ImageSwitcher.LayoutParams(
                        ViewGroup.LayoutParams.WRAP_CONTENT,
                        ViewGroup.LayoutParams.WRAP_CONTENT));
                return imageView;
            }
        });
        SimpleAdapter simpleAdapter = new SimpleAdapter(this , listItems ,R.layout.cell
                ,new String[]{"image"},new int[]{R.id.imageView});

        gridView.setAdapter(simpleAdapter);

        //列表被单击
        gridView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
            @Override
            public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
                imageSwitcher.setImageResource(imageIds[i]);
                Toast.makeText(MainActivity.this,"图片" + i +"被点击了" , Toast.LENGTH_LONG).show();
            }
        });

        //列表被选中
        gridView.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
            @Override
            public void onItemSelected(AdapterView<?> adapterView, View view, int i, long l) {
                imageSwitcher.setImageResource(imageIds[i]);
                Toast.makeText(MainActivity.this,"图片" + i +"被选中了" , Toast.LENGTH_LONG).show();
            }

            @Override
            public void onNothingSelected(AdapterView<?> adapterView) {

            }
        });
    }
}

3.3 ScrollView

  • 最多只能包含一个组件,或只支持垂直滚动,若需要水平滚动,则可借助另一个滚动视图-HorizontalScrollView

你可看可不看,但我必须得看的Tips:

  1. ScrolView了解不是很多,需要进一步去了解
  2. ListView很重要,和BaseAdapter搭配起来,天生一对
  3. 布局应快要学完了,16.7.23
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容