Android 调起第三方地图应用进行导航

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android 调起第三方地图应用进行导航相关的知识,希望对你有一定的参考价值。

参考技术A 最近公司项目中用到类似的功能,记录一下,权当笔记吧,能帮到其他人也算是功德一件了。

这里要着重说一下这个方法,网上这个方法有很多,但一般都是在这个finally的位置进行if-else判断,判断packageName是否为null。但是我用起来不太顺手,不知道为什么没有人说过这个问题。
  一旦程序出现异常,进入了catch语句,那之后的if判断是没有作用的。

这里提供各平台相关的地址,具体搜索:【平台名】+URI API,就可以找到想要的结果。
高德: http://lbs.amap.com/api/amap-mobile/guide/android/route
百度: http://lbsyun.baidu.com/index.php?title=uri/api/android
腾讯: http://lbs.qq.com/uri_v1/guide-route.html

各个平台基本上都有对应的Uri地址以及对应的调用示例,依样画葫芦基本上都可以实现。

具体方法名决定着跳转的页面,比如上面高德和腾讯方法为routeplan,线路规划,跳转的都是导航方式选择页面;而百度为geocoder,跳转的是目的地位置详情页面。

都是常用的一些功能,就不放图啦~就酱。

Android开发 PopupWindow弹窗调用第三方地图(百度,高德)实现导航功能

博客描述:后台返回地点的经纬度在地图上进行描点,点击导航弹出PopupWindow进行选择地图操作,如果手机中没有安装地图,提示没有,否则传值调起地图进行导航操作

看一下实现的效果,没图说再多都白搭

 

这里在打开第三方的时候可以不用传当前位置的经纬度,当你打开App时默认为当前位置为起点,只设置终点经纬度就可以,文档上也都有说明,

如果你有定位获取的经纬度,也可以拼接上,注意不要拼接错,还要注意的百度和高德地图使用的经纬度不能用同一个,需要转换,百度地图获

取到的经纬度需要通过方法处理成高德地图可以使用的,反之同理。

另外附上文档地址:

  百度地图:http://lbsyun.baidu.com/index.php?title=uri/api/android

  高德地图:http://lbs.amap.com/api/amap-mobile/guide/android/route

 

腾讯地图后续有时间就添加上。

看完效果图再看具体的实现方法

先看实现PopupWindow的实现

PopupWindow的布局文件

map_navgation_sheet.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:fitsSystemWindows="true"
    android:orientation="vertical">

    <TextView
        android:id="@+id/baidu_btn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="20px"
        android:layout_marginRight="20px"
        android:layout_marginTop="35px"
        android:background="@drawable/shape_map_navagation_sheet_bg"
        android:gravity="center"
        android:text="百度地图"
        android:textColor="#224E8F" />

    <!-- <View
         android:layout_width="match_parent"
         android:layout_height="1px"
         android:background="#cdcdcd" />-->

    <TextView
        android:id="@+id/gaode_btn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="20px"
        android:layout_marginRight="20px"
        android:layout_marginTop="10px"
        android:background="@drawable/shape_map_navagation_sheet_bg"
        android:gravity="center"
        android:text="高德地图"
        android:textColor="#224E8F" />

    <!--  <View
          android:layout_width="match_parent"
          android:layout_height="1px"
          android:background="#cdcdcd" />-->

    <TextView
        android:id="@+id/tencent_btn"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:gravity="center"
        android:paddingBottom="30px"
        android:paddingTop="30px"
        android:text="腾讯地图"
        android:visibility="gone" />


    <!--<View
        android:layout_width="match_parent"
        android:layout_height="1px"
        android:background="#cdcdcd" />-->

    <!--
        android:paddingBottom="30px"-->
    <TextView
        android:id="@+id/cancel_btn2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginBottom="40px"
        android:layout_marginLeft="20px"
        android:layout_marginRight="20px"
        android:layout_marginTop="15px"
        android:background="@drawable/shape_map_navagation_sheet_bg"
        android:gravity="center"
        android:text="取消"
        android:textColor="#224E8F" />
