Android片段生命周期:onResume调用了两次

Posted

技术标签:

【中文标题】Android片段生命周期:onResume调用了两次【英文标题】:Android fragment lifecycle: onResume called twice 【发布时间】:2017-10-16 21:16:12 【问题描述】:

在用户切换到另一个活动然后返回到当前活动后,我正在尝试保存和恢复我的片段中的滚动。 这是发生的事情:

正如您在第一个片段的 onResume 进入 onPause 之后直接看到的那样,尽管用户没有做任何事情。

问题是:我的代码有什么问题?

这是我的代码:

public class DisplayNoteActivity extends Activity 

private Context mContext;
//static final int SUBACT_EDITNOTE=100;
private long m_NoteID;
private String m_sTextNoAbbrev;


@Override
protected void onCreate(Bundle savedInstanceState) 
    super.onCreate(savedInstanceState);
    try
    
        Log.v("DEBUG", "DisplayNoteActivity.onCreate()");
        setContentView(R.layout.activity_display_note);

        Bundle extras = getIntent().getExtras();
        m_NoteID = extras.getLong("DisplayNote_NOTEID");
        m_sTextNoAbbrev = extras.getString("DisplayNote_TextNoAbbrev");

        if (savedInstanceState != null)
            RefreshFragment();
    
    catch(Exception e)
        Toast.makeText(this, e.getMessage() , Toast.LENGTH_LONG).show();
        e.printStackTrace();
    




public void onSaveInstanceState(Bundle savedInstanceState)

    Log.v("DEBUG", "DisplayNoteActivity.onSaveInstanceState()");
    super.onSaveInstanceState(savedInstanceState);
    savedInstanceState.putLong("DisplayNote_NOTEID", m_NoteID);
    savedInstanceState.putString("DisplayNote_TextNoAbbrev", m_sTextNoAbbrev);


public void onRestoreInstanceState(Bundle savedInstanceState) 
    super.onRestoreInstanceState(savedInstanceState);
    // Restore UI state from the savedInstanceState.
    // This bundle has also been passed to onCreate.
    Log.v("DEBUG", "DisplayNoteActivity.onRestoreInstanceState()");
    m_NoteID = savedInstanceState.getLong("DisplayNote_NOTEID");
    m_sTextNoAbbrev= savedInstanceState.getString("DisplayNote_TextNoAbbrev");


@Override
public void onResume() 
    Log.v("DEBUG", "DisplayNoteActivity.onResume()");
    super.onResume();

    RefreshFragment();



private void RefreshFragment()


    Bundle bundle = new Bundle();
    bundle.putLong("DisplayNote_NOTEID", m_NoteID);
    bundle.putString("DisplayNote_TextNoAbbrev", m_sTextNoAbbrev);

    android.app.FragmentManager fm=getFragmentManager();
    android.app.FragmentTransaction ft = fm.beginTransaction();

    DisplayNoteFragment dnf = new DisplayNoteFragment();
    dnf.setArguments(bundle);
    ft.replace(R.id.container, dnf);

    ft.commit();

片段代码:

public class DisplayNoteFragment extends Fragment 


private Context mContext;

private long mNoteID = 0;
private Note m_NoteData;

private long mTextID = 0;
private int m_levels = 0;

private String m_dbtype;
private String m_dblang;
GitaTextsHelper m_tdb;

private String m_sTextNo;
private ScrollView m_Scroll;
private int mScrollPos=0;

AutoResizeTextView m_atvHeaderTitle;
AutoResizeTextView m_atvHeaderSubTitle;

AutoResizeTextView m_atvBookTitle;
AutoResizeTextView m_atvChapterTitle;
TextView m_atvTags;
AutoResizeTextView m_atvVerse;

TextView m_tvGitaText;
ImageView m_NoteTypeIcon;
ImageView m_EditNoteIcon;
TextView m_tvNote;

int m_titlePrefixID;


long m_TextRowID = -1;
String m_sVerse;

public DisplayNoteFragment() 


@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
                         Bundle savedInstanceState) 
    View rootView = inflater.inflate(R.layout.fragment_display_note, container, false);
    Log.v("DEBUG", "DisplayNoteFragment onCreateView()");
    try 
        m_Scroll = (ScrollView)rootView.findViewById(R.id.ScrollView02);
        m_atvHeaderTitle = (AutoResizeTextView) rootView.findViewById(R.id.header_title);
        m_atvHeaderSubTitle = (AutoResizeTextView) rootView.findViewById(R.id.header_subtitle);

        m_atvBookTitle = (AutoResizeTextView) rootView.findViewById(R.id.book_title);
        m_atvChapterTitle = (AutoResizeTextView) rootView.findViewById(R.id.chapter_title);
        m_atvTags = (TextView) rootView.findViewById(R.id.note_tags);
        m_atvVerse = (AutoResizeTextView) rootView.findViewById(R.id.note_verse);

        m_tvGitaText = (TextView) rootView.findViewById(R.id.gita_text);
        m_tvNote = (TextView) rootView.findViewById(R.id.note_text);

        //***** INPUT
        mNoteID = getArguments().getLong("DisplayNote_NOTEID");
        m_NoteData = MyApp.mUserDB.GetNote(mNoteID);
        // common fields for text notes and custom notes
        m_atvHeaderSubTitle.setText(m_NoteData.sCreatedDate);
        m_NoteTypeIcon = (ImageView) rootView.findViewById(R.id.note_type_icon); m_NoteTypeIcon.setImageResource(MyApp.getNoteTypeIcon(m_NoteData.type));
        //tags
        String tags = MyApp.mUserDB.getNoteTags(mNoteID);

        if (tags.equals("")) tags = MyApp.Res.getString(R.string.no_tag_defined);
        m_atvTags.setText(tags);
        m_atvTags.setPaintFlags(m_atvTags.getPaintFlags() | Paint.UNDERLINE_TEXT_FLAG);
        m_atvTags.setOnClickListener(new View.OnClickListener() 
            public void onClick(View v) 
                if (MyApp.mC.hasFlag(Cookies.FL_Tagging, mContext, true)) 
                    Intent intent = new Intent(mContext, TagEdit.class);
                    intent.putExtra("TAGEDIT_NOTEID", mNoteID);
                    intent.putExtra("TAGEDIT_EDIT_TAGS_MODE", 1);

                    startActivity(intent);
                

            
        );


     catch (Exception e) 
        Toast.makeText(mContext, e.getMessage(), Toast.LENGTH_LONG).show();
        e.printStackTrace();
    

    return rootView;


