Flutter与Android通信的三种方式

1,326 阅读3分钟

「这是我参与2022首次更文挑战的第8天,活动详情查看:2022首次更文挑战

一、 MethodChannel
主要是flutter端调用android方法。flutter调取android方法,也可以android主动跟flutter通信,但是这个只能是传递数据,不是调方法。MethodChannel的flutter调取android方法,我之前写过,可以查看如下链接,www.jianshu.com/p/6b677ff33…

Android主动跟flutter通信,如下

  • Flutter端创建通道
static var platpform = const MethodChannel("com.flutter.guide.MyFlutterView");
  • Flutter 端监听发送过来的数据
 platpform.setMethodCallHandler((call) {
      nativeData = call.arguments["count"];
      setState(() {
      });

      print("count $nativeData");
    });
  • Flutter 端监听发送过来的数据
 platpform.setMethodCallHandler((call) {
      nativeData = call.arguments["count"];
      setState(() {
      });

      print("count $nativeData");
    });
  • Android端写一个类,来发送数据到
package com.example.flutter_demo

import io.flutter.plugin.common.BinaryMessenger
import io.flutter.plugin.common.MethodChannel
import java.util.*
import kotlin.concurrent.timerTask

class  MethodChannelDemo(var activity: MainActivity,messenger: BinaryMessenger){
    private  var channel: MethodChannel
    private var count = 0
    init {
        channel = MethodChannel(messenger, "com.flutter.guide.MyFlutterView")
        Timer().schedule(timerTask {
            activity.runOnUiThread {
                var map = mapOf("count" to count++)
                channel.invokeMethod("timer", map)
            }
        },0,1000)

    }
}
  • Android 端在MainActivity中初始化一下刚才创建的类。
MethodChannelDemo(this,flutterEngine.dartExecutor.binaryMessenger)

二、 BasicMessageChannel
它是可以双端通信的,flutter端可以给Android发送消息,Android也可以给Flutter发送消息。

  • Flutter 端创建一个通道
 static var basicChannel = const BasicMessageChannel("com.flutter.guide.MyFlutterView", StandardMessageCodec());
  • Flutter端发送消息,并接收返回数据resullt
 result =  await basicChannel.send({'name':'azy','age' : 19});
  • Android 端创建相同通道,并接收消息和发送消息
 var basicMessageChannel = BasicMessageChannel(messager, "com.flutter.guide.MyFlutterView", StandardMessageCodec())
        basicMessageChannel.setMessageHandler { message, reply ->
            var map = message as Map<String, Any>
            text.text = "原生的textvieww : ${map["name"]}, ${map["age"]}"
            //被动回复 flutter 消息
          reply.reply("I get it")
            //主动给 flutter 发消息
            basicMessageChannel.send("我还可以主动告诉你,我是个男的")
        }
  • Flutter 端接收Android端主动发送的消息
basicChannel.setMessageHandler((message) async {
     this.message = message;
   });

三、EventChannel
只能是原生发送消息给Flutter端,例如监听手机电量变化,网络变化,传感器等。

  • Android 创建通道并发送消息
 EventChannel(messager,"com.flutter.guide.MyFlutterView").setStreamHandler(object : EventChannel.StreamHandler{
                override fun onListen(arguments: Any?, events: EventChannel.EventSink?) {
                    for (i in 0 .. 10){
                        events?.success(i)
                    }
                    events?.endOfStream()
                }

                override fun onCancel(arguments: Any?) {

                }

            })
  • Flutter 端创建通道
 static var eventChannel = const EventChannel("com.flutter.guide.MyFlutterView");
  • Flutter端监听返回消息
   eventChannel.receiveBroadcastStream().listen((event) {
      print("event "+event.toString());
    },onDone: (){
      print("done");
    },onError: (object){
      print("on error $object");
    });
2020-10-08 16:12:29.578 14515-14594/com.example.flutter_demo I/flutter: event 0
2020-10-08 16:12:29.578 14515-14594/com.example.flutter_demo I/flutter: event 1
2020-10-08 16:12:29.578 14515-14594/com.example.flutter_demo I/flutter: event 2
2020-10-08 16:12:29.578 14515-14594/com.example.flutter_demo I/flutter: event 3
2020-10-08 16:12:29.579 14515-14594/com.example.flutter_demo I/flutter: event 4
2020-10-08 16:12:29.579 14515-14594/com.example.flutter_demo I/flutter: event 5
2020-10-08 16:12:29.579 14515-14594/com.example.flutter_demo I/flutter: event 6
2020-10-08 16:12:29.579 14515-14594/com.example.flutter_demo I/flutter: event 7
2020-10-08 16:12:29.579 14515-14594/com.example.flutter_demo I/flutter: event 8
2020-10-08 16:12:29.579 14515-14594/com.example.flutter_demo I/flutter: event 9
2020-10-08 16:12:29.580 14515-14594/com.example.flutter_demo I/flutter: event 10
2020-10-08 16:12:29.570 14515-14515/com.example.flutter_demo W/1.raster: type=1400 audit(0.0:2234682): avc: denied { read } for name="u:object_r:vendor_default_prop:s0" dev="tmpfs" ino=22601 scontext=u:r:untrusted_app:s0:c129,c256,c512,c768 tcontext=u:object_r:vendor_default_prop:s0 tclass=file permissive=0
2020-10-08 16:12:29.580 14515-14595/com.example.flutter_demo E/libc: Access denied finding property "vendor.debug.egl.swapinterval"
2020-10-08 16:12:29.580 14515-14594/com.example.flutter_demo I/flutter: done

总结一下:
MethodChannel 用于传递方法调用(method invocation),是flutter调取原生方法的,也可以原生主动传递数据给Flutter。
BasicMessageChannel 用于传递字符串和半结构化的信息。是两个端相互发送数据,接收数据的。
EventChannel 用于数据流(event streams)的通信。通长用于Nativie向flutter的通信,如:手机电量变化,网络连接变化,陀螺仪,传感器等;

tip:多种类型的通道混用可能会出现报错问题。