16

是否可以startActivityForResult()从非活动类调用以获取结果?

场景是这样的:

我有一个类NonActivity(它不是从 Activity 派生的,因为它不是 UI)。这个类将有一堆功能(基本上是步骤)来运行。其中一个步骤需要显示 UI(Activity),然后获取结果(用户输入内容)。然后能够在接下来的步骤中使用该数据。

由于我没有 UI 组件,如何在不从活动类派生的情况下实现这一点?另外,由于我不想从活动类派生,这意味着我不能覆盖OnActivityResult(). 结果实际上来自哪里?

4

4 回答 4

23

startActivityForResult()只能从真正的屏幕活动中获得,因为它是 Activity 中的一种方法。请重新设计您的应用程序,以便用户界面由活动驱动。

另一方面,如果您的非 Activity 类是从屏幕上的 Activity 初始化和使用的,则可以将该 Activity 的实例作为构造函数中的参数传递给您的类,并使用它来启动其他 Activity。

不过要小心。使用此方法会增加内存泄漏的风险,因为外部类(在我的示例中为 Utils)可能会保留对 Activity 的引用,即使它消失了。

如果您只想访问数据,那么您可以尝试将其写入 SharedPreferences 或数据库或某些文件,然后使用应用程序上下文(再次通过构造函数传入)来读取它。这降低了内存泄漏的风险。就像是:

MyApiClass myApiClass = new MyApiClass(getApplicationContext());

示例代码

主要活动:

public class Main extends FragmentActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        Utils util = new Utils(this);
        util.startTest();

    }

    @Override
    protected void onActivityResult(int arg0, int arg1, Intent arg2) {
        Toast.makeText(this, "onActivityResult called", Toast.LENGTH_LONG).show();

        super.onActivityResult(arg0, arg1, arg2);
    }

}

Utils 类(为结果而启动):

public class Utils {

    Activity activity;

    public Utils(Activity ac) {
        activity = ac;
    }

    public void startTest() {
        Intent i = new Intent(activity, Test.class);
        activity.startActivityForResult(i, 1);

    }

}

测试活动:

public class Test extends FragmentActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Toast.makeText(this, "Test", Toast.LENGTH_LONG).show();

        this.setResult(Activity.RESULT_OK);
        this.finish();

    }

}
于 2013-01-10T20:00:02.373 回答
7

使用没有可见 GUI 的片段的类的 StartActivityForResult。您可能会在实用程序类中找到类似的内容。

请参阅下面的 runGetUserAccount。它创建自己的片段并执行 startActivityForResult。然后它有它自己的onActivityResult。

public class MyGooglePlay {

    private static final int CONNECTION_FAILURE_RESOLUTION_REQUEST = 31502;
    private ActionBarActivity activity;
    private FragmentManager fragManager;

    public MyGooglePlay(ActionBarActivity activity) {
        this.activity = activity;
        this.fragManager = activity.getSupportFragmentManager();
    }


     /**
     * Starts an activity in Google Play Services so the user can pick an
     * account
     */
    private String mEmail = "";
    static final int REQUEST_CODE_PICK_ACCOUNT = 1000;    
    public void runGetUserAccount() {
        if (TextUtils.isEmpty(mEmail)) {
            // run this code in gui less fragment so we can pickup the 
            // on activity result from inside the mygoogleplay class.
            Fragment f = new Fragment() {
                @Override
                public void onAttach(Activity activity) {
                    super.onAttach(activity);
                    String[] accountTypes = new String[] { "com.google" };
                    Intent intent = AccountPicker.newChooseAccountIntent(null,
                            null, accountTypes, false, null, null, null, null);
                    startActivityForResult(intent, REQUEST_CODE_PICK_ACCOUNT);
                }

                @Override
                public void onActivityResult(int requestCode, int resultCode,
                        Intent data) {
                    if (requestCode == REQUEST_CODE_PICK_ACCOUNT) {
                        if (resultCode == Activity.RESULT_OK) {
                            set_Email(data
                                    .getStringExtra(AccountManager.KEY_ACCOUNT_NAME));
                            // getUsername();
                        }
                    super.onActivityResult(requestCode, resultCode, data);
                }

                //this is to verify the fragment has been removed.
                //you can log or put a breakpoint to verify
                @Override public void onDestroy(){
                    super.onDestroy();
                }
            };
            FragmentTransaction fragmentTransaction = this.fragManager
                    .beginTransaction();
            fragmentTransaction.add(f, "getusername");
            fragmentTransaction.commit();
        }
    }

    /**
     * @param mEmail
     *            the mEmail to set
     */
    private void set_Email(String mEmail) {
        this.mEmail = mEmail;
        if (!TextUtils.isEmpty(mEmail)) {
            // TODO notify caller email is ready;
            // activity.onEmailReady(mEmail);
        }
        //we are done with the "getusername" fragment
        Fragment f = fragManager.findFragmentByTag("getusername");
        if (f!=null) {
            fragManager.beginTransaction().remove(f).commit();
        }
    }
}
于 2014-10-06T20:04:51.160 回答
0

您应该将上下文作为活动传递,然后您将获得解决方案。试试下面的代码。它会工作

在非活动类

public class nonActivity {

public static void method(Activity activity)
 {
     Intent intent = new Intent(activity, SecondActivity.class);
     activity. startActivityForResult(intent, REQUEST_CODE);
 }
}

在SecondActivity

 Intent intent = getIntent();
 intent.putExtra("data", "data");   //here u can pass data to previous activity
 setResult(RESULT_OK, intent);
 finish();

在第一个活动中

protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
    try {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == REQUEST_CODE && resultCode == RESULT_OK) {
            String status = data.getStringExtra("data");
            //Do what u want with data
        }
    } catch (Exception e) {
        System.out.println("=====Exception=====" + e.toString());
    }
}
于 2019-12-10T13:17:14.263 回答
-1

如果您希望将活动的结果返回到您的普通类,假设它是一个带有自定义适配器的类。

  1. 您不能使用 startActivityForResult,因为您不在活动中
  2. 我所做的是,我有意识地从课堂上发起了活动。然后我计算或做了我必须做的。从这个活动中,我将信息发送到使用方法 MainActivity.the_method() 假定的主类,并且在主活动中我更改了自定义适配器,完成了我必须使用适配器对象并调用 adapter.getItem(position)

希望这可以给你一个想法

于 2014-12-01T09:50:12.427 回答