国内最全IT社区平台 联系我们 | 收藏本站
华晨云阿里云优惠2
您当前位置:首页 > php开源 > 综合技术 > Android限制EditText只能输入中文或者指定内容的实现

Android限制EditText只能输入中文或者指定内容的实现

来源:程序员人生   发布时间:2017-03-13 17:30:21 阅读次数:5772次

最近项目中要限制EditText中只能输入中文,之前写过1个限制EditText只能输入中文的实现,不过存在1些问题,而且扩大性不是很好,所以换了1种方法来实现.
先看1下效果图:

这里写图片描述


具体实现

1般对EditText的操作及处理都是用addTextChangedListener方法来对EditText进行监听,以后在监听方法中去做处理.这里也打算用这个种方法来做,大体的思路是监听EditText中输入的内容,然后将不是中文的部份清除掉,也就是置为空.所以大概应当这样写

 mLimitEt.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }

            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
            // 1.处理输入的内容s:清除其中不是中文的部份
            ...
            // 2.设置处理完的s
            mLimitEt.setText("处理以后的s");
            }

            @Override
            public void afterTextChanged(Editable s) {

            }
        });

处理的方法这里先不写,先来看1下这样写会出现的1个问题,运行1下,输入1些内容会发现程序崩溃了,查看崩溃信息,会发现出现了StackOverflowError异常,这是甚么缘由呢?带着疑问去扒了1下源码(看源码时遇到1个问题,升级完Studio以后,发现没法查看源码了,查了1些资料解决了,也有相同问题的童鞋可以参考下我写的 Mac版Android Studio查看不到源码的解决方法,windows版解决方法也类似)出现异常的位置在 mLimitEt.setText()这句代码上,所以先看1下setText()方法.setText方法在TextView中,看1下实现(这里只关心引发异常的部份,其他部份的内容不讨论)

private void setText(CharSequence text, BufferType type,
                         boolean notifyBefore, int oldlen) {
   ...
   // Text改变前的回调解理
   sendBeforeTextChanged(mText, 0, oldlen, text.length());
   ...
   // Text改变中的回调解理
   sendOnTextChanged(text, 0, oldlen, textLength);
   ...
   // Text改变后的回调解理
   sendAfterTextChanged((Editable) text); 

}

在setText方法中可以看到这几个方法,然后看1下这些方法做的处理是甚么

     /**
     * Not private so it can be called from an inner class without going
     * through a thunk.
     */
       void sendOnTextChanged(CharSequence text, int start, int before, int after) {
        if (mListeners != null) {
            final ArrayList<TextWatcher> list = mListeners;
            final int count = list.size();
            for (int i = 0; i < count; i++) {
                list.get(i).onTextChanged(text, start, before, after);
            }
        }

        if (mEditor != null) mEditor.sendOnTextChanged(start, after);
    }

     /**
     * Not private so it can be called from an inner class without going
     * through a thunk.
     */
    void sendAfterTextChanged(Editable text) {
        if (mListeners != null) {
            final ArrayList<TextWatcher> list = mListeners;
            final int count = list.size();
            for (int i = 0; i < count; i++) {
                list.get(i).afterTextChanged(text);
            }
        }
        hideErrorIfUnchanged();
    }

看1下这些方法,能不能发现点甚么,可以看到有1个ArrayList< TextWatcher >对象,先进行判空处理,如果这个对象中存在TextWatcher监听,则逐条进行回调操作.再回头看1下之前写的EditText中回调方法的实现,在回调中,对这个EditText进行了setText操作,由于EditText实现了TextWatcher的回调接口,这样就致使了无穷循环 setText->onTextChanged->setText…… 终究致使程序崩溃.那该如何解决这个问题呢.其实很简单,看1下代码

 mLimitEt.addTextChangedListener(new TextWatcher() {
            @Override
            public void beforeTextChanged(CharSequence s, int start, int count, int after) {

            }

            @Override
            public void onTextChanged(CharSequence s, int start, int before, int count) {
            // 1.处理输入的内容s:清除其中不是中文的部份
            ...
            // 2.删除监听
            mLimitEt.removeTextChangedListener(this);
            // 3.设置处理完的s
            mLimitEt.setText("处理以后的s");
            // 4.重新添加监听
            mLimitEt.addTextChangedListener(this);
            }

            @Override
            public void afterTextChanged(Editable s) {

            }
        });

