flutter开发实战-ijkplayer视频播放器功能
flutter开发实战-ijkplayer视频播放器功能
使用better_player播放器进行播放视频时候,在Android上会出现解码失败的问题,better_player使用的是video_player,video_player很多视频无法解码。最终采用ijkplayer播放器插件,在flutter上使用fijkplayer插件。
一、引入fijkplayer
在使用fijkplayer前可以先看下https://fijkplayer.befovy.com/docs/zh/fijkplayer-api.html
在工程的pubspec.yaml中引入插件
fijkplayer: ^0.11.0
fijkPlayer 就是对 native C 层 ijkplayer 的一个 dart 包装,接口都保持一致。 FijkPlayer 处理所有播放相关的工作,实际工作都是由 native C 层 ijkplayer 完成,包含检查 dataSource 中的媒体信息,打开解码器和解码线程、打开音频输出设备、将解码后数据输出给音频设备或显示设备。
二、使用fijkplayer
2.1、IJKVideoPlayerController控制常用操作
使用fijkplayer,这里创建了IJKVideoPlayer来嵌套一下FijkView,使用IJKVideoPlayerController来控制常用功能操作
IJKVideoPlayerController如下
import 'dart:async';class IJKVideoPlayerController {FutureOr Function()? stop;FutureOr Function()? pause;FutureOr Function()? play;FutureOr Function(int msec)? seekTo;FutureOr Function(double volume)? setVolume;FutureOr Function(double speed)? setSpeed;FutureOr Function(int loopCount)? setLoop;FutureOr Function()? isPlaying;
}
IJKVideoPlayerController来控制停止、暂停、播放、seek、设置音量、设置播放速率、设置循环次数、获取是否在播放中等。
- 播放视频
void play() {if (videoPlayerController.play != null) {videoPlayerController.play!.call();}}
- 暂停视频播放
void pause() {if (videoPlayerController.pause != null) {videoPlayerController.pause!.call();}}
- 停止视频播放
void stop() {if (videoPlayerController.stop != null) {videoPlayerController.stop!.call();}}
- seek指定位置
void seekTo(int msec) {if (videoPlayerController.seekTo != null) {videoPlayerController.seekTo!.call(msec);}}
- 设置音量
void setVolume(double volume) {if (videoPlayerController.setVolume != null) {videoPlayerController.setVolume!.call(volume);}}
- 设置播放速率
void setSpeed(double speed) {if (videoPlayerController.setSpeed != null) {videoPlayerController.setSpeed!.call(speed);}}
- 设置循环次数
void setLoop(int loopCount) {if (videoPlayerController.setLoop != null) {videoPlayerController.setLoop!.call(loopCount);}}
- 获取是否播放中
Future<bool?> isPlaying() async {if (videoPlayerController.isPlaying != null) {bool videoIsPlaying = await videoPlayerController.isPlaying!.call();return videoIsPlaying;}return Future.value(null);}
2.2、在ijkplayer设置source,使用FijkPlayer
在设置播放器的时候,需要设置source类型。fijkplayer提供了两种方式,一种是本地工程文件、一种是网络视频地址。
- 设置网络视频源
/// usage/// autoPlay 为 true 时等同于连续调用 setDataSource、prepareAsync、startfplayer.setDataSource("http://samplevideo.com/sample.flv", autoPlay: true);
- 设置本地资源作为播放源
/// pubspec.yml 中需要指定assets 内容/// assets:/// - assets/butterfly.mp4////// scheme 是 `asset`, `://` 是 scheme 分隔符, `/` 是路径起始符号fplayer.setDataSource("asset:///assets/butterfly.mp4", autoPlay: true);
在setDataSource还有autoPlay(自动播放),showCover(是否显示视频封面,视频默认获取第一帧作为视频封面)
2.3、FijkView显示视频的控件Widget
在fijkplayer中,使用FijkView来显示视频。
FijkView({required this.player,this.width,this.height,this.fit = FijkFit.contain,this.fsFit = FijkFit.contain,this.panelBuilder = defaultFijkPanelBuilder,this.color = const Color(0xFF607D8B),this.cover,this.fs = true,this.onDispose,});
可以设置显示fit、全屏的fit、背景颜色color、封面图(设置之后会显示在视频播放的上面)、是否全屏等。
在这里我们如果需要自定义样式,可以替换掉panelBuilder。
2.4、自定义控件IJKVideoPanel
在这里我们如果需要自定义样式,可以替换掉panelBuilder。我们自定义一个IJKVideoPanel,这个大部分代码来源default,这里调整了部分样式。
IJKVideoPanel完整代码如下
import 'dart:async';
import 'dart:math';import 'package:fijkplayer/fijkplayer.dart';
import 'package:flutter/material.dart';class IJKVideoPanel extends StatefulWidget {const IJKVideoPanel({super.key,required this.player,required this.buildContext,required this.viewSize,required this.texturePos,});final FijkPlayer player;final BuildContext buildContext;final Size viewSize;final Rect texturePos;@overrideState<IJKVideoPanel> createState() => _IJKVideoPanelState();
}class _IJKVideoPanelState extends State<IJKVideoPanel> {FijkPlayer get player => widget.player;Duration _duration = Duration();Duration _currentPos = Duration();Duration _bufferPos = Duration();bool _playing = false;bool _prepared = false;String? _exception;// bool _buffering = false;double _seekPos = -1.0;StreamSubscription? _currentPosSubs;StreamSubscription? _bufferPosSubs;//StreamSubscription _bufferingSubs;Timer? _hideTimer;bool _hideStuff = true;double _volume = 1.0;final barHeight = 40.0;@overridevoid initState() {super.initState();_duration = player.value.duration;_currentPos = player.currentPos;_bufferPos = player.bufferPos;_prepared = player.state.index >= FijkState.prepared.index;_playing = player.state == FijkState.started;_exception = player.value.exception.message;// _buffering = player.isBuffering;player.addListener(_playerValueChanged);_currentPosSubs = player.onCurrentPosUpdate.listen((v) {setState(() {_currentPos = v;});});_bufferPosSubs = player.onBufferPosUpdate.listen((v) {setState(() {_bufferPos = v;});});}void _playerValueChanged() {FijkValue value = player.value;if (value.duration != _duration) {setState(() {_duration = value.duration;});}bool playing = (value.state == FijkState.started);bool prepared = value.prepared;String? exception = value.exception.message;if (playing != _playing ||prepared != _prepared ||exception != _exception) {setState(() {_playing = playing;_prepared = prepared;_exception = exception;});}}void _playOrPause() {if (_playing == true) {player.pause();} else {player.start();}}@overridevoid dispose() {super.dispose();_hideTimer?.cancel();player.removeListener(_playerValueChanged);_currentPosSubs?.cancel();_bufferPosSubs?.cancel();}void _startHideTimer() {_hideTimer?.cancel();_hideTimer = Timer(const Duration(seconds: 3), () {setState(() {_hideStuff = true;});});}void _cancelAndRestartTimer() {if (_hideStuff == true) {_startHideTimer();}setState(() {_hideStuff = !_hideStuff;});}Widget _buildVolumeButton() {IconData iconData;if (_volume <= 0) {iconData = Icons.volume_off;} else {iconData = Icons.volume_up;}return IconButton(icon: Icon(iconData, color: Colors.white),padding: EdgeInsets.only(left: 10.0, right: 10.0),onPressed: () {setState(() {_volume = _volume > 0 ? 0.0 : 1.0;player.setVolume(_volume);});},);}AnimatedOpacity _buildBottomBar(BuildContext context) {double duration = _duration.inMilliseconds.toDouble();double currentValue =_seekPos > 0 ? _seekPos : _currentPos.inMilliseconds.toDouble();currentValue = min(currentValue, duration);currentValue = max(currentValue, 0);return AnimatedOpacity(opacity: _hideStuff ? 0.0 : 0.8,duration: Duration(milliseconds: 400),child: Container(height: barHeight,decoration: BoxDecoration(gradient: LinearGradient(colors: [Colors.transparent, Colors.black45],begin: Alignment.topCenter,end: Alignment.bottomCenter,),),child: Row(children: <Widget>[_buildVolumeButton(),Padding(padding: EdgeInsets.only(right: 5.0, left: 5),child: Text('${_duration2String(_currentPos)}',style: TextStyle(fontSize: 14.0, color: Colors.white),),),_duration.inMilliseconds == 0? Expanded(child: Center()): Expanded(child: Padding(padding: EdgeInsets.only(right: 0, left: 0),child: FijkSlider(value: currentValue,cacheValue: _bufferPos.inMilliseconds.toDouble(),min: 0.0,max: duration,onChanged: (v) {_startHideTimer();setState(() {_seekPos = v;});},onChangeEnd: (v) {setState(() {player.seekTo(v.toInt());print("seek to $v");_currentPos =Duration(milliseconds: _seekPos.toInt());_seekPos = -1;});},),),),// duration / position_duration.inMilliseconds == 0? Container(child: const Text("LIVE")): Padding(padding: EdgeInsets.only(right: 5.0, left: 5),child: Text('${_duration2String(_duration)}',style: TextStyle(fontSize: 14.0, color: Colors.white),),),// IconButton(
// icon: Icon(widget.player.value.fullScreen
// ? Icons.fullscreen_exit
// : Icons.fullscreen),
// padding: EdgeInsets.only(left: 10.0, right: 10.0),
// // color: Colors.transparent,
// onPressed: () {
// widget.player.value.fullScreen
// ? player.exitFullScreen()
// : player.enterFullScreen();
// },
// )//],),),);}@overrideWidget build(BuildContext context) {// Rect rect = player.value.fullScreen// ? Rect.fromLTWH(0, 0, widget.viewSize.width, widget.viewSize.height)// : Rect.fromLTRB(// max(0.0, widget.texturePos.left),// max(0.0, widget.texturePos.top),// min(widget.viewSize.width, widget.texturePos.right),// min(widget.viewSize.height, widget.texturePos.bottom));Rect rect =Rect.fromLTWH(0, 0, widget.viewSize.width, widget.viewSize.height);return Positioned.fromRect(rect: rect,child: GestureDetector(onTap: _cancelAndRestartTimer,child: AbsorbPointer(absorbing: _hideStuff,child: Column(children: <Widget>[Container(height: barHeight),Expanded(child: GestureDetector(onTap: () {_cancelAndRestartTimer();},child: Container(color: Colors.transparent,height: double.infinity,width: double.infinity,child: Center(child: _exception != null? Text(_exception!,style: TextStyle(color: Colors.white,fontSize: 25,),): (_prepared ||player.state == FijkState.initialized)? AnimatedOpacity(opacity: _hideStuff ? 0.0 : 0.85,duration: Duration(milliseconds: 400),child: IconButton(iconSize: barHeight * 2,icon: Icon(_playing? Icons.pause: Icons.play_arrow,color: Colors.white,size: 44,),padding: EdgeInsets.only(left: 10.0, right: 10.0),onPressed: _playOrPause)): SizedBox(width: barHeight * 1.5,height: barHeight * 1.5,child: CircularProgressIndicator(valueColor: AlwaysStoppedAnimation(Colors.white)),)),),),),_buildBottomBar(context),],),),),);}
}String _duration2String(Duration duration) {if (duration.inMilliseconds < 0) return "-: negtive";String twoDigits(int n) {if (n >= 10) return "$n";return "0$n";}String twoDigitMinutes = twoDigits(duration.inMinutes.remainder(60));String twoDigitSeconds = twoDigits(duration.inSeconds.remainder(60));int inHours = duration.inHours;return inHours > 0? "$inHours:$twoDigitMinutes:$twoDigitSeconds": "$twoDigitMinutes:$twoDigitSeconds";
}
2.5、嵌套FijkView的IJKVideoPlayer
在使用时候,使用了IJKVideoPlayer来封装了一层FijkView。
在IJKVideoPlayer中设置了videoPlayerController控制播放的操作 如停止、暂停、播放、seek、设置音量、设置播放速率、设置循环次数、获取是否在播放中。
IJKVideoPlayer完整代码如下
import 'package:fijkplayer/fijkplayer.dart';
import 'package:flutter/material.dart';
import 'package:flutter_app_demolab/ijk_player/ijk_video_panel.dart';import 'ijk_video_player_controller.dart';/// usage
/// autoPlay 为 true 时等同于连续调用 setDataSource、prepareAsync、start
/// fplayer.setDataSource("http://samplevideo.com/sample.flv", autoPlay: true);
///
/// 设置本地资源作为播放源,
/// pubspec.yml 中需要指定assets 内容
/// assets:
/// - assets/butterfly.mp4
///
/// scheme 是 `asset`, `://` 是 scheme 分隔符, `/` 是路径起始符号
/// fplayer.setDataSource("asset:///assets/butterfly.mp4", autoPlay: true);class IJKVideoPlayer extends StatefulWidget {const IJKVideoPlayer({super.key,required this.path,this.autoPlay = false,this.showCover = true,this.fit = FijkFit.contain,this.cover,this.color = Colors.black,this.width,this.height,this.videoPlayerController,});final double? width;final double? height;final String path;final bool autoPlay;final bool showCover;final FijkFit fit;final Widget? cover;final Color color;final IJKVideoPlayerController? videoPlayerController;@overrideState<IJKVideoPlayer> createState() => _IJKVideoPlayerState();
}class _IJKVideoPlayerState extends State<IJKVideoPlayer> {final FijkPlayer player = FijkPlayer();@overridevoid initState() {super.initState();player.setDataSource(widget.path,autoPlay: widget.autoPlay,showCover: widget.showCover,);addVideoPlayerFun();}void addVideoPlayerFun() {if (widget.videoPlayerController != null) {widget.videoPlayerController!.play = () {// 触发播放player.start();};widget.videoPlayerController!.stop = () {// 触发停止player.stop();};widget.videoPlayerController!.pause = () {// 触发暂停player.pause();};widget.videoPlayerController!.setLoop = (int loopCount) {// 触发setLoopif (loopCount < 0) {loopCount = 1;}player.setLoop(loopCount);};widget.videoPlayerController!.seekTo = (int msec) {// 触发seekif (msec < 0) {msec = 0;}player.seekTo(msec);};widget.videoPlayerController!.setVolume = (double volume) {// 触发setVolumeif (volume < 0.0) {volume = 0.0;}player.setVolume(volume);};widget.videoPlayerController!.setSpeed = (double speed) {// 触发setSpeedif (speed < 0.0) {speed = 1.0;}player.setSpeed(speed);};widget.videoPlayerController!.isPlaying = () {// 触发setVolumeif (FijkState.started == player.state) {return true;} else {return false;}};}}@overridevoid dispose() {super.dispose();player.release();}void onIJKDispose(FijkData fijkData) {}@overrideWidget build(BuildContext context) {return Container(alignment: Alignment.center,child: Stack(alignment: Alignment.center,children: [widget.cover != null ? widget.cover! : Container(),FijkView(width: widget.width,height: widget.height,player: player,fit: widget.fit,fsFit: widget.fit,color: widget.color,onDispose: onIJKDispose,panelBuilder: (FijkPlayer player, FijkData data,BuildContext context, Size viewSize, Rect texturePos) {return IJKVideoPanel(player: player,buildContext: context,viewSize: viewSize,texturePos: texturePos,);},),],),);}
}
三、最后使用IJKVideoPlayer的IJKVideoPage页面
这里我创建了一个IJKVideoPage来使用IJKVideoPlayer视频播放,IJKVideoPlayer中需要path与videoPlayerController
IJKVideoPage完整代码如下
import 'dart:async';import 'package:flutter/material.dart';import 'ijk_player/ijk_video_player.dart';
import 'ijk_player/ijk_video_player_controller.dart';class IJKVideoPage extends StatefulWidget {const IJKVideoPage({super.key,required this.url,});final String url;@overrideState<IJKVideoPage> createState() => _IJKVideoPageState();
}class _IJKVideoPageState extends State<IJKVideoPage> {final IJKVideoPlayerController videoPlayerController =IJKVideoPlayerController();@overridevoid initState() {super.initState();}@overrideWidget build(BuildContext context) {Size size = MediaQuery.of(context).size;return Scaffold(appBar: AppBar(title: Text("Fijkplayer Example")),body: Center(child: Container(width: size.width,height: size.width * 9.0 / 16.0,alignment: Alignment.center,child: IJKVideoPlayer(path: widget.url,videoPlayerController: videoPlayerController,color: Colors.black,),),),);}@overridevoid dispose() {super.dispose();}
}
如果外面的页面跳转到播放页面,需要设置url
void testIJKVideoPage(BuildContext context) {Navigator.of(context).push(MaterialPageRoute(builder: (BuildContext context) {return IJKVideoPage(url: "https://vd2.bdstatic.com/mda-maif0tt1rirqp27q/540p/h264_cae/1611052585/mda-maif0tt1rirqp27q.mp4");}));}
https://brucegwo.blog.csdn.net/article/details/136024588
四、小结
flutter开发实战-ijkplayer视频播放器功能
学习记录,每天不停进步。
相关文章:

flutter开发实战-ijkplayer视频播放器功能
flutter开发实战-ijkplayer视频播放器功能 使用better_player播放器进行播放视频时候,在Android上会出现解码失败的问题,better_player使用的是video_player,video_player很多视频无法解码。最终采用ijkplayer播放器插件,在flutt…...

SpringFramework实战指南(五)
SpringFramework实战指南(五) 4.3 基于 注解 方式管理 Bean4.3.1 实验一: Bean注解标记和扫描 (IoC)4.3.2 实验二: 组件(Bean)作用域和周期方法注解4.3.3 实验三: Bean属性赋值:引用类型自动装配 (DI)4.3.4 实验四: Bean属性赋值:基本类型属性赋值 (DI)4.3.5 实验五:…...

力扣 121. 买卖股票的最佳时机
题目来源:https://leetcode.cn/problems/best-time-to-buy-and-sell-stock/description/ 好久没写代码了,啥啥都忘了 C题解1:贪心算法。(来源代码随想录) 因为股票就买卖一次,那么贪心的想法很自然就是取…...

【STM32+HAL库+CubeMX】UART轮询收发、中断收发、DMA收发方法及空闲中断详解
(转载)原文链接:https://blog.csdn.net/qq_39344192/article/details/131470735 1. 什么是UART? UART是一种异步串行通信接口,常用于通过串口与外部设备进行通信。它通过发送和接收数据帧来实现数据传输,使…...

基于Java医院管理系统设计与实现(源码+部署文档)
博主介绍: ✌至今服务客户已经1000、专注于Java技术领域、项目定制、技术答疑、开发工具、毕业项目实战 ✌ 🍅 文末获取源码联系 🍅 👇🏻 精彩专栏 推荐订阅 👇🏻 不然下次找不到 Java项目精品实…...
PHP://filter过滤器
今天刷题遇到了php://filter过滤器的知识点考察;不会,看了几篇写的不错的文章,本来想转载的,但是代码复制过来后发现格式很乱,和原文格式差太多了;算了,直接把文章连接拿过来吧,在这…...
蓝桥杯刷题day05——2023
1、题目描述 请求出在12345678 (含) 至 98765432 (含) 中 ,有多少个数中完全不包含 2023。 完全不包含 2023是指 无论将这个数的哪些数位移除都不能得到2023。 例如 20322175,33220022 都完全不包含 2023, 而20230415,20193213 …...

【51单片机】开发板和单片机的介绍(2)
前言 大家好吖,欢迎来到 YY 滴单片机系列 ,热烈欢迎! 本章主要内容面向接触过单片机的老铁 主要内容含: 欢迎订阅 YY滴C专栏!更多干货持续更新!以下是传送门! YY的《C》专栏YY的《C11》专栏YY的…...

《剑指 Offer》专项突破版 - 面试题 30 和 31:详解如何设计哈希表以及利用哈希表设计更加高级、复杂的数据结构
目录 一、哈希表的基础知识 二、哈希表的设计 2.1 - 插入、删除和随机访问都是 O(1) 的容器 2.2 - 最近最少使用缓存 一、哈希表的基础知识 哈希表是一种常见的数据结构,在解决算法面试题的时候经常需要用到哈希表。哈希表最大的优点是高效,在哈希表…...
回顾2023年及过去五年的成长经历
现在是2024年2月4日,我想回顾下过去两年的经历和感悟。总结下过去五年的成长经历。 最大的感悟就两点。第一,我相比于两年前成长了很多、也成熟了很多,不管是心智上还是心态上。而这些成长来自于读书、思考和结合实践的反思。第二࿰…...

99例电气实物接线及52个自动化机械手动图
给大家分享一些流水线设计中常见的一些结构,这些动态图很直观,有助于大家了解其原理,非常好懂。 1.家庭总电箱接线图 2.经典双控灯接线 3.五孔一开接线 4.电动机点动控制接线(不安全) 5.电动机自锁接线图(…...
SQL中聚合函数
SQL中的聚合函数是用于对一组值执行计算,并返回单个值的函数。它们通常在SELECT语句的SELECT列表中使用,并与GROUP BY子句结合使用来汇总数据。聚合函数忽略NULL值,只对非NULL值进行计算。以下是一些最常用的SQL聚合函数: 1. COU…...
深度学习预备知识1——数据操作
所有机器学习方法都涉及从数据中提取信息,因此需要一些关于数据的实用技能,包括存储、操作和预处理数据。 机器学习通常需要处理大型数据集。线性代数和矩阵是计算大量数据的有力工具,需要一些矩阵运算相关的线性代数知识。 深度学习是关于…...

【云原生运维问题记录】kubesphere登录不跳转问题
文章目录 现象问题排查 结论先行:kubesphere-system名称空间下reids宕机重启,会判断是否通过registry-proxy重新拉取镜像,该镜像原本是通过阿里云上拉取,代理上没有出现超时情况,导致失败。解决方案:删除re…...
深入学习Prometheus! 一款开源的监控和警报工具!
深入学习Prometheus! 一款开源的监控和警报工具! Prometheus是一个开源的监控和警报工具,它广泛用于记录和收集各种指标(如硬件资源使用情况、应用性能等),并提供强大的查询语言以帮助用户分析和查看这些数据。本文将…...
【webrtc】跟webrtc学list遍历
m98 代码:RTT G:\CDN\rtcCli\m98\src\video\call_stats.cc遍历list 进行删除 :remove_if void RemoveOldReports(int64_t now, std::list<CallStats::RttTime>* reports) {static constexpr const <...

网络安全产品之准入控制系统
文章目录 一、什么是准入控制系统二、准入控制系统的主要功能1. 接入设备的身份认证2. 接入设备的安全性检查 三、准入控制系统的工作原理四、准入控制系统的特点五、准入控制系统的部署方式1. 网关模式2. 控制旁路模式 六、准入控制系统的应用场景七、企业如何利用准入控制系统…...
为什么免费ip代理不适用于分布式爬虫?
费IP代理通常是一些公开免费提供的IP地址和端口,供用户免费使用。然而,这些免费IP代理并不适用于分布式爬虫的使用,原因如下: 1. 不稳定性 免费IP代理通常是由个人或组织提供的,没有稳定的维护和管理机制。因此&…...

【HTML 基础】元数据 meta 标签
文章目录 1. 设置字符集2. 描述网页内容3. 设置关键词4. 网页重定向5. 移动端优化注意事项结语 在网页开发中,<meta> 标签是一种十分重要的 HTML 元数据标签。通过巧妙使用 <meta> 标签,我们能够设置各种元数据,从而影响网页在浏…...

考研中常见的算法-逆置
元素逆置 概述:其实就是将 第一个元素和最后一个元素交换,第二个元素和倒数第二个元素交换,依次到中间位置。用途:可用于数组的移动,字符串反转,链表反转操作,栈和队列反转等操作。 逆置图解 …...
ubuntu搭建nfs服务centos挂载访问
在Ubuntu上设置NFS服务器 在Ubuntu上,你可以使用apt包管理器来安装NFS服务器。打开终端并运行: sudo apt update sudo apt install nfs-kernel-server创建共享目录 创建一个目录用于共享,例如/shared: sudo mkdir /shared sud…...

智慧工地云平台源码,基于微服务架构+Java+Spring Cloud +UniApp +MySql
智慧工地管理云平台系统,智慧工地全套源码,java版智慧工地源码,支持PC端、大屏端、移动端。 智慧工地聚焦建筑行业的市场需求,提供“平台网络终端”的整体解决方案,提供劳务管理、视频管理、智能监测、绿色施工、安全管…...

深入理解JavaScript设计模式之单例模式
目录 什么是单例模式为什么需要单例模式常见应用场景包括 单例模式实现透明单例模式实现不透明单例模式用代理实现单例模式javaScript中的单例模式使用命名空间使用闭包封装私有变量 惰性单例通用的惰性单例 结语 什么是单例模式 单例模式(Singleton Pattern&#…...
HTML前端开发:JavaScript 常用事件详解
作为前端开发的核心,JavaScript 事件是用户与网页交互的基础。以下是常见事件的详细说明和用法示例: 1. onclick - 点击事件 当元素被单击时触发(左键点击) button.onclick function() {alert("按钮被点击了!&…...

全志A40i android7.1 调试信息打印串口由uart0改为uart3
一,概述 1. 目的 将调试信息打印串口由uart0改为uart3。 2. 版本信息 Uboot版本:2014.07; Kernel版本:Linux-3.10; 二,Uboot 1. sys_config.fex改动 使能uart3(TX:PH00 RX:PH01),并让boo…...
Go 语言并发编程基础:无缓冲与有缓冲通道
在上一章节中,我们了解了 Channel 的基本用法。本章将重点分析 Go 中通道的两种类型 —— 无缓冲通道与有缓冲通道,它们在并发编程中各具特点和应用场景。 一、通道的基本分类 类型定义形式特点无缓冲通道make(chan T)发送和接收都必须准备好࿰…...
省略号和可变参数模板
本文主要介绍如何展开可变参数的参数包 1.C语言的va_list展开可变参数 #include <iostream> #include <cstdarg>void printNumbers(int count, ...) {// 声明va_list类型的变量va_list args;// 使用va_start将可变参数写入变量argsva_start(args, count);for (in…...

认识CMake并使用CMake构建自己的第一个项目
1.CMake的作用和优势 跨平台支持:CMake支持多种操作系统和编译器,使用同一份构建配置可以在不同的环境中使用 简化配置:通过CMakeLists.txt文件,用户可以定义项目结构、依赖项、编译选项等,无需手动编写复杂的构建脚本…...
0x-3-Oracle 23 ai-sqlcl 25.1 集成安装-配置和优化
是不是受够了安装了oracle database之后sqlplus的简陋,无法删除无法上下翻页的苦恼。 可以安装readline和rlwrap插件的话,配置.bahs_profile后也能解决上下翻页这些,但是很多生产环境无法安装rpm包。 oracle提供了sqlcl免费许可,…...
WEB3全栈开发——面试专业技能点P7前端与链上集成
一、Next.js技术栈 ✅ 概念介绍 Next.js 是一个基于 React 的 服务端渲染(SSR)与静态网站生成(SSG) 框架,由 Vercel 开发。它简化了构建生产级 React 应用的过程,并内置了很多特性: ✅ 文件系…...