flutter共享(flutter共享界面)

网友投稿 1292 2022-12-25

本篇文章给大家谈谈flutter共享,以及flutter共享界面对应的知识点,希望对各位有所帮助,不要忘了收藏本站喔。 今天给各位分享flutter共享的知识,其中也会对flutter共享界面进行解释,如果能碰巧解决你现在面临的问题,别忘了关注本站,现在开始吧!

本文目录一览:

Flutter 之 数据传递

InheritedWidget 是 Flutter 中的一个功能型 Widget,适用于在 Widget 树中共享数据的场景。
class CounterPage extends StatefulWidget {

CounterPage({Key ? key}) :super(key: key);

@override

  _CounterPageState createState() =_CounterPageState();

}

//

class _CounterPageState extends State {

int count =0;

//3我们通过 InheritedCountContainer.of 方法找到它,获取计数状态 count 并展示:

  void _incrementCounter() = setState(() {count++;});// 修改计数器

  @override

  Widget build(BuildContext context) {

//2我们使用 CountContainer 作为根节点,并用 0 初始化 count。

    return CountContainer(

model:this,

increment: _incrementCounter,// 提供修改数据的方法

        child:Counter()

);

}

}

/*1

* 首先,为了使用 InheritedWidget,我们定义了一个继承自它的新类 CountContainer。

* 然后,我们将计数器状态 count 属性放到 CountContainer 中,

* 并提供了一个 of 方法方便其子 Widget 在 Widget 树中找到它。

* 最后,我们重写了 updateShouldNotify 方法,这个方法会在 Flutter

*  判断 InheritedWidget 是否需要重建,从而通知下层观察者组件更新数据时被调用到。

* 在这里,我们直接判断 count 是否相等即可。

* */

class CountContainer extends InheritedWidget {

static CountContainer?of(BuildContext context) {

return context.dependOnInheritedWidgetOfExactType();

}

final _CounterPageState model;// 直接使用 MyHomePage 中的 State 获取数据

  final Function()increment;

CountContainer({

Key ?key,

required this.model,

required this.increment,

required Widget child,

}):super(key: key, child: child);

@override

  bool updateShouldNotify(CountContainer oldWidget) =model != oldWidget.model;

}

class Counter extends StatelessWidget {

@override

  Widget build(BuildContext context) {

// 获取 InheritedWidget 节点

    CountContainer?state =CountContainer.of(context);

return Scaffold(

appBar:AppBar(

title:Text("InheritedWidget demo"),

),

body:Text(

'You have pushed the button this many times: ${state?.model.count}',// 关联数据读方法

      ),

floatingActionButton:FloatingActionButton(onPressed:state?.increment),// 关联数据修改方法

    );

}

}

Notification 是 Flutter 中进行跨层数据共享的另一个重要的机制。如果说 InheritedWidget 的数据流动方式是从父 Widget 到子 Widget 逐层传递,那 Notificaiton 则恰恰相反,数据流动方式是从子 Widget 向上传递至父 Widget。这样的数据传递机制适用于子 Widget 状态变更,发送通知上报的场景。

class CustomNotification extends Notification {

CustomNotification(this.msg);

final String msg;

}

class CustomChild extends StatelessWidget {

@override

  Widget build(BuildContext context) {

return RaisedButton(

//按钮点击时分发通知

      onPressed: () =CustomNotification("Hi").dispatch(context),

child:Text("Fire Notification"),

);

}

}

class NotificationWidget extends StatefulWidget {

@override

  StatecreateState()=_NotificationState();

}

class _NotificationState extends State {

String _msg ="通知:";

@override

  Widget build(BuildContext context) {

//监听通知

    return NotificationListener(

onNotification: ( notification) {

setState(() {_msg += notification.msg+"  ";});

return true;

},

child:Column(

mainAxisAlignment:MainAxisAlignment.center,

children: [Text(_msg),CustomChild()],

)

);

}

}

无论是 InheritedWidget 还是 Notificaiton,它们的使用场景都需要依靠 Widget 树,也就意味着只能在有父子关系的 Widget 之间进行数据共享。但是,组件间数据传递还有一种常见场景:这些组件间不存在父子关系。这时,事件总线 EventBus 就登场了。

事件总线是在 Flutter 中实现跨组件通信的机制。它遵循发布 / 订阅模式,允许订阅者订阅事件,当发布者触发事件时,订阅者和发布者之间可以通过事件进行交互。发布者和订阅者之间无需有父子关系,甚至非 Widget 对象也可以发布 / 订阅。