private void openText(int iScroll) 

    Intent intent = new Intent(mContext, DisplayText.class);
    intent.putExtra("rowID", m_TextRowID);
    intent.putExtra("Scroll", iScroll);
    intent.putExtra("Language", m_dblang);
    intent.putExtra("DBType", m_dbtype);
    startActivity(intent);



public void onPause() 
    super.onPause();

    mScrollPos=m_Scroll.getScrollY();
    Log.v("DEBUG", "DisplayNoteFragment onPause(). Scroll="+mScrollPos);



public void onResume() 
    Log.v("DEBUG", "DisplayNoteFragment onResume(). Scroll="+mScrollPos);
    super.onResume();
    if(mScrollPos!=0) 
        m_Scroll.scrollTo(0, mScrollPos);
        Log.v("DEBUG", "DisplayNoteFragment onResume(). Scrolled to="+mScrollPos);
    



@Override
public void onSaveInstanceState(Bundle outState) 
    super.onSaveInstanceState(outState);
    if(outState==null) return;
    Log.v("DEBUG", "DisplayNoteFragment onSaveInstanceState(). Scroll="+mScrollPos);
    outState.putInt("mScrollPos", mScrollPos);



@Override
public void onActivityCreated(Bundle savedInstanceState) 
    super.onActivityCreated(savedInstanceState);
    if(savedInstanceState==null) return;
    mScrollPos = savedInstanceState.getInt("mScrollPos");
    Log.v("DEBUG", "DisplayNoteFragment onActivityCreated(). Scroll="+mScrollPos);

【问题讨论】:

因为 RefreshFragment();你是从 Onresume 打来的 我猜他的意图是刷新所有事务上的片段 UI 元素 在用户编辑数据后,我找不到更好的方法来刷新片段字段中的数据。这就是我从活动的 onResume() 调用 RefreshFragment() 的原因。你能指出我正确的做法吗?到目前为止无法谷歌... 将 RefreshFragment() 方法添加到“SAVE”按钮侦听器(如果您在某处有的话),当用户对字段进行一些更改并点击保存按钮时,您可以刷新 UI跨度> 我没有这样的保存按钮。编辑发生在另一个活动中,它将更改保存到数据库中。回到我的 DisplayNoteActivity 后,我需要用数据库中的新数据刷新片段 【参考方案1】:

片段被替换两次,这就是片段的 onResume 被调用两次的原因。您应该只从活动的 onCreate() 调用 RefreshFragment(),而不需要从活动的 onResume() 调用。

【讨论】:

好的,在我在活动的 onResume() 中评论 RefreshFragment() 之后,滚动保存现在可以工作了。但是,在用户编辑片段的字段后,片段不会刷新。我应该怎么做才能使片段的字段更新为新值? 可以将fragment的引用保存为activity类的成员变量,通过一些公共方法更新值。 那么我应该在片段的已保存实例上调用 Activity 的 onResume() 中的 Fragment 类的哪个方法以使其重绘? 您需要重构您的代码,以便在第一次运行时使用 getArguments 中的数据,之后使用新方法中的数据进行未来更新。【参考方案2】:

首先,我认为你应该检查DisplayNoteActivityonResume() 是否也被调用了两次?如果是这样的话, 1. 检查requestPermission() 是否在DisplayNoteActivity 启动期间被调用。 2. 检查onConfigurationChanged()是否未被处理。

如果DisplayNoteActivity 中的onResume() 只被调用一次,那么请检查DisplayNoteFragment 有什么问题。

【讨论】:

谢谢。我在 onViewCreated() 中调用 requestPermission(),因此调用了 onResume(),然后调用了 onPause(),然后又调用了 onResume()。由于我已经获得了权限,所以没有显示权限对话框,但当然调用了方法, 不客气。您应该首先检查您的权限使用ActivityCompat.checkSelfPermission(),仅在未授予权限时调用ActivityCompat.requestPermissions()

以上是关于Android片段生命周期:onResume调用了两次的主要内容,如果未能解决你的问题,请参考以下文章

关于片段生命周期,何时调用片段的 onActivityResult?

了解活动生命周期

Android中的片段事务问题

导航架构片段重载问题

android的生命周期

android生命周期