在setText之前先删除之前的回调监听,setText时由于没有TextWatcher的监听方法,所以不会出现无穷循环的情况,当setText以后再重新添加回调监听,这样就避免了崩溃的产生.以后看1下清除非中文部份的实现,直接看代码

    /**
     * 清除不是中文的内容
     *
     * @param regex
     * @return
     */
    private String clearLimitStr(String regex, String str) {
        return str.replaceAll("[^\u4E00-\u9FA5]", "");
    }

用了String的replaceAll方法来处理输入的内容(用了正则表达式,使用起来很简单).在onTextChanged和afterTextChanged方法中,得到的输入内容实际上是整体的输入内容,所以用replaceAll方法,可以去打印1下这几个方法中的参数,这里就不做了.看1下整体代码

LimitInputTextWatcher:

package com.example.junweiliu.limitinputdemo;

import android.text.Editable;
import android.text.TextWatcher;
import android.widget.EditText;

/**
 * Created by junweiliu on 17/1/6.
 */
public class LimitInputTextWatcher implements TextWatcher {
    /**
     * et
     */
    private EditText et = null;
    /**
     * 挑选条件
     */
    private String regex;
    /**
     * 默许的挑选条件(正则:只能输入中文)
     */
    private String DEFAULT_REGEX = "[^\u4E00-\u9FA5]";

    /**
     * 构造方法
     *
     * @param et
     */
    public LimitInputTextWatcher(EditText et) {
        this.et = et;
        this.regex = DEFAULT_REGEX;
    }

    /**
     * 构造方法
     *
     * @param et    et
     * @param regex 挑选条件
     */
    public LimitInputTextWatcher(EditText et, String regex) {
        this.et = et;
        this.regex = regex;
    }

    @Override
    public void beforeTextChanged(CharSequence charSequence, int i, int i1, int i2) {

    }

    @Override
    public void onTextChanged(CharSequence charSequence, int i, int i1, int i2) {

    }

    @Override
    public void afterTextChanged(Editable editable) {
        String str = editable.toString();
        String inputStr = clearLimitStr(regex, str);
        et.removeTextChangedListener(this);
        // et.setText方法可能会引发键盘变化,所以用editable.replace来显示内容
        editable.replace(0, editable.length(), inputStr.trim());
        et.addTextChangedListener(this);

    }

    /**
     * 清除不符合条件的内容
     *
     * @param regex
     * @return
     */
    private String clearLimitStr(String regex, String str) {
        return str.replaceAll(regex, "");
    }
}

为了扩大性,提出来了1个类,提供了两个构造方法,如果需要限制其他的特殊内容,可以设置正则的规则.固然如果很简单的话,用EidtText自带的digits属性就能够了.还有1个问题,需要注意,代码中没有用et.setText方法,是由于setText方法可能引发键盘变化异常,所以这里用 editable.replace(0, editable.length(), inputStr.trim());这个方法和setText方法的实现效果是1样的.不过也需要对监听进行处理,缘由也是由于会引发无穷循环,感兴趣的童鞋可以去看1下.

完全代码

MainActivity:

package com.example.junweiliu.limitinputdemo;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.EditText;

public class MainActivity extends AppCompatActivity {
    /**
     * et
     */
    private EditText mLimitEt;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mLimitEt = (EditText) findViewById(R.id.et_limit);
        mLimitEt.addTextChangedListener(new LimitInputTextWatcher(mLimitEt));
        // 去除除a-z  A-Z与0⑼和中文的其他符号
//        mLimitEt.addTextChangedListener(new LimitInputTextWatcher(mLimitEt, "[^a-zA-Z0⑼\u4E00-\u9FA5]"));
    }
}

activity_main:

<?xml version="1.0" encoding="utf⑻"?>
<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="com.example.junweiliu.limitinputdemo.MainActivity">
    <!--输入框-->
    <!--android:digits="1234567890"-->
    <EditText
            android:id="@+id/et_limit"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerHorizontal="true"
            android:hint="我是1个受限制的输入框"/>
    <!--输入框-->
    <EditText
            android:layout_below="@+id/et_limit"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerHorizontal="true"
            android:text="复制我fuzhiwo845"
            android:hint=""/>
</RelativeLayout>
生活不易,码农辛苦
如果您觉得本网站对您的学习有所帮助,可以手机扫描二维码进行捐赠
程序员人生
------分隔线----------------------------
分享到:
------分隔线----------------------------
关闭
程序员人生