У меня есть два класса. Первый - это активность, второй - фрагмент, где у меня есть какой-то EditText. В деятельности у меня есть подкласс с async-task и в методе doInBackground я получаю некоторый результат, который сохраняю в переменную. Как я могу отправить эту переменную из подкласса "моя деятельность" в этот фрагмент?
Отправить данные из активности на фрагмент в Android
Ответ 1
Из операции вы отправляете данные с намерением как:
Bundle bundle = new Bundle();
bundle.putString("edttext", "From Activity");
// set Fragmentclass Arguments
Fragmentclass fragobj = new Fragmentclass();
fragobj.setArguments(bundle);
и в методе Fragment onCreateView:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    String strtext = getArguments().getString("edttext");    
    return inflater.inflate(R.layout.fragment, container, false);
}
Ответ 2
Также вы можете получить доступ к данным активности из фрагмента:
Активность:
public class MyActivity extends Activity {
    private String myString = "hello";
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my);
        ...
    }
    public String getMyData() {
        return myString;
    }
}
Фрагмент:
public class MyFragment extends Fragment {
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        MyActivity activity = (MyActivity) getActivity();
        String myDataFromActivity = activity.getMyData();
        return view;
    }
}
Ответ 3
Я нашел много ответов здесь @stackoverflow.com, но определенно это правильный ответ:
"Отправка данных из активности в фрагмент в android".
Активность:
        Bundle bundle = new Bundle();
        String myMessage = "Stackoverflow is cool!";
        bundle.putString("message", myMessage );
        FragmentClass fragInfo = new FragmentClass();
        fragInfo.setArguments(bundle);
        transaction.replace(R.id.fragment_single, fragInfo);
        transaction.commit();
Фрагмент:
Чтение значения в фрагменте
        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        Bundle bundle = this.getArguments();
        String myValue = bundle.getString("message");
        ...
        ...
        ...
        }
или просто
        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        String myValue = this.getArguments().getString("message");
        ...
        ...
        ...
        }
Ответ 4
Этот ответ может быть слишком поздно. но это будет полезно для будущих читателей.
У меня есть несколько критериев. Я закодировал для выбора файла от намерения. и выбранный файл для передачи в конкретный фрагмент для дальнейшего процесса. У меня есть много фрагментов, имеющих функциональность выбора файлов. в то время каждый раз проверять условие и получать фрагмент и передавать значение довольно отвратительно. Итак, я решил передать значение с помощью интерфейса.
Шаг 1: Создайте интерфейс в Главном действии.
   public interface SelectedBundle {
    void onBundleSelect(Bundle bundle);
   }
Шаг 2. Создайте ссылку SelectedBundle для той же операции
   SelectedBundle selectedBundle;
Шаг 3: создайте метод в той же операции
   public void setOnBundleSelected(SelectedBundle selectedBundle) {
       this.selectedBundle = selectedBundle;
   }
 Шаг 4: Необходимо инициализировать ссылку SelectedBundle, для которой все фрагменты нуждаются в функциональности средства выбора файлов. Вы помещаете этот код в свой фрагмент на onCreateView(..)
    ((MainActivity)getActivity()).setOnBundleSelected(new MainActivity.SelectedBundle() {
          @Override
         public void onBundleSelect(Bundle bundle) {
            updateList(bundle);
        }
     });
Шаг 5: onActivityResult из MainActivity, передайте значения фрагментам, используя интерфейс.
 @Override
 protected void onActivityResult(int requestCode, int resultCode, Intent  data) {
       selectedBundle.onBundleSelect(bundle);
  }
Это все. Реализуйте каждый фрагмент, который вам нужен, в FragmentClass. Ты замечательный. ты сделал. ВОТ ЭТО ДА...
Ответ 5
Основная идея использования фрагментов (F) заключается в создании многократно используемых компонентов пользовательского интерфейса в приложениях для Android. Эти фрагменты содержатся в действиях, и есть общий (лучший) способ создания путей пути связи от A → F и F-A. Необходимо обмениваться между F-F посредством Activity, потому что тогда только фрагменты становятся развязанными и самоподдерживающимися.
Таким образом, передача данных из A → F будет такой же, как объясняется ρяσѕρєя K. В дополнение к этому ответу. После создания фрагментов внутри Activity мы также можем передавать данные методам вызова фрагментов в фрагментах,
Например:
    ArticleFragment articleFrag = (ArticleFragment)
                    getSupportFragmentManager().findFragmentById(R.id.article_fragment);
    articleFrag.updateArticleView(position);
