Navigation(3)

本文详细介绍了如何在Android应用中利用ViewModel和LiveData组件进行数据状态管理,包括创建MyViewModel类,定义数据操作函数,以及在MasterFragment和DetailFragment中通过Navigation进行页面跳转和数据绑定。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在这里插入图片描述
在这里插入图片描述
APP架构:一个Activity通过Navigation导航到多个fragment

示例:
在这里插入图片描述

//MyViewModel
//创建MyViewModel类,在类内定义LiveData
//定义数据操作的函数
public class MyViewModel extends ViewModel  {
    private MutableLiveData<Integer>number;

    public MutableLiveData<Integer>getNumber(){
        if(number == null){
            number = new MutableLiveData<>();
            number.setValue(0);
        }
        return number;
    }
    public void add(int x) {
        number.setValue(number.getValue() + x);
        if(number.getValue() < 0) {
            number.setValue(0);
        }
    }
}
/**
 * 定义主页面Fgment
 * //在匿名内部类中调用需要加final修饰符
 *
 * 1.创建myViewModel对象
 * 2.关联viewmodel和activity
 * 3.创建碎片binding类对象
 * 4.binding XML
 * 5.binding 数据源
 * 6.binding 观察者
 *
 *
 */
public class MasterFragment extends Fragment {


    public MasterFragment() {
        // Required empty public constructor
    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {

        //在匿名内部类中调用需要加final修饰符
        final MyViewModel myViewModel;

        myViewModel = new ViewModelProvider(getActivity()).get(MyViewModel.class);
        FragmentMasterBinding binding;
        binding = DataBindingUtil.inflate(inflater,R.layout.fragment_master,container,false);
        binding.setData(myViewModel);
        binding.setLifecycleOwner(getActivity());

        binding.button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                NavController controller = Navigation.findNavController(v);
                controller.navigate(R.id.action_masterFragment_to_detailFragment);
            }
        });
        binding.seekBar.setProgress(myViewModel.getNumber().getValue());

        binding.seekBar.setOnSeekBarChangeListener(new SeekBar.OnSeekBarChangeListener() {
            @Override
            public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
                myViewModel.getNumber().setValue(progress);
            }

            @Override
            public void onStartTrackingTouch(SeekBar seekBar) {
                
            }

            @Override
            public void onStopTrackingTouch(SeekBar seekBar) {

            }
        });
        return binding.getRoot();
    }
}
//同理,通过设置viewmodel,livedata,binding后
// 数据的操作在View Model类内,Fgment内只留下导航代码,控件代码
public class DetailFragment extends Fragment {

    MyViewModel myViewModel ;

    public DetailFragment() {
        // Required empty public constructor
    }


    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
                             Bundle savedInstanceState) {

        myViewModel = new ViewModelProvider(getActivity()).get(MyViewModel.class);
        FragmentDetailBinding binding;
        binding = DataBindingUtil.inflate(inflater,R.layout.fragment_detail,container,false);
        binding.setData(myViewModel);
        binding.setLifecycleOwner(getActivity());

        binding.button4.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                NavController navController = Navigation.findNavController(v);
                navController.navigate(R.id.action_detailFragment_to_masterFragment);
            }
        });
        return binding.getRoot();


    }
}
### TurtleBot3Navigation2集成及使用 #### 集成背景 为了使机器人能够在未知环境中自主导航,通常会采用SLAM(Simultaneous Localization And Mapping)技术来构建地图并定位自身位置。对于已知环境,则可以直接加载预先创建的地图文件,并基于此实现路径规划和避障功能[^1]。 #### 安装依赖包 确保安装了必要的ROS 2软件包,这些软件包包含了支持TurtleBot3以及Navigation2所需的一切工具和服务: ```bash sudo apt update && sudo apt install ros-humble-navigation2 \ ros-humble-nav2-bringup \ ros-humble-tb3-description \ ros-humble-slam-toolbox ``` #### 启动基本设置 启动Gazebo仿真器中的世界模型,并初始化TurtleBot3的相关配置: ```bash source /opt/ros/humble/setup.bash export TURTLEBOT3_MODEL=waffle_pi ros2 launch turtlebot3_gazebo turtlebot3_world.launch.py ``` #### 加载静态地图 如果已经有一个现成的地图文件(例如`my_map.yaml`),可以通过以下命令将其加载到内存中供后续操作使用: ```bash ros2 launch nav2_bringup bringup_launch.py map:=<path_to_your_map>/my_map.yaml use_sim_time:=True params_file:=<path_to_params>/nav2_params.yaml ``` 此处需要注意的是参数文件(`params_file`)的选择应该匹配具体的硬件平台特性,比如传感器布局、运动学约束条件等[^4]。 #### 实施自动导航任务 当一切准备就绪之后,就可以向目标点发送动作请求以执行移动指令了。这里给出一段Python脚本作为示范,用于设定目的地坐标(x,y),并通过客户端接口提交给导航栈处理: ```python import rclpy from geometry_msgs.msg import PoseStamped from action_tutorials_interfaces.action import NavigateToPose def send_goal(client_node, goal_pose): client = ActionClient(client_node, NavigateToPose, &#39;navigate_to_pose&#39;) while not client.wait_for_server(timeout_sec=1.0): print(&#39;Waiting for server...&#39;) goal_msg = NavigateToPose.Goal() goal_msg.pose.header.frame_id = "map" goal_msg.pose.pose.position.x = float(goal_pose[&#39;x&#39;]) goal_msg.pose.pose.position.y = float(goal_pose[&#39;y&#39;]) future = client.send_goal_async(goal_msg) rclpy.init(args=None) node = rclpy.create_node(&#39;simple_navigation_client&#39;) try: target_position = {&#39;x&#39;: 1.0, &#39;y&#39;: -2.0} send_goal(node, target_position) except Exception as e: node.get_logger().error(f&#39;Failed to reach goal position: {e}&#39;) finally: node.destroy_node() rclpy.shutdown() ``` 这段代码展示了如何通过Action API发起一次前往指定坐标的航行请求,在实际应用当中还可以进一步扩展逻辑以便更好地适应复杂场景下的需求变化[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值