</LinearLayout>

PopupWindow弹出方法:

 @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    private void dhPopupView() {
        // : 2016/5/17 构建一个popupwindow的布局
//        View popupView = ShopDetailsMap.this.getLayoutInflater().inflate(R.layout.map_navagation_sheet, null);
        View popupView = getLayoutInflater().inflate(R.layout.map_navagation_sheet, null);
//        Point position = getNavigationBarSize(mContext);

        TextView baidu_btn = popupView.findViewById(R.id.baidu_btn);
        TextView gaode_btn = popupView.findViewById(R.id.gaode_btn);
        TextView tencent_btn = popupView.findViewById(R.id.tencent_btn);
        TextView cancel_btn2 = popupView.findViewById(R.id.cancel_btn2);

        baidu_btn.setOnClickListener(this);
        gaode_btn.setOnClickListener(this);
        tencent_btn.setOnClickListener(this);
        cancel_btn2.setOnClickListener(this);
        // : 2016/5/17 创建PopupWindow对象,指定宽度和高度
//        window = new PopupWindow(popupView, ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
        window = new PopupWindow();

        window.setContentView(popupView);
        //设置宽高
        window.setWidth(ViewGroup.LayoutParams.MATCH_PARENT);
        window.setHeight(ViewGroup.LayoutParams.WRAP_CONTENT);


        // : 2016/5/17 设置动画
//        window.setAnimationStyle(R.style.popup_window_anim);
        // : 2016/5/17 设置背景颜色
        window.setBackgroundDrawable(new ColorDrawable(Color.parseColor("#88323232")));
        // : 2016/5/17 设置可以获取焦点
        window.setFocusable(true);
        // : 2016/5/17 设置可以触摸弹出框以外的区域
        window.setOutsideTouchable(true);
        // :更新popupwindow的状态
        window.update();
        window.setClippingEnabled(false);
//        int windowPos[] = calculatePopWindowPos(view, windowContentViewRoot);
//        window.showAtLocation(popupView,Gravity.BOTTOM,);
//        window.showAsDropDown(popupView, Gravity.BOTTOM, 0, 0);
        Rect rect = new Rect();
        getWindow().getDecorView().getWindowVisibleDisplayFrame(rect);
        int winHeight = getWindow().getDecorView().getHeight();
        window.showAtLocation(popupView, Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, winHeight - rect.bottom);
//        window.showAsDropDown(btnPopup, 0, 20);
      /*  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
//            window.showAsDropDown(this.getWindow().getDecorView(), Gravity.BOTTOM, 0, 0);
        }*/
    }

 

检测是否安装了第三方应用

 /*检测应用是否安装*/
    private boolean isAvilible(Context context, String packageName) {
        final PackageManager packageManager = context.getPackageManager();//获取packagemanager
        List<PackageInfo> pinfo = packageManager.getInstalledPackages(0);//获取所有已安装程序的包信息
        List<String> pName = new ArrayList<String>();//用于存储所有已安装程序的包名
        //从pinfo中将包名字逐一取出,压入pName list中
        if (pinfo != null) {
            for (int i = 0; i < pinfo.size(); i++) {
                String pn = pinfo.get(i).packageName;
                pName.add(pn);
            }
        }
        return pName.contains(packageName);//判断pName中是否有目标程序的包名,有TRUE,没有FALSE
    }

 

调用百度地图方法

/*百度地图*/
    public void baiduMap() {
        if (isAvilible(this, "com.baidu.BaiduMap")) {//传入指定应用包名
            Intent il = new Intent();
            il.setData(Uri.parse("baidumap://map/direction?destination=" + lng + "," + lat + "&mode=driving"));
            startActivity(il);
        } else {//未安装
            //market为路径,id为包名
            //显示手机上所有的market商店
            Toast.makeText(this, "您尚未安装百度地图", Toast.LENGTH_LONG).show();
      //显示手机上所有的market商店
            Uri uri = Uri.parse("market://details?id=com.baidu.BaiduMap");
            intent = new Intent(Intent.ACTION_VIEW, uri);
            startActivity(intent);
} window.dismiss(); }

 