Ответ 6
Лучший и удобный подход - это вызов экземпляра фрагмента и отправка данных в это время. каждый фрагмент по умолчанию имеет метод экземпляра
Например: если ваше имя фрагмента MyFragment
поэтому вы будете называть свой фрагмент из действия следующим образом:
getSupportFragmentManager().beginTransaction().add(R.id.container, MyFragment.newInstance("data1","data2"),"MyFragment").commit();
* R.id.container - это идентификатор моего FrameLayout
поэтому в MyFragment.newInstance( "data1", "data2" ) вы можете отправить данные в фрагмент, а в вашем фрагменте вы получите эти данные в MyFragment newInstance (String param1, String param2 )
public static MyFragment newInstance(String param1, String param2) {
        MyFragment fragment = new MyFragment();
        Bundle args = new Bundle();
        args.putString(ARG_PARAM1, param1);
        args.putString(ARG_PARAM2, param2);
        fragment.setArguments(args);
        return fragment;
    }
а затем в onCreate методе фрагмента вы получите данные:
@Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        if (getArguments() != null) {
            mParam1 = getArguments().getString(ARG_PARAM1);
            mParam2 = getArguments().getString(ARG_PARAM2);
        }
    }
так что теперь mParam1 имеют data1 и mParam2 имеют данные2
теперь вы можете использовать этот mParam1 и mParam2 в своем фрагменте.
Ответ 7
Если вы передадите ссылку на фрагмент (конкретный подкласс) в задачу async, вы можете напрямую обратиться к фрагменту.
Некоторые способы передачи ссылки на фрагмент в асинхронную задачу:
-  Если ваша задача async является полноценным классом (class FooTask extends AsyncTask), передайте фрагмент в конструктор.
- Если ваша задача async является внутренним классом, просто объявите конечную переменную Fragment в области, заданной задачей async, или как поле внешнего класса. Вы сможете получить доступ к этому из внутреннего класса.
Ответ 8
Я хотел бы добавить для начинающих, что разница между двумя наиболее часто задаваемыми ответами здесь заключается в различном использовании фрагмента.
Если вы используете фрагмент в классе java, где у вас есть данные, которые вы хотите передать, вы можете применить первый ответ для передачи данных:
Bundle bundle = new Bundle();
bundle.putString("edttext", "From Activity");
Fragmentclass fragobj = new Fragmentclass();
fragobj.setArguments(bundle);
Если, однако, вы используете, например, код по умолчанию, предоставленный Android Studio для фрагментов с вкладками, этот код работать не будет.
Это не будет работать, даже если вы замените PlaceholderFragment по умолчанию на FragmentClasses, и даже если вы исправите FragmentPagerAdapter в новой ситуации, добавив переключатель для getItem() и другой переключатель для getPageTitle() (как показано здесь)
Предупреждение: упомянутый выше клип содержит ошибки кода, которые я объясню позже, но полезно посмотреть, как вы переходите от кода по умолчанию к редактируемому коду для фрагментов с вкладками)! Остальная часть моего ответа имеет гораздо больше смысла, если учесть java-классы и xml файлы из этого клипа (представитель для первого использования фрагментов с вкладками в сценарии новичка).
Основная причина, по которой наиболее одобренный ответ на этой странице не сработает, заключается в том, что в этом коде по умолчанию для фрагментов с вкладками фрагменты используются в другом Java-классе: FragmentPagerAdapter!
Таким образом, для отправки данных у вас возникает желание создать пакет в MotherActivity и передать его в FragmentPagerAdapter, используя ответ № 2.
Только это снова не так. (Возможно, вы могли бы сделать это так, но это просто осложнение, которое на самом деле не нужно).
Я думаю, что правильный/более простой способ сделать это - передать данные непосредственно в рассматриваемый фрагмент, используя ответ № 2. Да, между деятельностью и фрагментом будет жесткая связь, НО, для фрагментов с вкладками, что является своего рода ожидаемым. Я бы даже посоветовал вам создать фрагменты с вкладками внутри Java-класса MotherActivity (в виде подклассов, так как они никогда не будут использоваться вне MotherActivity) - это просто, просто добавьте в Java-класс MotherActivity столько фрагментов, сколько вам нужно, вот так:
 public static class Tab1 extends Fragment {
    public Tab1() {
    }
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {
        View rootView = inflater.inflate(R.layout.your_layout_name_for_fragment_1, container, false);
        return rootView;
    }
}.
Итак, чтобы передать данные из MotherActivity в такой фрагмент, вам нужно создать частные строки/связки выше onCreate вашей деятельности Mother, которые вы можете заполнить данными, которые хотите передать фрагментам, и передать их через метод, созданный после onCreate (здесь он называется getMyData()).
public class MotherActivity extends Activity {
    private String out;
    private Bundle results;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_mother_activity);
       // for example get a value from the previous activity
        Intent intent = getIntent();
        out = intent.getExtras().getString("Key");
    }
    public Bundle getMyData() {
        Bundle hm = new Bundle();
        hm.putString("val1",out);
        return hm;
    }
}
А затем в классе фрагмента вы используете getMyData:
public static class Tab1 extends Fragment {
        /**
         * The fragment argument representing the section number for this
         * fragment.
         */
        public Tab1() {
        }
        @Override
        public View onCreateView(LayoutInflater inflater, ViewGroup container,
                                 Bundle savedInstanceState) {
            View rootView = inflater.inflate(R.layout.your_layout_name_for_fragment_1, container, false);
            TextView output = (TextView)rootView.findViewById(R.id.your_id_for_a_text_view_within_the_layout);
            MotherActivity activity = (MotherActivity)getActivity();
            Bundle results = activity.getMyData();
            String value1 = results.getString("val1");
            output.setText(value1);
            return rootView;
        }
    }
