android自定义如何解说,实例讲解Android应用中自定义组合控件的方法

本文介绍了如何在Android中实现自定义View,通过组合已有的View(如Button和ImageView)创建一个自定义的ImageButton。作者详细解释了如何使用FrameLayout将这两个组件包裹,并在 attrs.xml 中定义属性,然后在Java代码中读取这些属性并设置图片资源。此外,还提供了一个组合多个TextView和View的例子,展示了如何创建一个可定制的页面指示条。
摘要由CSDN通过智能技术生成

自定义view大概可以分成:

继承view

组合view

自己画的view(用paint和canvas)

自定义viewgroup(widget)

最近又跟同学聊起来了,于是就准备自己实现一个imagebutton来练练手。(最简单的了)以及后面一个把imageview与textview结合使用的案例。

ImageButton要实现ImageButton最容易想到的不就是把button和imageview合在一起么。嗯,人容易就想到这是一个组合自定义view。可是如何把一个image放到button上呢,自然就是用一个FrameLayout把button和imageview包裹起来,然后将这个layout当作一个weidgt。把一个layout当成一个控件我开始不太理解。后来了解到一些东西。

补充知识点(安卓的顶层视图)在android里面,你的app界面的活动的视图并非是最顶级的视图。大家应该都知道viewtree。在安卓里面,最顶级的视图是一个叫DecorView的东西。它包括你的状态栏,标题栏,以及你的活动界面。而且这个活动界面,安卓会自动帮你提前弄进去一个FrameLayout。可以这么理解吧,在安卓这个上帝的面前,你的activity就是放在他的framelayout的一个个view。上一张图~

d409d4498be9877d557c746cbc507ee4.png

DecorView->LinearLayout(状态栏和activity)->FrameLayout(activity)->你自己的acitivty视图。

实例讲解好啦,回归正题。我上代码了。

myimagebutton_layout.xml

xmlns:android="http://schemas.android.com/apk/res/android"

android:layout_width="match_parent"

android:layout_height="match_parent">

android:layout_width="match_parent"

android:layout_height="match_parent"

android:id="@+id/button_imagebutton" />

android:layout_width="match_parent"

android:layout_height="match_parent"

android:id="@+id/imageview_button" />

用一个framelayout把两个控件装起来,然后组合。组合了之后我们就去自定义我们这个控件的属性。button不用变对吧,用原生的就好了,就是需要有一个图片的src这个属性就好了。

在values目录下的attrs.xml文件_

一般来说,declare-styleable标签的名字都是你的自定义控件的名字。attr就是你的自定义控件属性。format是这个属性的值的数据类型。这里的reference表示引用。dimension表示的是dp或者sp的大小。还有一些可以自己去看看。定义好属性,我们就开始写控件了。

MyImageButton.java

public class MyImageButton2 extends FrameLayout{

ImageView mImageView;

Button mButton;

int resId;

public MyImageButton2(Context context) {

super(context);

}

public MyImageButton2(Context context, AttributeSet attrs) {

super(context, attrs);

//导入布局

LayoutInflater.from(context).inflate(R.layout.myimagebutton_layout, this);

mImageView=(ImageView)findViewById(R.id.imageview_button);

mButton = (Button) findViewById(R.id.button_imagebutton);

//获得这个控件对应的属性。

TypedArray a = getContext().obtainStyledAttributes(attrs, R.styleable.MyImageButton);

try{

//获得属性值

resId = a.getResourceId(R.styleable.MyImageButton_Imagesrc, 0);

}finally {

//回收这个对象

a.recycle();

}

if(resId != 0){

mImageView.setImageResource(resId);

}

}

public void setImage(int resId){

mImageView.setImageResource(resId);

}

}

一般来说,继承view(就这么说吧,反正都是都是继承view的),实现两三个构造函数就可以了。在第二个里面有一个attr,这就是传进来的属性。如果resId不为默认值,就表示用户在xml里面传入了这个属性值,你也必须要写一个set方法,让用户可以在java代码中去改变属性值。

test.myimagebutton.layout.xml_

android:orientation="vertical"

android:layout_width="match_parent"

android:layout_height="match_parent">

xmlns:pt = "http://schemas.android.com/apk/res/com.example.think.testview"

android:layout_height="100dp"

android:layout_width="match_parent"

pt:Imagesrc = "@drawable/ic_launcher"

/>

效果图

2f93cc77d998a5291b6598866c37c28a.png

另一个实例

这里我们来看一个view组合控件的例子:

2beeb2a3d7e6a56ecbad31feceada68a.png

测试1

测试2

测试3

测试4

android:layout_width="match_parent"

android:layout_height="match_parent"

android:background="@color/gray_common_background"

android:orientation="vertical">

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:orientation="horizontal">

android:id="@+id/txt_viewpage_bar1"