调用高德地图方法

 /*高德地图*/
    private void gaodeMap() {

        if (isAvilible(this, "com.autonavi.minimap")) {//传入指定应用包名
            Intent intent = new Intent();
            intent.setAction(Intent.ACTION_VIEW);
            intent.addCategory(Intent.CATEGORY_DEFAULT);
            intent.setPackage("com.autonavi.minimap");
            try {
//                intent = Intent.getIntent("amapuri://route/plan/?sid=BGVIS1&slat=39.92848272&slon=116.39560823&sname=A&did=BGVIS2&dlat=39.98848272&dlon=116.47560823&dname=B&dev=0&t=0");
                intent = Intent.getIntent("amapuri://route/plan/?dlat=" + lng + "&dlon=" + lat + "&d&dev=0&t=0");
                startActivity(intent);
            } catch (URISyntaxException e) {
                e.printStackTrace();
            }
        } else {//未安装
            //market为路径,id为包名
            //显示手机上所有的market商店
            Toast.makeText(this, "您尚未安装高德地图", Toast.LENGTH_LONG).show();
            Uri uri = Uri.parse("market://details?id=com.autonavi.minimap");
            intent = new Intent(Intent.ACTION_VIEW, uri);
            startActivity(intent);
        }
        window.dismiss();
    }

注意:代码中的lat和lng是后台传递过来的经纬度,百度和高德使用的时候注意放置的位置。

下面贴总的代码

/**
 * Created by dingchao on 2018/3/12.
 */

import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.graphics.Color;
import android.graphics.Rect;
import android.graphics.drawable.ColorDrawable;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.annotation.RequiresApi;
import android.view.Gravity;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.PopupWindow;
import android.widget.TextView;
import android.widget.Toast;
import android.widget.ZoomControls;

import com.baidu.mapapi.SDKInitializer;
import com.baidu.mapapi.map.BaiduMap;
import com.baidu.mapapi.map.BitmapDescriptor;
import com.baidu.mapapi.map.BitmapDescriptorFactory;
import com.baidu.mapapi.map.InfoWindow;
import com.baidu.mapapi.map.MapStatusUpdate;
import com.baidu.mapapi.map.MapStatusUpdateFactory;
import com.baidu.mapapi.map.MapView;
import com.baidu.mapapi.map.MarkerOptions;
import com.baidu.mapapi.map.OverlayOptions;
import com.baidu.mapapi.model.LatLng;import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.List;/**
 * 商铺详情查看地图的时候显示
 */
public class ShopDetailsMap extends BaseActivity implements View.OnClickListener {

    private MapView mMapView = null;
    private BaiduMap mBaiduMap = null;
    private BitmapDescriptor bitmapDescriptor;
    private ImageView iv_map_return;

    private Intent intent;
    private String lat;
    private String lng;
    private String shopName;
    private String shopAddress;
    PopupWindow window;

    /*定位*/

    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        intent = getIntent();
        lat = intent.getStringExtra("lat");
        lng = intent.getStringExtra("lng");
        shopName = intent.getStringExtra("shopName");
        shopAddress = intent.getStringExtra("shopAddress");
        //在使用SDK各组件之前初始化context信息,传入ApplicationContext
        //注意该方法要再setContentView方法之前实现
        SDKInitializer.initialize(getApplicationContext());
        setContentView(R.layout.shop_details_map);