Если у вас есть запросы к базе данных, я советую вам делать их в MotherActivity (и передавать их результаты в виде строк/целых чисел, прикрепленных к ключам внутри пакета, как показано выше), как внутри фрагментов с вкладками, ваш синтаксис станет более сложным ( например, это становится getActivity(), а getIntent становится getActivity(). getIntent), но у вас также есть возможность сделать то, что вы хотите.
Мой совет для начинающих - сосредоточиться на маленьких шагах. Во-первых, получите намерение открыть очень простое действие с вкладками, не передавая НИКАКИХ данных. Это работает? Открывает ли он ожидаемые вами вкладки? Если нет, то почему?
Начните с этого, и применив решения, подобные представленным в этом ролике, посмотрите, чего не хватает. Для этого конкретного клипа файл mainactivity.xml никогда не отображается. Это, безусловно, смущает вас. Но если вы обратите внимание, вы увидите, что, например, контекст (tools: context) неверен в файлах фрагментов xml. Каждый фрагмент XML должен указывать на правильный класс фрагмента (или подкласс, используя разделитель $).
Вы также увидите, что в основной Java-класс активности вам нужно добавить tabLayout.setupWithViewPager(mViewPager) - сразу после строки TabLayout tabLayout = (TabLayout) findViewById (R.id.tabs); без этой строки ваше представление фактически не связано с XML файлами фрагментов, но показывает ТОЛЬКО XML файл основного действия.
В дополнение к строке в Java-классе основного действия в XML файле основного действия необходимо изменить вкладки в соответствии с вашей ситуацией (например, добавить или удалить TabItems). Если у вас нет вкладок в основном XML-задании, возможно, вы не выбрали правильный тип действия при первом его создании (новое действие - действие с вкладками).
Обратите внимание, что в последних 3 параграфах я говорю о видео! Поэтому, когда я говорю основной вид деятельности XML, это основной вид деятельности XML в видео, который в вашей ситуации является XML файлом MotherActivity.
Ответ 9
Очень старый пост, все же я осмелюсь добавить небольшое объяснение, которое было бы полезно для меня.
Технически вы можете напрямую установить элементы любого типа в фрагмент из действия. 
Итак, почему Bundle? 
Причина очень проста - Bundle обеспечивает единый способ обработки: 
 - создание/открытие фрагмента 
- реконфигурация (поворот экрана) - просто добавьте исходный/обновленный комплект для outState в onSaveInstanceState() 
- восстановление приложения после сбора мусора в фоновом режиме (как при реконфигурации). 
Вы можете (если вам нравятся эксперименты) создать обходное решение в простых ситуациях, но подход Bundle просто не видит разницы между одним фрагментом и одной тысячей в стоге - он остается простым и понятным. 
 Почему ответ  @Elenasys является самым элегантным и универсальным решением. 
 И почему ответ, заданный  @Martin, имеет  ловушки