// 所以在这里,我们传输数据的载体就选择了一个有字符串属性的自定义事件类 CustomEvent:

class CustomEvent {

String msg;

CustomEvent(this.msg);

}

// 建立公共的 event bus

EventBus eventBus =new EventBus();

/*

* 我们定义了一个全局的 eventBus 对象,并在第一个页面监听了 CustomEvent 事件,

* 一旦收到事件,就会刷新 UI。

* 需要注意的是,千万别忘了在 State 被销毁时清理掉事件注册,

* 否则你会发现 State 永远被 EventBus 持有着,无法释放,从而造成内存泄漏:

*

*  */

class FirstPage extends StatefulWidget {

@override

  StatecreateState()=_FirstPageState();

}

class _FirstPageState extends State {

String msg ="通知:";

late StreamSubscription subscription;

@override

  void initState() {

//监听CustomEvent事件,刷新UI

    subscription =eventBus.on().listen((event) {

print(event.msg);

setState(() {

msg += event.msg;

});

});

super.initState();

}

dispose() {

subscription.cancel();//State销毁时,清理注册

    super.dispose();

}

@override

  Widget build(BuildContext context) {

return Scaffold(

appBar:AppBar(title:Text("First Page"),),

body:Text(msg),

floatingActionButton:FloatingActionButton(onPressed: ()=Navigator.push(context,MaterialPageRoute(builder: (context) =SecondPage()))),

);

}

}

//我们在第二个页面以按钮点击回调的方式,触发了 CustomEvent 事件:

class SecondPage extends StatelessWidget {

Widget build(BuildContext context) {

return Scaffold(

appBar:AppBar(title:Text("Second Page"),),

body:RaisedButton(

child:Text('Fire Event'),

// 触发CustomEvent事件

          onPressed: ()=eventBus.fire(CustomEvent("hello"))

),

);

}

}

Flutter Android外接纹理实现图片共享

Android 利用surface实现Flutter外接纹理
https://gitee.com/woshishui1243/external_plugin

①Java层FlutterRenderer创建SurfaceTexture和textureId。
②将surfaceTexture和textureId通过JNI向引擎层注册
③向引擎注册过程中通过层层方法最后在texture.ccflutter共享的TextureRegistry由map以键值对形式缓存实例对象。
④将需要显示图片在SurfaceTexture上离屏渲染。
⑤Java层创建flutter共享的textureId通过Channel传递到Dart层作为Texture组件入参。
⑥Dartflutter共享的Texture组件接收textureId入参后向下层组件实例化。
⑦在SceneBuilder调用addTexture时执行引擎层创建TextureLayer。
⑧最终在texture.cc中TextureRegistryflutter共享的map根据TextureId获取SurfaceTexture实例。

使用Native(以Android为例)播放器构建Flutter播放插件
https://juejin-/post/6844903662649606157

iOS 实现Flutter外接纹理

https://github.com/woshishui1243/flutter_texture

Flutter浪潮下的音视频研发探索

文/陈炉军

整理/LiveVideoStack
大家好flutter共享,我是阿里巴巴闲鱼事业部flutter共享的陈炉军,本次分享flutter共享的主题是Flutter浪潮下的音视频研发探索,主要内容是针对闲鱼APP在当下流行的跨平台框架Flutter的大规模实践,介绍其在音视频领域碰到的一些困难以及解决方案。
分享内容主要分为四个方面,首先会对Flutter有一个简单介绍以及选择Flutter作为跨平台框架的原因,其次会介绍Flutter中与音视频关系非常大的外接纹理概念,以及对它做出的一些优化。之后会对闲鱼在音视频实践过程中碰到的一些Flutter问题提出了一些解决方案——TPM音视频框架。最后是闲鱼Flutter多媒体开源组件的介绍。

Flutter
Flutter是一个跨平台框架,以往的做法是将音频、视频和网络这些模块都下沉到C++层或者ARM层,在其上封装成一个音视频的SDK,供UI层的PC、iOS和Android调用。

而Flutter做为一个UI层的跨平台框架,顾名思义就是在UI层也实现了一个跨平台开发。可以预想的是未Flutter发展的好的话,会逐渐变为一个从底层到UI层的一个全链路的跨平台开发,技术人员分别负责SDK和UI层的开发。
在Flutter之前已经有很多跨平台UI解决方案,那为什么选择Flutter呢flutter共享