        iv_map_return = (ImageView) findViewById(R.id.iv_map_return);
        iv_map_return.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                finish();
            }
        });




        /*显示信息*/
        //infowindow中的布局
        /*TextView tv = new TextView(ShopDetailsMap.this);
        tv.setBackgroundResource(R.mipmap.maptextbg);
        tv.setPadding(20, 10, 20, 20);
        tv.setTextColor(android.graphics.Color.WHITE);
        tv.setText("商铺名称" + "\\n" + "北京市北京南站附近");
        tv.setGravity(Gravity.LEFT);*/

        View view1 = ViewGroup.inflate(ShopDetailsMap.this, R.layout.activity_shop_detail_map, null);
        TextView textView1 = view1.findViewById(R.id.tv_shop_details_map_text1);
        TextView textView2 = view1.findViewById(R.id.tv_shop_details_map_text2);

        textView1.setText(shopName);
        textView2.setText(shopAddress);


//        bitmapDescriptor = BitmapDescriptorFactory.fromView(tv);
        bitmapDescriptor = BitmapDescriptorFactory.fromView(view1);


        //显示infowindow,-47是偏移量,使infowindow向上偏移,不会挡住marker
        //获取地图控件引用
        mMapView = (MapView) findViewById(R.id.bmapView);
        LatLng llText = new LatLng(Double.parseDouble(lng), Double.parseDouble(lat));
//        InfoWindow infoWindow = new InfoWindow(bitmapDescriptor, llText, -47, null);////不可触发点击
        InfoWindow infoWindow = new InfoWindow(bitmapDescriptor, llText, -47, new InfoWindow.OnInfoWindowClickListener() {//可触发点击
            @Override
            public void onInfoWindowClick() {
//                Toast.makeText(ShopDetailsMap.this, "被点击", Toast.LENGTH_SHORT).show();
                dhPopupView();
            }
        });

        //在地图上添加该文字对象并显示
        //获取地图控件引用*/
        mMapView = (MapView) findViewById(R.id.bmapView);
        mBaiduMap = mMapView.getMap();
        mBaiduMap.showInfoWindow(infoWindow);
        /*这里重点讲解zoomBy后面的那个浮点型变量
            大家知道百度地图一共有{"10米","20米","50米","100米","200米","500米","1千米","2千米","5千米",
            "10千米","20千米","25千米","50千米","100千米","200千米","500千米","1000千米","2000千米"}*/
        MapStatusUpdate mapStatusUpdate = MapStatusUpdateFactory.zoomBy(3);
        mBaiduMap.setMapStatus(MapStatusUpdateFactory.newLatLng(llText));//设置定位的位置在屏幕的中间位置
        mBaiduMap.animateMapStatus(mapStatusUpdate);
        //构建Marker图标,设置位置圆点
        BitmapDescriptor bitmap = BitmapDescriptorFactory
                .fromResource(R.mipmap.mapdian);
        //构建MarkerOption,用于在地图上添加Marker

        OverlayOptions option = new MarkerOptions()
                .position(llText)
                .icon(bitmap);

        // 隐藏logo
        View child = mMapView.getChildAt(1);
        if (child != null && (child instanceof ImageView || child instanceof ZoomControls)) {
            child.setVisibility(View.INVISIBLE);
        }
        //在地图上添加Marker,并显示
        mBaiduMap.addOverlay(option);


//        mBaiduMap.addOverlay(option);
//        mBaiduMap.addOverlay(textOption);
    }

    //    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    private void dhPopupView() {
        // : 2016/5/17 构建一个popupwindow的布局
//        View popupView = ShopDetailsMap.this.getLayoutInflater().inflate(R.layout.map_navagation_sheet, null);
        View popupView = getLayoutInflater().inflate(R.layout.map_navagation_sheet, null);
//        Point position = getNavigationBarSize(mContext);

        TextView baidu_btn = popupView.findViewById(R.id.baidu_btn);
        TextView gaode_btn = popupView.findViewById(R.id.gaode_btn);
        TextView tencent_btn = popupView.findViewById(R.id.tencent_btn);
        TextView cancel_btn2 = popupView.findViewById(R.id.cancel_btn2);

        baidu_btn.setOnClickListener(this);
        gaode_btn.setOnClickListener(this);
        tencent_btn.setOnClickListener(this);
        cancel_btn2.setOnClickListener(this);
        // : 2016/5/17 创建PopupWindow对象,指定宽度和高度