Ответ 10
Из Activity вы отправляете данные с Bundle как:
Bundle bundle = new Bundle();
bundle.putString("data", "Data you want to send");
// Your fragment
MyFragment obj = new MyFragment();
obj.setArguments(bundle);
И в Fragment метод onCreateView получить данные:
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) 
{
    String data = getArguments().getString("data");// data which sent from activity  
    return inflater.inflate(R.layout.myfragment, container, false);
}
Ответ 11
лучший способ отправки данных из класса активности в фрагмент проходит через методы setter. Как 
FragmentClass fragmentClass = new FragmentClass();
fragmentClass.setMyList(mylist);
fragmentClass.setMyString(myString);
fragmentClass.setMyMap(myMap);
и легко получить эти данные из класса.
Ответ 12
Иногда вы можете получать Intent в своей деятельности, и вам нужно передать информацию в ваш рабочий фрагмент. 
Даны ответы, если вам нужно запустить фрагмент, но если он все еще работает, setArguments() не очень полезен. 
Другая проблема возникает, если переданная информация приведет к взаимодействию с вашим пользовательским интерфейсом. В этом случае вы не можете назвать что-то вроде myfragment.passData(), потому что андроид быстро сообщит, что может взаимодействовать только поток, создавший представление.
Итак, мое предложение - использовать приемник. Таким образом, вы можете отправлять данные из любого места, включая активность, но задание будет выполняться в контексте фрагмента.
В фрагменте onCreate():
protected DataReceiver dataReceiver;
public static final String REC_DATA = "REC_DATA";
@Override
public void onCreate(Bundle savedInstanceState) {
    data Receiver = new DataReceiver();
    intentFilter = new IntentFilter(REC_DATA);
    getActivity().registerReceiver(dataReceiver, intentFilter);
}
private class DataReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        int data= intent.getIntExtra("data", -1);
        // Do anything including interact with your UI
    }
}
В вашей деятельности:
// somewhere
Intent retIntent = new Intent(RE_DATA);
retIntent.putExtra("data", myData);
sendBroadcast(retIntent);
Ответ 13
Если activity необходимо сделать fragment выполнить действие после инициализации, самый простой способ - заставить activity вызвать метод в экземпляре fragment. В fragment добавьте метод:
public class DemoFragment extends Fragment {
  public void doSomething(String param) {
      // do something in fragment
  }
}
 а затем в activity, получите доступ к fragment с помощью менеджера fragment и вызовите method:
public class MainActivity extends FragmentActivity {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        DemoFragment fragmentDemo = (DemoFragment) 
            getSupportFragmentManager().findFragmentById(R.id.fragmentDemo);
        fragmentDemo.doSomething("some param");
    }
}
 а затем activity может напрямую связываться с fragment, вызвав это method.
Ответ 14
Вы можете создать открытый статический метод в фрагменте, где вы получите статическую ссылку на этот фрагмент, а затем передаете данные этой функции и установите эти данные в аргумент в том же методе и получите данные через getArgument on oncreate метод фрагмента и установите данные к локальным переменным.
Ответ 15
Самый умный и проверенный способ передачи данных между фрагментами и активностью - это создание переменных, например:
class StorageUtil {
  public static ArrayList<Employee> employees;
}
Затем, чтобы передать данные из фрагмента в активность, мы делаем это в методе onActivityCreated:
//a field created in the sending fragment
ArrayList<Employee> employees;
@Override
    public void onActivityCreated(@Nullable Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
         employees=new ArrayList();
       //java 7 and above syntax for arraylist else use employees=new ArrayList<Employee>() for java 6 and below
     //Adding first employee
        Employee employee=new Employee("1","Andrew","Sam","1984-04-10","Male","Ghanaian");
        employees.add(employee);
      //Adding second employee
       Employee employee=new Employee("1","Akuah","Morrison","1984-02-04","Female","Ghanaian");
         employees.add(employee);
        StorageUtil.employees=employees;
    }
Теперь вы можете получить значение StorageUtil.employees со всего мира. Гудлак!
Ответ 16
Используйте следующий интерфейс для связи между активностью и фрагментом
public interface BundleListener {
    void update(Bundle bundle);
    Bundle getBundle();
}
Или используйте следующий универсальный прослушиватель для двусторонней связи с использованием интерфейса
 /**
 * Created by Qamar4P on 10/11/2017.
 */
