33

我在这一行得到一个空指针异常:

public void hideKeyboard(){ 
InputMethodManager inputManager = (InputMethodManager)            
            this.getSystemService(Context.INPUT_METHOD_SERVICE);    
inputManager.hideSoftInputFromWindow(this.getCurrentFocus().getWindowToken(),      
            InputMethodManager.HIDE_NOT_ALWAYS);
}

这是从以下方法调用的:

@Override
public void onBackPressed() {
    super.onBackPressed();
    hideKeyboard();
}

这是唯一的活动。从片段中按下后退按钮。

堆:

09-28 19:14:40.301: E/InputEventSender(30324): Exception dispatching finished signal.
09-28 19:14:40.301: E/MessageQueue-JNI(30324): Exception in MessageQueue callback: handleReceiveCallback
09-28 19:14:40.325: E/MessageQueue-JNI(30324): java.lang.NullPointerException
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at com.example.ecohelp.MainActivity.hideKeyboard(MainActivity.java:75)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at com.example.ecohelp.MainActivity.onBackPressed(MainActivity.java:31)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.app.Activity.onKeyUp(Activity.java:2159)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.KeyEvent.dispatch(KeyEvent.java:2647)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.app.Activity.dispatchKeyEvent(Activity.java:2389)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at com.android.internal.policy.impl.PhoneWindow$DecorView.dispatchKeyEvent(PhoneWindow.java:1860)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$ViewPostImeInputStage.processKeyEvent(ViewRootImpl.java:3791)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$ViewPostImeInputStage.onProcess(ViewRootImpl.java:3774)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3379)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3429)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3398)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:3483)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:3406)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$AsyncInputStage.apply(ViewRootImpl.java:3540)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3379)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3429)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3398)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.apply(ViewRootImpl.java:3406)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.deliver(ViewRootImpl.java:3379)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.onDeliverToNext(ViewRootImpl.java:3429)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$InputStage.forward(ViewRootImpl.java:3398)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$AsyncInputStage.forward(ViewRootImpl.java:3516)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.ViewRootImpl$ImeInputStage.onFinishedInputEvent(ViewRootImpl.java:3666)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.inputmethod.InputMethodManager$PendingEvent.run(InputMethodManager.java:1982)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.inputmethod.InputMethodManager.invokeFinishedInputEventCallback(InputMethodManager.java:1698)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.inputmethod.InputMethodManager.finishedInputEvent(InputMethodManager.java:1689)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.inputmethod.InputMethodManager$ImeInputEventSender.onInputEventFinished(InputMethodManager.java:1959)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.view.InputEventSender.dispatchInputEventFinished(InputEventSender.java:141)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.os.MessageQueue.nativePollOnce(Native Method)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.os.MessageQueue.next(MessageQueue.java:132)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.os.Looper.loop(Looper.java:124)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at android.app.ActivityThread.main(ActivityThread.java:5103)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at java.lang.reflect.Method.invokeNative(Native Method)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at java.lang.reflect.Method.invoke(Method.java:525)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:737)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:553)
09-28 19:14:40.325: E/MessageQueue-JNI(30324):  at dalvik.system.NativeStart.main(Native Method)
4

14 回答 14

49

我只需要在隐藏键盘之前检查是否有焦点视图。

例如,如果您EditText的活动或片段中有一个,它可能是焦点视图。当EditText不再聚焦时,getCurrentFocus()可能会返回 null(除非其他一些视图聚焦)。

void hideKeyboard() {
    InputMethodManager inputManager = (InputMethodManager) getActivity().getSystemService(
            Context.INPUT_METHOD_SERVICE);
    View focusedView = getActivity().getCurrentFocus();
    /*
     * If no view is focused, an NPE will be thrown
     * 
     * Maxim Dmitriev
     */
    if (focusedView != null) {
        inputManager.hideSoftInputFromWindow(focusedView.getWindowToken(),
                InputMethodManager.HIDE_NOT_ALWAYS);
    }
}
于 2014-09-18T09:23:34.303 回答
34

正如 CommonsWare 所提到的,getCurrentFocus()为空,因为当前Activity中没有View组件持有焦点。

如果您的 Activity 中已有视图,请使用它来获取窗口令牌。例如,如果我有一个 Button 组件:

inputManager.hideSoftInputFromWindow(myButton.getWindowToken(), InputMethodManager.HIDE_NOT_ALWAYS);

或者更糟糕的是,如果我的 Activity 中没有任何视图,我可以这样做:

inputManager.hideSoftInputFromWindow(new View(this).getWindowToken(), InputMethodManager.HIDE_NOT_ALWAYS);

这将解决您的 NPE 问题,但我希望您发现描述有用。

关于键盘的另一件事是,当用户在键盘可见时按下后退按钮时,键盘会接收并使用后退键来隐藏自己。或者至少大多数键盘都是这样的。

于 2013-09-29T21:06:58.073 回答
26

上面的每个人都正确地指出getWindowToken()返回 null。

getCurrentFocus().getWindowToken()当我遇到同样的问题时,我正在使用默认代码隐藏键盘。

然后我意识到,由于没有 View 获得焦点,我得到了NullPointerException.

我们可以将上面的内容改为:

anyView.getWindowToken()

anyView布局中的任何视图都在哪里。

于 2014-03-10T13:50:12.193 回答
12

如果要在触摸屏幕时隐藏键盘,请使用以下代码

       @Override
public boolean onTouchEvent(MotionEvent event) {
    InputMethodManager imm = (InputMethodManager)this.getSystemService(Context.
            INPUT_METHOD_SERVICE);
    imm.hideSoftInputFromWindow(this.getWindow().getDecorView().getRootView().getWindowToken(), 0);
    return true;
}