我们主要考虑性能和跨平台的能力。

以往的跨平台方案比如Weex,ReactNative,Cordova等等因为架构的原因无法满足性能要求,尤其是在音视频这种性能要求几乎苛刻的场景。

而诸如Xamarin等,虽然性能可以和原生App一致,但是大部分逻辑还是需要分平台实现。
我们可以看一下,为什么Flutter可以实现高性能flutter共享

原生的native组件渲染以IOS为例,苹果的UIKit通过调用平台自己的绘制框架QuaztCore来实现UI的绘制,图形绘制也是调用底层的API,比如OpenGL、Metal等。

而Flutter也是和原生API逻辑一致,也是通过调用底层的绘制框架层SKIA实现UI层。这样相当于Flutter他自己实现了一套UI框架,提供了一种性能超越原生API的跨平台可能性。
但是我们说一个框架最终性能怎样,其实取决于设计者和开发者。至于现在到底是一个什么状况:

在闲鱼的实践中,我们发现在正常的开发没有特意的去优化UI代码的情况下,在一些低端机上,Flutter界面的流畅性是比Native界面要好的。

虽然现在闲鱼某些场景下会有卡顿闪退等情况,但是这是一个新事物发展过程中的必然问题,我们相信未来性能肯定不会成为限制Flutter发展的瓶颈的。
在闲鱼实践Flutter的过程中,混合栈和音视频是其中比较难解决的两个问题,混合栈是指一个APP在Flutter过程中不可能一口气将所有业务全部重写为Flutter,所以这是一个逐步迭代的过程,这期间原生native界面与Flutter界面共存的状态就称之为混合栈。闲鱼在混合栈上也有一些比较好的输出,例如FlutterBoost。

外接纹理
在讲音视频之前需要简要介绍一下外接纹理的概念,我们将它称之为是Flutter和Frame之间的桥梁。
Flutter渲染一帧屏幕数据首先要做的是,GPU发出的VC信号在Flutter的UI线程,通过AOT编译的机器码结合当前Dart Runtime,生成Layer Tree UI树,Layer Tree上每一个叶子节点都代表了当前屏幕上所需要渲染的每一个元素,包含了这些元素渲染所需要的内容。将Layer Tree抛给GPU线程,在GPU线程内调用Skia去完成整个UI的渲染过程。Layer Tree中有PictureLayer和TextureLayer两个比较重要的节点。PictureLayer主要负责屏幕图片的渲染,Flutter内部实现了一套图片解码逻辑,在IO线程将图片读取或者从网络上拉取之后,通过解码能够在IO线程上加载出纹理,交给GPU线程将图片渲染到屏幕上。但是由于音视频场景下系统API太过繁多,业务场景过于复杂。Flutter没有一套逻辑去实现跨平台的音视频组件,所以说Flutter提出了一种让第三方开发者来实现音视频组件的方式,而这些音视频组件的视频渲染出口,就是TextureLayer。

在整个Layer Tree渲染的过程中,TextureLayer的数据纹理需要由外部第三方开发者来指定,可以把视频数据和播放器数据送到TextureLayer里,由Flutter将这些数据渲染出来。
TextureLayer渲染过程:首先判断Layer是否已经初始化,如果没有就创建一个Texture,然后将Texture Attach到一个SufaceTexture上。

这个SufaceTexture是音视频的native代码可以获取到的对象,通过这个对象创建的Suface,我们可以将视频数据、摄像头数据解码放到Suface中,然后Flutter端通过监听SufaceTexture的数据更新就可以顺利把刚才创建的数据更新到它的纹理中,然后再将纹理交给SKIA渲染到屏幕上。
然而我们如果需要用Flutter实现美颜,滤镜,人脸贴图等等功能,就需要将视频数据读取出来,更新到纹理中,再将GPU纹理经过美颜滤镜处理后生成一个处理后的纹理。按Flutter提供的现有能力,必须先将纹理中的数据从GPU读出到CPU中,生成Bitmap后再写入Surface中,这样在Flutter中才能顺利的更新到视频数据,这样做对系统性能的消耗很大。
通过对Flutter渲染过程分析,我们知道Flutter底层需要渲染的数据就是GPU纹理,而我们经过美颜滤镜处理完成以后的结果也是GPU纹理,如果可以将它直接交给Flutter渲染,那就可以避免GPU-CPU-GPU这样的无用循环。这样的方法是可行的,但是需要一个条件,就是OpenGL上下文共享。