style="@style/style_common_pagebar"

android:textColor="@color/theme_red" />

android:id="@+id/txt_viewpage_bar2"

style="@style/style_common_pagebar" />

android:id="@+id/txt_viewpage_bar3"

style="@style/style_common_pagebar" />

android:id="@+id/txt_viewpage_bar4"

style="@style/style_common_pagebar" />

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:orientation="horizontal">

android:id="@+id/view_viewpage_bar1"

style="@style/style_common_pagebar_view" />

android:id="@+id/view_viewpage_bar2"

style="@style/style_common_pagebar_view" />

android:id="@+id/view_viewpage_bar3"

style="@style/style_common_pagebar_view" />

android:id="@+id/view_viewpage_bar4"

style="@style/style_common_pagebar_view" />

android:id="@+id/common_viewpage_buttom"

android:layout_width="match_parent"

android:layout_height="0dp"

android:layout_weight="1" />

public interface SlbPageViewOnPageChangeListener {

void onPageSelected(int position);

void onPageScrollStateChanged(int state);

}

public class SlbPageViewBar extends LinearLayout {

private TextView[] mTotalTxtViews;

private View[] mTotalViews;

private List mTxtViews;

private List mViews;

private ViewPager mViewPager;

private SlbPageViewOnPageChangeListener mSlbPageViewOnPageChangeListener;

public SlbPageViewBar(Context context, AttributeSet attrs) {

super(context, attrs);

mTotalTxtViews = new TextView[4];

mTotalViews = new View[4];

mTxtViews = new ArrayList<>();

mViews = new ArrayList<>();

initView();

}

private void initView() {

View.inflate(getContext(), R.layout.layout_common_top_pageview_bar, this);

mViewPager = (ViewPager) findViewById(R.id.common_viewpage_buttom);

mTotalTxtViews[0] = (TextView) findViewById(R.id.txt_viewpage_bar1);

mTotalTxtViews[1] = (TextView) findViewById(R.id.txt_viewpage_bar2);

mTotalTxtViews[2] = (TextView) findViewById(R.id.txt_viewpage_bar3);

mTotalTxtViews[3] = (TextView) findViewById(R.id.txt_viewpage_bar4);

mTotalViews[0] = findViewById(R.id.view_viewpage_bar1);

mTotalViews[1] = findViewById(R.id.view_viewpage_bar2);

mTotalViews[2] = findViewById(R.id.view_viewpage_bar3);

mTotalViews[3] = findViewById(R.id.view_viewpage_bar4);

}

/**

* @param title 标题名称数组

*/

public void setTitleView(String[] title) {

int legth = title.length > 4 ? 4 : title.length;

for (int i = 0; i < legth; i++) {

mTotalTxtViews[i].setText(title[i]);

mTotalTxtViews[i].setVisibility(View.VISIBLE);

mTotalViews[i].setVisibility(View.INVISIBLE);

mTotalTxtViews[i].setOnClickListener(new SlbTextViewOnLister(i));

mTxtViews.add(mTotalTxtViews[i]);

mViews.add(mTotalViews[i]);

}

mTotalViews[0].setVisibility(View.VISIBLE);

mViewPager.setOnPageChangeListener(mOnButtomPageChangeListener);

}

public ViewPager getViewPager(){

return mViewPager;

}

public void setSlbPageViewOnPageChangeListener(SlbPageViewOnPageChangeListener slbPageViewOnPageChangeListener) {

this.mSlbPageViewOnPageChangeListener = slbPageViewOnPageChangeListener;

}

private void changeViewBg(int textViewId, int viewId) {

for (View view : mViews) {

if (viewId == view.getId()) {

view.setVisibility(View.VISIBLE);

} else {

view.setVisibility(View.INVISIBLE);

}

}

for (TextView view : mTxtViews) {

if (textViewId == view.getId()) {

view.setTextColor(getResources().getColor(R.color.theme_red));

} else {

view.setTextColor(getResources().getColor(R.color.black_normal_text_33333));

}

}

}

class SlbTextViewOnLister implements OnClickListener {

private int mItem;

public SlbTextViewOnLister(int item) {

this.mItem = item;

}

@Override

public void onClick(View v) {

mViewPager.setCurrentItem(mItem);

}

}

private ViewPager.OnPageChangeListener mOnButtomPageChangeListener = new ViewPager.OnPageChangeListener() {

@Override

public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {

}

@Override

public void onPageSelected(int position) {

switch (position) {

case 0:

changeViewBg(R.id.txt_viewpage_bar1, R.id.view_viewpage_bar1);

break;

case 1:

changeViewBg(R.id.txt_viewpage_bar2, R.id.view_viewpage_bar2);

break;

case 2:

changeViewBg(R.id.txt_viewpage_bar3, R.id.view_viewpage_bar3);

break;

case 3:

changeViewBg(R.id.txt_viewpage_bar4, R.id.view_viewpage_bar4);

break;

}

if (mSlbPageViewOnPageChangeListener != null)

mSlbPageViewOnPageChangeListener.onPageSelected(position);

}

@Override

public void onPageScrollStateChanged(int state) {

if (mSlbPageViewOnPageChangeListener != null)

mSlbPageViewOnPageChangeListener.onPageScrollStateChanged(state);

}

};

}