public interface GenericConnector<T,E> {
    T getData();
    void updateData(E data);
    void connect(GenericConnector<T,E> connector);
}
метод показа фрагмента
public static void show(AppCompatActivity activity) {
        CustomValueDialogFragment dialog = new CustomValueDialogFragment();
        dialog.connector = (GenericConnector) activity;
        dialog.show(activity.getSupportFragmentManager(),"CustomValueDialogFragment");
    }
вы также можете указать свой контекст на GenericConnector в onAttach(Context)
в вашей деятельности
CustomValueDialogFragment.show(this);
в вашем фрагменте
...
@Override
    public void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        connector.connect(new GenericConnector() {
            @Override
            public Object getData() {
                return null;
            }
            @Override
            public void updateData(Object data) {
            }
            @Override
            public void connect(GenericConnector connector) {
            }
        });
    }
...
    public static void show(AppCompatActivity activity, GenericConnector connector) {
            CustomValueDialogFragment dialog = new CustomValueDialogFragment();
            dialog.connector = connector;
            dialog.show(activity.getSupportFragmentManager(),"CustomValueDialogFragment");
        }
Примечание. Никогда не используйте его как "".toString().toString().toString();.
Ответ 17
Мое решение - написать статический метод внутри фрагмента:
public TheFragment setData(TheData data) {
    TheFragment tf = new TheFragment();
    tf.data = data;
    return tf;
}
Таким образом, я уверен, что все данные, которые мне нужны, находятся внутри фрагмента перед любой другой возможной операцией, которая могла бы работать с ней. Также он выглядит более чистым, на мой взгляд.
Ответ 18
Проверьте этот ответ, если вы хотите передать данные фрагменту после создания fooobar.com/questions/35125/...
Ответ 19
Я столкнулся с подобной проблемой при использовании новейшего компонента архитектуры навигации. Попробовал весь вышеупомянутый код с передачей пакета из моей деятельности по вызову во Fragment.
Наилучшим решением, следуя последним тенденциям развития Android, является использование View Model (входит в состав Android Jetpack).
Создайте и инициализируйте класс ViewModel в родительском Activity. Обратите внимание, что эта ViewModel должна быть общей для действия и фрагмента.
Теперь внутри onViewCreated() фрагмента инициализируйте Same ViewModel и настройте Observers для прослушивания полей ViewModel.
Вот полезный, подробный учебник, если вам нужно.
Ответ 20
только что наткнулся на этот вопрос, в то время как большинство методов, описанных выше, будут работать. Я просто хочу добавить, что вы можете использовать Event Bus Library, особенно в сценариях, где компонент (Activity или Fragment) не был создан, он подходит для всех размеров проектов Android и многих вариантов использования. Я лично использовал его в нескольких проектах на PlayStore.
Ответ 21
Добавление к ответу ρяσѕρєя K. При настройке аргумента необходимо поддерживать последовательность выполнения операторов. Например, в последовательности ответов есть:
Bundle bundle = new Bundle();
bundle.putString("edttext", "From Activity");
// set Fragmentclass Arguments
Fragmentclass fragobj = new Fragmentclass();
fragobj.setArguments(bundle);
Но если вы сначала попытаетесь выделить Фрагмент, а затем выделите объект Bundle и попытайтесь установить его как аргумент выделенного фрагмента, его иногда можно принять за значение null в методе onCreateView внутри фрагмента. Например, используя нижеприведенный сегмент кода, иногда getArguments внутри onCreateView могут возвращать значение null.
Отказоустойчивый сегмент кода:
Fragmentclass fragobj = new Fragmentclass();    
Bundle bundle = new Bundle();
bundle.putString("edttext", "From Activity");
// set Fragmentclass Arguments
fragobj.setArguments(bundle);
Я столкнулся с той же проблемой довольно давно, но когда я переключил порядок высказываний, моя проблема была решена. Было бы здорово, если бы кто-то мог дать твердую причину такого поведения. Скорее всего, это зависит от времени выполнения. Более опытный человек должен объяснить причину этого.
Ответ 22
В вашей деятельности объявите статическую переменную
public static HashMap<String,ContactsModal> contactItems=new HashMap<String, ContactsModal>();
Тогда в вашем фрагменте сделайте как следуйте
ActivityName.contactItems.put(Number,contactsModal);