OpenGL
在说上下文之前,得提到一个和上线文息息相关的概念:线程。

Flutter引擎启动后会启动四个线程:

第一个线程是UI线程,这是Flutter自己定义的UI线程,主要负责GPU发出的VSync信号时候用当前Dart编译的机器码和当前运行环境创建出Layer Tree。

还有就是IO线程和GPU线程。和大部分OpenGL处理解决方案中一样,Flutter也采取一个线程责资源加载,一部分负责资源渲染这种思路。

两个线程之间纹理共享有两种方式。一种是EGLImage(IOS是 CVOpenGLESTextureCache)。一种是OpenGL Share Context。Flutter通过Share Context来实现纹理共享,将IO线程的Context和GPU线程的Context进行Share,放到同一个Share Group下面,这样两个线程下资源是互相可见可以共享的。

Platform线程是主线程,Flutter中有一个很奇怪的设定,GPU线程和主线程共用一个Context。并且在主线程也有很多OpenGL 操作。

这样的设计会给音视频开发带来很多问题,后面会详细说。
音视频端美颜处理完成的OpenGL纹理能够让Flutter直接使用的条件就是Flutter的上下文需要和平台音视频相关的OpenGL上下文处在一个Share Group下面。

由于Flutter主线程的Context就是GPU的Context,所以在音视频端主线程中有一些OpenGL操作的话,很有可能使Flutter整个OpenGL被破坏掉。所以需要将所有的OpenGL操作都限制在子线程中。

通过上述这两个条件的处理,我们就可以在没有增加GPU消耗的前提下实现美颜和滤镜等等功能。

TPM
在经过demo验证之后,我们将这个方案应用到闲鱼音视频组件中,但改造过程中发现了一些问题。

上图是摄像头采集数据转换为纹理的一段代码,其中有两个操作:首先是切进程,将后面的OpenGL操作都切到cameraQueue中。然后是设置一次上下文。然后这种限制条件或者说是潜规则往往在开发过程中容易被忽略的。而这个条件一旦忽略后果就是出现一些莫名其妙的诡异问题极难排查。因此我们就希望能抽象出一套框架,由框架本身实现线程的切换、上下文和模块生命周期等的管理,开发者接入框架以后只需要安心实现自己的算法,而不需要关心这些潜规则还有其他一些重复的逻辑操作。
在引入Flutter之前闲鱼的音视频架构与大部分音视频逻辑一样采用分层架构:

1:底层是一些独立模块

2:SDK层是对底层模块的封装

3:最上层是UI层。

引入Flutter之后,通过分析各个模块的使用场景,我们可以得出一个假设或者说是抽象:音视频应用在终端上可以归纳为视频帧解码之后视频数据帧在各个模块之间流动的过程,基于这种假设去做Flutter音视频框架的抽象。

咸鱼Flutter多媒体开源组件
整个Flutter音视频框架抽象分为管线和数据的抽象、模块的抽象、线程统一管理和上下文同一管理四部分。
管线,其实就是视频帧流动的管道。数据,音视频中涉及到的数据包括纹理、Bit Map以及时间戳等。结合现有的应用场景我们定义了管线流通数据以Texture为主数据,同时可以选择性的添加Bit Map等作为辅助数据。这样的数据定义方式,避免重复的创建和销毁纹理带来的性能开销以及多线程访问纹理带来的一些问题。也满足一些特殊模块对特殊数据的需求。同时也设计了纹理池来管理管线中的纹理数据。
模块:如果把管线和数据比喻成血管和血液,那框架音视频的场景就可以比喻成器官,我们根据模块所在管线的位置抽象出采集、处理和输出三个基类。这三个基类里实现了刚才说的线程切换,上下文切换,格式转换等等共同逻辑,各个功能模块通过集成自这些基类,可以避免很多重复劳动。
线程:每一个模块初始化的时候,初始化函数就会去线程管理的模块去获取自己的线程,线程管理模块可以决定给初始化函数分配新的线程或者已经分配过其他模块的线程。

这样有三个好处:

一是可以根据需要去决定一个线程可以挂载多少模块,做到线程间的负载均衡。第二,多线程并发式能够保证模块内的OpenGL操作是在当前线程内而不会跑到主线程去,彻底避免Flutter的OpenGL 环境被破坏。第三,多线程并行可以充分利用CPU多核架构,提升处理速度。
从Flutter端修改Flutter引擎将Context取出后,根据Context创建上下文的统一管理模块,每一个模块在初始化的时候会获取它的线程,获取之后会调用上下文管理模块获取自己的上下文。这样可以保证每一个模块的上下文都是与Flutter的上下文进行Share的,每个模块之间资源都是共享可见的,Flutter和音视频native之间也是互相共享可见的。
基于上述框架如果要实现一个简单的场景,比如画面实时预览和滤镜处理功能,

1:需要选择功能模块,功能模块包括摄像头模块、滤镜处理模块和Flutter画面渲染模块,

2:需要配置模块参数,比如采集分辨率、滤镜参数和前后摄像头设置等,

3:在创建视频管线后使用已配置的参数创建模块

4:最后管线搭载模块,开启管线就可以实现这样简单的功能。
上图为整个功能实现的代码和结构图。
结合上述音视频框架,闲鱼实现了Flutter多媒体开源组件。

组要包含四个基本组件分别是:

1:视频图像拍摄组件

2:播放器组件

3:视频图像编辑组件

4:相册选择组件

现在这些组件正在走内部开源流程。预计9月份,相册和播放器会实现开源。

后续展望和规划
1:实现开头所说的从底层SDK到UI的全链路的跨端开发。目前底层框架层和模块层都是各个平台各自实现,反而是Flutter的UI端进行了跨平台的统一,所以后续会将底层也按照音视频常用做法把逻辑下沉到C++层,尽可能的实现全链路跨平台。

2:第二部分内容为开源共建,闲鱼开源的内容不仅包括拍摄、编辑组件,还包括了很多底层模块,希望有开发者在基于Flutter开发音视频应用时可以充分利用闲鱼开源出的音视频模块能力,搭建APP框架,开发者只要去负责实现特殊需求模块就可以,尽可能的减少重复劳动。

Flutter中InheritedWidget的使用

在Tree中从上往下高效传递数据的基类widget , 定义为:abstract class InheritedWidget extends ProxyWidget

Flutter的响应式开发与React类似flutter共享,数据都是自顶向下的。

假设有祖先组点A,中间经过结点B, C,然后到结点D,D需要从A中获取数据f,那按照自顶向下数据流转,f需要依次传递给B及C,最后才到C。这样开发极为不灵活,成本也比较高。所有Flutter需要有跨结点(只能是祖先后代节点,不能跨兄弟节点)高效传递数据的方案。

大体意思如下:

InheritedWidget 是在树中高效向下传递信息的基类部件flutter共享

调用[BuildContext.inheritFromWidgetOfExactType]方法可以从 BuildContext 中获取到最近的 InheritedWidget 类型的实例;

在 InheritedWidget 类型的控件被引用,也就是调用过 inheritFromWidgetOfExactType 方法后,当 InheritedWidget 自身状态改变时,会导致引用了 InheritedWidget 类型的子控件重构(rebuild)。

这里随便定义一个人 Person 类。

创建一个类继承 InheritedWidget,并实现 updateShouldNotify 方法。

之前说到调用[BuildContext.inheritFromWidgetOfExactType]方法可以从 BuildContext 中获取到最近的 InheritedWidget 类型的实例,所以此处定义一个静态的 of 方法,通过传入的 context 获取到最近的 InheriedDataWidget 实例。

1.定义数据模型

这里随便定义一个 Person 类。

2.自定义 InheritedWidget 控件类

创建一个类继承 InheritedWidget,并实现 updateShouldNotify 方法。

之前说到调用[BuildContext.inheritFromWidgetOfExactType]方法可以从 BuildContext 中获取到最近的 InheritedWidget 类型的实例,所以此处定义一个静态的 of 方法,通过传入的 context 获取到最近的 InheriedDataWidget 实例。

3.InheriedDataWidget 的使用

InheriedDataWidget 使用起来也很简单,它本身也是一个控件,只要在任意一个页面的子控件调用其构造方法就行,这里我们定义一个形如的 Widget 树。