public class CommonViewpageAdapter extends FragmentPagerAdapter {

private List mFragments;

private String [] mTitles;

public CommonViewpageAdapter(FragmentManager fm, List fragments, String [] titles) {

super(fm);

mFragments = fragments;

mTitles = titles;

}

@Override

public Fragment getItem(int position) {

return mFragments.get(position);

}

@Override

public int getCount() {

return mFragments.size();

}

@Override

public CharSequence getPageTitle(int position) {

return mTitles[position];

}

@Override

public void destroyItem(ViewGroup container, int position, Object object) {

super.destroyItem(container, position, object);

}

}

android:id="@+id/slbpv_buy_huoqibao_SlbPageViewBarshow"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:orientation="vertical"/>

private void initButtomPageFragment(SlbPageViewBar slbPageViewBar) {

List fragmentList = new ArrayList<>();

Test1Fragment test1Fragment = new Test1Fragment();

Test2Fragment test2Fragment = new Test2Fragment();

Test3Fragment test3Fragment = new Test3Fragment();

Test4Fragment test4Fragment = new Test4Fragment();

fragmentList.add(test1Fragment);

fragmentList.add(test2Fragment);

fragmentList.add(test3Fragment);

fragmentList.add(test4Fragment);

String[] titles = getResources().getStringArray(R.array.buy_huoqi_bao_zanquan_detail);

slbPageViewBar.setTitleView(titles);

CommonViewpageAdapter viewPagerAdapter = new CommonViewpageAdapter(

getSupportFragmentManager(), fragmentList, titles);

mButtomPageView.setAdapter(viewPagerAdapter);

mButtomPageView.setOffscreenPageLimit(3);

mButtomPageView.setCurrentItem(0);

}

深度学习是机器学习的一个子领域,它基于人工神经网络的研究,特别是利用多层次的神经网络来进行学习和模式识别。深度学习模型能够学习数据的高层次特征,这些特征对于图像和语音识别、自然语言处理、医学图像分析等应用至关重要。以下是深度学习的一些关键概念和组成部分: 1. **神经网络(Neural Networks)**:深度学习的基础是人工神经网络,它是由多个层组成的网络结构,包括输入层、隐藏层和输出层。每个层由多个神经元组成,神经元之间通过权重连接。 2. **前馈神经网络(Feedforward Neural Networks)**:这是最常见的神经网络类型,信息从输入层流向隐藏层,最终到达输出层。 3. **卷积神经网络(Convolutional Neural Networks, CNNs)**:这种网络特别适合处理具有网格结构的数据,如图像。它们使用卷积层来提取图像的特征。 4. **循环神经网络(Recurrent Neural Networks, RNNs)**:这种网络能够处理序列数据,如时间序列或自然语言,因为它们具有记忆功能,能够捕捉数据的时间依赖性。 5. **长短期记忆网络(Long Short-Term Memory, LSTM)**:LSTM 是一种特殊的 RNN,它能够学习长期依赖关系,非常适合复杂的序列预测任务。 6. **生成对抗网络(Generative Adversarial Networks, GANs)**:由两个网络组成,一个生成器和一个判别器,它们相互竞争,生成器生成数据,判别器评估数据的真实性。 7. **深度学习框架**:如 TensorFlow、Keras、PyTorch 等,这些框架提供了构建、训练和部署深度学习模型的工具和库。 8. **激活函数(Activation Functions)**:如 ReLU、Sigmoid、Tanh 等,它们在神经网络用于添加非线性,使得网络能够学习复杂的函数。 9. **损失函数(Loss Functions)**:用于评估模型的预测与真实值之间的差异,常见的损失函数包括均方误差(MSE)、交叉熵(Cross-Entropy)等。 10. **优化算法(Optimization Algorithms)**:如梯度下降(Gradient Descent)、随机梯度下降(SGD)、Adam 等,用于更新网络权重,以最小化损失函数。 11. **正则化(Regularization)**:技术如 Dropout、L1/L2 正则化等,用于防止模型过拟合。 12. **迁移学习(Transfer Learning)**:利用在一个任务上训练好的模型来提高另一个相关任务的性能。 深度学习在许多领域都取得了显著的成就,但它也面临着一些挑战,如对大量数据的依赖、模型的解释性差、计算资源消耗大等。研究人员正在不断探索新的方法来解决这些问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值