//        window = new PopupWindow(popupView, ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
        window = new PopupWindow();

        window.setContentView(popupView);
        //设置宽高
        window.setWidth(ViewGroup.LayoutParams.MATCH_PARENT);
        window.setHeight(ViewGroup.LayoutParams.WRAP_CONTENT);


        // : 2016/5/17 设置动画
//        window.setAnimationStyle(R.style.popup_window_anim);
        // : 2016/5/17 设置背景颜色
        window.setBackgroundDrawable(new ColorDrawable(Color.parseColor("#88323232")));
        // : 2016/5/17 设置可以获取焦点
        window.setFocusable(true);
        // : 2016/5/17 设置可以触摸弹出框以外的区域
        window.setOutsideTouchable(true);
        // :更新popupwindow的状态
        window.update();
        window.setClippingEnabled(false);
//        int windowPos[] = calculatePopWindowPos(view, windowContentViewRoot);
//        window.showAtLocation(popupView,Gravity.BOTTOM,);
//        window.showAsDropDown(popupView, Gravity.BOTTOM, 0, 0);
        Rect rect = new Rect();
        getWindow().getDecorView().getWindowVisibleDisplayFrame(rect);
        int winHeight = getWindow().getDecorView().getHeight();
        window.showAtLocation(popupView, Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, winHeight - rect.bottom);
//        window.showAsDropDown(btnPopup, 0, 20);
      /*  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
//            window.showAsDropDown(this.getWindow().getDecorView(), Gravity.BOTTOM, 0, 0);
        }*/
    }


    /*百度地图*/
    public void baiduMap() {
        if (isAvilible(this, "com.baidu.BaiduMap")) {//传入指定应用包名
            Intent il = new Intent();
            il.setData(Uri.parse("baidumap://map/direction?destination=" + lng + "," + lat + "&mode=driving"));
            startActivity(il);
        } else {//未安装
            //market为路径,id为包名
            //显示手机上所有的market商店
            Toast.makeText(this, "您尚未安装百度地图", Toast.LENGTH_LONG).show();
            Uri uri = Uri.parse("market://details?id=com.baidu.BaiduMap");
            intent = new Intent(Intent.ACTION_VIEW, uri);
            startActivity(intent);
        }
        window.dismiss();
    }


    /*高德地图*/
    private void gaodeMap() {

        if (isAvilible(this, "com.autonavi.minimap")) {//传入指定应用包名
            Intent intent = new Intent();
            intent.setAction(Intent.ACTION_VIEW);
            intent.addCategory(Intent.CATEGORY_DEFAULT);
            intent.setPackage("com.autonavi.minimap");
            try {
//                intent = Intent.getIntent("amapuri://route/plan/?sid=BGVIS1&slat=39.92848272&slon=116.39560823&sname=A&did=BGVIS2&dlat=39.98848272&dlon=116.47560823&dname=B&dev=0&t=0");
                intent = Intent.getIntent("amapuri://route/plan/?dlat=" + lng + "&dlon=" + lat + "&d&dev=0&t=0");
                startActivity(intent);
            } catch (URISyntaxException e) {
                e.printStackTrace();
            }
        } else {//未安装
            //market为路径,id为包名
            //显示手机上所有的market商店
            Toast.makeText(this, "您尚未安装高德地图", Toast.LENGTH_LONG).show();
            Uri uri = Uri.parse("market://details?id=com.autonavi.minimap");
            intent = new Intent(Intent.ACTION_VIEW, uri);
            startActivity(intent);
        }
        window.dismiss();
    }

    /*检测应用是否安装*/
    private boolean isAvilible(Context context, String packageName) {
        final PackageManager packageManager = context.getPackageManager();//获取packagemanager
        List<PackageInfo> pinfo = packageManager.getInstalledPackages(0);//获取所有已安装程序的包信息
        List<String> pName = new ArrayList<String>();//用于存储所有已安装程序的包名
        //从pinfo中将包名字逐一取出,压入pName list中
        if (pinfo != null) {
            for (int i = 0; i < pinfo.size(); i++) {
                String pn = pinfo.get(i).packageName;
                pName.add(pn);
            }
        }
        return pName.contains(packageName);//判断pName中是否有目标程序的包名,有TRUE,没有FALSE
    }


    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case baidu_btn:
//                Toast.makeText(this, "baidu", Toast.LENGTH_SHORT).show();
                baiduMap();
                break;
            case R.id.gaode_btn:
//                Toast.makeText(this, "gaode", Toast.LENGTH_SHORT).show();
                gaodeMap();
                break;
            case R.id.tencent_btn:
//                Toast.makeText(this, "tengxun", Toast.LENGTH_SHORT).show();
                break;
            case R.id.cancel_btn2:
//                Toast.makeText(this, "quxiao", Toast.LENGTH_SHORT).show();
                if (window != null) {
                    window.dismiss();
                }
                break;
        }
    }


    @Override
    protected void onDestroy() {
        super.onDestroy();
        //在activity执行onDestroy时执行mMapView.onDestroy(),实现地图生命周期管理
        mMapView.onDestroy();
    }

    @Override
    protected void onResume() {
        super.onResume();
        //在activity执行onResume时执行mMapView. onResume (),实现地图生命周期管理
        mMapView.onResume();
    }

    @Override
    protected void onPause() {
        super.onPause();
        //在activity执行onPause时执行mMapView. onPause (),实现地图生命周期管理
        mMapView.onPause();
    }


    /**
     * 启动百度App进行导航
     *
     * @param address 目的地
     * @param lat     必填 纬度
     * @param lon     必填 经度
     */
    public static void goToBaiduActivity(Context context, String address, double lat, double lon) {
        double[] doubles = gcj02_To_Bd09(lat, lon);
        //启动路径规划页面
        Intent naviIntent = new Intent("android.intent.action.VIEW", android.net.Uri.parse("baidumap://map/direction?origin=" + doubles[0] + "," + doubles[1] + "&destination=" + address + "&mode=driving"));
        context.startActivity(naviIntent);
    }

    /**
     * 火星坐标系 (GCJ-02) 与百度坐标系 (BD-09) 的转换算法 将 GCJ-02 坐标转换成 BD-09 坐标
     *
     * @param lat
     * @param lon
     */
    public static double[] gcj02_To_Bd09(double lat, double lon) {
        double x = lon, y = lat;
//        double z = Math.sqrt(x * x + y * y) + 0.00002 * Math.sin(y * x_pi);
        double z = Math.sqrt(x * x + y * y) + 0.00002;
//        double theta = Math.atan2(y, x) + 0.000003 * Math.cos(x * x_pi);
        double theta = Math.atan2(y, x) + 0.000003;
        double tempLat = z * Math.sin(theta) + 0.006;
        double tempLon = z * Math.cos(theta) + 0.0065;
        double[] gps = {tempLat, tempLon};
        return gps;
    }


}

 

以上就是使用uri调用第三方地图的所有代码,如果有不理解的或者代码有问题欢迎发送到我的邮箱:  dingchao7323@qq.com

 

以上是关于Android 调起第三方地图应用进行导航的主要内容,如果未能解决你的问题,请参考以下文章

Android开发 PopupWindow弹窗调用第三方地图(百度,高德)实现导航功能

ReactNative 调用手机地图(高德百度)导航 Android

Android仿微信调用第三方地图应用导航(高德百度腾讯)

IOS实现应用内打开第三方地图app进行导航

Android 第三方 SDK 之 高德地图(一)

Android 仿微信调用第三方应用导航(百度,高德腾讯)