WidgetA 是一个 StatefulWidget 类型的控件,可以调用 setState 刷新,如果是继承人 Stateless 类型的控件,那我们也可以通过 Stream 或者其他方式刷新数据,感兴趣的请看[什么是 Stream? Dart

WidgetA1_1 类

WidgetA1_2 类

WidgetA1_3 类

当我们点击 floatingActionButton 的时候,WidgetA1, WidgetA1_1, WidgetA1_2 的控件都会更新 Person 的信息,而且每点 floatingActionButton 一次, 当我们点击 floatingActionButton 的时候,WidgetA1, WidgetA1_1, WidgetA1_2 的控件都会更新 Person 的信息,而且每点 floatingActionButton 一次,都会输出:

如果我们试图在和 WidgetA 的同一层级的兄弟节点去访问 InheriedDataWidget 的 Person 数据,是不行的,因为父节点中并没有插入 InheriedDataWidget。

把 WidgetB 和 WidgetA 保持同一节点

这也体现了 Inheried(遗传) 这一单词的特性,遗传只存在于父子。兄弟不存在遗传的关系。

这种数据共享的方式在某些场景还是很有用的,就比如说全局主题,字体大小,字体颜色的变更,只要在 App 根层级共享出这些配置数据,然后在触发数据改变之后,所有引用到这些共享数据的地方都会刷新,这换主题,字体是不是就很轻松,事实上 Theme.of(context).primaryColor 之流就是这么干的。

以上就是有关InheritedWidget的使用。

自己也是从事Android开发5年有余了;整理了一些Android开发技术核心笔记和面经题纲,有关更多Android开发进阶技术资料、面经题纲、核心技术笔记; 想要进阶自己、拿高薪的同学请私信我回复“核心笔记”或“面试”领取flutter共享

flutter 如何实现app与打印机连接打印文件?

点击开始----控制版面-----打印机和传真机----点击添加打印机----在弹出的添加打印机向导对话框中点下一步----选第一项本地打印机再点下一步----下一步直到出现厂商一栏时,点击从磁盘安装.再将你的打印机驱动放入光驱中..向下就行了
右键单击要共享的打印机,然后单击“共享”。 根据计算机上是否启用了共享,在“共享”选项卡上的选项会各不相同。要了解下一步怎么做,请在“共享”选项卡上单击有相应说明的选项。 一条消息提示打印机共享必须打开 需要运行“网络安装向导”,以启用打印机共享。首先单击“共享”选项卡上的链接,然后执行系统指示以启动共享。一旦启用了共享,请再次执行该过程。 显示共享或不共享打印机的选项 在“共享”选项卡上,单击“共享这台打印机”,然后在“共享名”框中键入共享打印机的名称。 如果与硬件或操作系统不同的用户共享打印机,请单击“其他驱动程序”。单击其他计算机的环境和操作系统,然后单击“确定”,以安装其他驱动程序。 运行 Windows 其他版本(Windows 95、Windows 98 或者 Windows NT 4.0)的用户可在支持 CD 中找到驱动程序。但其中没有 Windows NT 3.1 和 Windows NT 3.5 的打印机驱动程序。 单击“确定”。如果已经安装了其他驱动程序,则单击“关闭”。 如果已遵循上述步骤,但仍无法共享打印机,那么请打开 Windows 防火墙。(要打开“Windows 防火墙”,请依次单击“开始”、“控制面板”,然后双击“Windows 防火墙”。) 然后,在“例外”选项卡上,选择“文件和打印机共享”复选框。 注意 要打开“打印机和传真”,请依次单击“开始”、“控制面板”,然后双击“打印机和传真”。 还可以在“打印机和传真”中共享打印机,方法是:单击要共享的打印机,然后单击左侧窗格“打印机任务”下面的“共享此打印机”。只有当文件夹设置为网页风格,并且当前选中打印机时,该选项才可用。详细信息,请单击“相关主题”。 默认情况下,当打印机安装在 Windows XP Professional 上时打印机没有共享,但可以选择将安装在计算机上的任何打印机共享。 当在 Active Directory 中发布打印机后,登录到 Windows 域的其他用户可以根据打印机的位置和特性(诸如每分钟打印多少页、是否支持彩色打印等)来搜索打印机。 关于flutter共享和flutter共享界面的介绍到此就结束了,不知道你从中找到你需要的信息了吗 ?如果你还想了解更多这方面的信息,记得收藏关注本站。 flutter共享的介绍就聊到这里吧,感谢你花时间阅读本站内容,更多关于flutter共享界面、flutter共享的信息别忘了在本站进行查找喔。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:SpringBoot actuator 健康检查不通过的解决方案
下一篇:微信小程序下载的app在哪里(微信下载小程序在哪找?)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~