第一点,涉及的东东其实蛮多,不同语言在不同编码中占据字节数等,不同语言在U8等编码的表示范围等,这一整块知识很丰富,
自己暂时没有理的特别顺,稍后整理再说吧。
第二点,目前主流app的处理方案也各有不同,qq5.0以前的版本,发表说说貌似是没有字数限制的(我试了一个350字左右的照样发),
5.0以后限制了,这样如果用户输入字数超过限制便显示负数,但是仍然允许用户输入,就是不能发表了。这样就把处理权全部交给用户了,
其实更省事了!
实现吧。android:maxLength="10" // 即限制最大输入字符个数为10。
(1)中英文都算一个字符。
editText.setFilters(new InputFilter[]{new InputFilter.LengthFilter(10)});
(1)中英文都算一个字符。
crash问题:使用搜狗输入法时,在字数将要或者已经达到限制之后,一次输入大量中文(不按空格,一直打字,知道搜狗的缓存字符达到上限),
crash,stackoverflow 内存溢出。呵呵,百度输入法不会哈。
解决:监听器中动态去除、添加就ok了。见下边方案一private TextWatcher textWatcher = new TextWatcher() { private int editStart; private int editEnd; private int maxLen = 10; // the max byte @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { Log.d("TextChanged", "---->beforeTextChanged : start = " + start + " count = " + count + " after = " + after); } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { } @Override public void afterTextChanged(Editable s) { editStart = opinion.getSelectionStart(); editEnd = opinion.getSelectionEnd(); // 先去掉监听器,否则会出现栈溢出 opinion.removeTextChangedListener(textWatcher); if (!TextUtils.isEmpty(opinion.getText())) { String etstring = opinion.getText().toString().trim(); while (calculateLength(s.toString()) > maxLen) { s.delete(editStart - 1, editEnd); editStart--; editEnd--; Log.d("TextChanged", "editStart = " + editStart + " editEnd = " + editEnd); } } opinion.setText(s); opinion.setSelection(editStart); // 恢复监听器 opinion.addTextChangedListener(textWatcher); // end by zyf -------------------------- } private int calculateLength(String etstring) { char[] ch = etstring.toCharArray(); int varlength = 0; for (int i = 0; i < ch.length; i++) { // changed by zyf 0825 , bug 6918,加入中文标点范围 , TODO 标点范围有待具体化 if ((ch[i] >= 0x2E80 && ch[i] <= 0xFE4F) || (ch[i] >= 0xA13F && ch[i] <= 0xAA40) || ch[i] >= 0x80) { // 中文字符范围0x4e00 0x9fbb varlength = varlength + 2; } else { varlength++; } } Log.d("TextChanged", "varlength = " + varlength); // 这里也可以使用getBytes,更准确嘛 // varlength = etstring.getBytes(CharSet.forName("GBK")).lenght;// 编码根据自己的需求,注意u8中文占3个字节... return varlength; } };
titleTv.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) { } @Override public void afterTextChanged(Editable s) { // add by zyf 0825 . 多余的从新输入的位置删除,而不是最后 editStart = opinion.getSelectionStart(); editEnd = opinion.getSelectionEnd(); if (!TextUtils.isEmpty(titleTv.getText())) { int varlength = 0; int size = 0; String etstring = titleTv.getText().toString().trim(); char[] ch = etstring.toCharArray(); for (int i = 0; i < ch.length; i++) { size++; if (ch[i] >= 0x4e00 && ch[i] <= 0x9fbb) { varlength = varlength + 2; } else varlength++; if (varlength > 80) { break; } } if (varlength > 80) { s.delete(size - 1, etstring.length()); // add by zyf 0825 . 多余的从新输入的位置删除,而不是最后 // s.delete(editStart - 1, editEnd); // crash stackoverflow,解决方法参考上边方案一 } } } });