如果您想在特定视图中执行此操作(EditText)

      public void  hideKeyBoard(EditText edt) {
    InputMethodManager imm = (InputMethodManager)this.getSystemService(Context.
            INPUT_METHOD_SERVICE);
    imm.hideSoftInputFromWindow(edt.getWindowToken(), 0);

}

或者您可以使用任何视图。

获取当前视图

    imm.hideSoftInputFromWindow(this.getWindow().getDecorView().getRootView().getWindowToken(), 0);
于 2016-09-08T07:36:01.660 回答
9

这对我有用。只需添加 getWindow().getDecorView().getRootView().getWindowToken() 即可,而不是使用 getCurrectFocus()。之后,您可以在活动中的任何地方使用此方法。

public static void hideSoftKeyboard(Activity activity) {
        InputMethodManager inputMethodManager =
                (InputMethodManager) activity.getSystemService(
                        Activity.INPUT_METHOD_SERVICE);
        inputMethodManager.hideSoftInputFromWindow(
                activity.getWindow().getDecorView().getRootView().getWindowToken(), 0);
    }
于 2018-07-19T07:58:11.133 回答
7

我遇到了 getCurrentFocus() 返回 null 的相同问题。所以这个方法对我有用,我会在我的 onClick 上调用它来隐藏键盘,如果它显示或者即使没有显示键盘也不会抛出空指针异常:

public void hiddenInputMethod() {

    InputMethodManager imm = (InputMethodManager) getSystemService(MyActivity.this.INPUT_METHOD_SERVICE);
    if (getCurrentFocus() != null)
        imm.hideSoftInputFromWindow(this.getCurrentFocus().getWindowToken(), InputMethodManager.HIDE_NOT_ALWAYS);
}
于 2015-01-13T03:22:38.173 回答
2

如果您遇到错误,请使用此选项:原因:java.lang.NullPointerException:尝试在空对象引用上调用虚拟方法 'android.os.IBinder android.view.View.getWindowToken()'


InputMethodManager inputMethodManager = (InputMethodManager) MainActivity.this.getSystemService(Activity.INPUT_METHOD_SERVICE);

   if(MainActivity.this.getCurrentFocus() != null)
        {
         inputMethodManager.hideSoftInputFromWindow(MainActivity.this.getCurrentFocus().
getWindowToken(), 0);
        }
于 2016-06-16T05:51:52.150 回答
1

我在隐藏键盘时遇到了同样的问题,我通过在触摸模式下使布局的根视图成为焦点找到了简单的解决方案。

<RelativeLayout xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:clickable="true"
android:id="@+id/rlSignActivity"
android:focusableInTouchMode="true"
xmlns:android="http://schemas.android.com/apk/res/android" >
    .......
</RelativeLayout>
于 2017-08-10T10:52:54.603 回答
1

如果 InputMethodManager 的实例为空,它会显示错误。试试下面对我有用的代码。

void hideKeyboard() {
InputMethodManager inputManager = (InputMethodManager) 
getActivity().getSystemService(
        Context.INPUT_METHOD_SERVICE);
View focusedView = getActivity().getCurrentFocus();

if (focusedView != null) {

    try{
    assert inputManager != null;
    inputManager.hideSoftInputFromWindow(focusedView.getWindowToken(),
            InputMethodManager.HIDE_NOT_ALWAYS);
}catch(AssertionError e{
     e.printStackTrace();
    }
  }
}
于 2018-05-12T19:28:09.873 回答
1

当没有关注元素时会出现问题,这个验证就足够了:

if(activity.getCurrentFocus() != null)
{
    InputMethodManager inputMethodManager =
    (InputMethodManager) activity.getSystemService(
    Activity.INPUT_METHOD_SERVICE);
    inputMethodManager.hideSoftInputFromWindow(
    activity.getCurrentFocus().getWindowToken(), 0);
}
于 2019-09-11T22:31:39.657 回答
1

主要问题在

getCurrentFocus().getWindowToken()

这里getcurrentFocus()是问题所在。通过提供驻留在当前屏幕上的有效视图可以轻松解决此问题,例如

yourButtonOnScreen.getWindowToken()

该按钮不会为空,因为它显示在您的屏幕上,因此它将解决问题。

不过,如果您的屏幕上没有任何有效视图,只需更换

getCurrentFocus().getWindowToken()...

//in case of fragment

new View(getActivity()).getWindowToken()

//in case of activity

new View(this).getWindowToken()

希望这能解决问题..祝你好运!

于 2017-07-20T10:52:48.910 回答
0

在我检查输入管理器也是!= null 后警告停止

View view = this.getCurrentFocus();
    InputMethodManager imm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
    if (view != null && imm != null) {
        imm.hideSoftInputFromWindow(view.getWindowToken(), 0);
    }
于 2020-02-18T15:29:53.150 回答
-2

只需使用这个:

public void hideKeyboard(View v){ 
  InputMethodManager imm = (InputMethodManager) context.getSystemService(Context.INPUT_METHOD_SERVICE);
  imm.hideSoftInputFromWindow(v.getWindowToken(), 0);
}
于 2014-05-27T12:35:35.740 回答
-3

尝试这个:

public void hideKeyboard(){ 
InputMethodManager inputManager = (InputMethodManager)            
            this.getSystemService(Context.INPUT_METHOD_SERVICE);    
inputManager.hideSoftInputFromWindow(this.getCurrentFocus().getWindowToken(), 0);
}
于 2013-09-28T17:19:02.077 回答