- Android中动态广播注册和收发原理
xzkyd outpaper
android系统开发知识android计算机八股
1.注册广播:封装Binder对象到AMS动态广播的注册核心在于将接收器与AMS(ActivityManagerService)建立跨进程通信通道,具体流程如下:Binder对象生成当应用调用registerReceiver()动态注册广播时,系统会通过LoadedApk创建ReceiverDispatcher对象,其内部封装了InnerReceiver(继承自IIntentReceiver.St
- 【Spring Cloud Gateway 实战系列】进阶篇:过滤器高级用法、动态路由配置与性能优化
一、过滤器高级用法:从基础到复杂场景1.1过滤器执行顺序深度解析SpringCloudGateway的过滤器执行顺序由Order接口控制,数值越小优先级越高。全局过滤器(GlobalFilter)需通过GatewayFilterAdapter适配为局部过滤器,默认过滤器(default-filters)优先级高于局部过滤器。1.1.1顺序控制示例@Component@Order(1)//优先级高于
- spring security 3 auto-config=“true”
springsecurity3设置auto-config=true时,会配置十个默认过滤器SecurityContextPersistenceFilterLogoutFilterUsernamePasswordAuthenticationFilterBasicAuthenticationFilterRequestCacheAwareFilterSecurityContextHolderAwareR
- Android Jetpack Compose + MVI 开发流程深度分析
你过来啊你
androidcomposemvi
MVI架构核心原理MVI(Model-View-Intent)是一种基于单向数据流的架构模式,其核心组件关系如下:[View]--Intents-->[ViewModel]--States-->[View]||用户交互事件处理业务逻辑||[View]=emptyList(),valisLoading:Boolean=false,valerror:String?=null,valnewTodoTit
- 2022-08-18(listening)
Catholi_costudy
Vocabularypeculiarlyad.特别,尤其,古怪地perspectiven.视角,观点,透视画法isolatev.孤立,分离,隔离burdenn.负担,责任,负重overemphasizev.过分强调astonishv.使吃惊,使惊讶intentionn.意图,目的,打算,愈合unselfishnessn.不自私,慷慨practicala.真实的,实际的starvationn.饥饿,
- vue实现超出字数中间用省略号显示
显示效果:传统节日里的氛围......传统节日里的氛围原理:利用vue中的过滤器filterhtml代码:{{hashName|ellipsis}}js代码:filters:{ellipsis(value){letlen=value.length;if(!value)return''if(value.length>20){returnvalue.substring(0,8)+'......'+va
- SIGMOD论文解读|在自下而上优化中添加布隆过滤器
Gauss松鼠会
技术交流数据库gaussdbdatabase
6月22日至27日,2025ACMSIGMOD/PODS国际学术会议在德国柏林举行。25日,华为多伦多分布式调度和数据引擎实验室主任工程师TimothyZeyl受邀出席,就入选的《IncludingBloomFiltersinBottom-upOptimization》论文进行了解读该论文创新性地首次提出了在自下而上的优化器的基于成本的优化过程中添加布隆过滤器(BloomFilter)的技术。该技
- 怎么安装自定义分词器
思静鱼
#elasticsearches
安装自定义分词器的完整步骤在Elasticsearch中安装自定义分词器,通常需要修改索引配置或开发插件。以下是详细方法:一、基于配置实现自定义分词器(无需插件)适用于通过组合Elasticsearch内置的CharacterFilters、Tokenizers和TokenFilters实现的分词器。1.定义分词规则在创建索引时,通过settings.analysis配置自定义分词器:PUT/my
- 深入理解安卓系统架构与开发技术
本文还有配套的精品资源,点击获取简介:安卓基础知识课程为初学者及进阶开发者提供安卓系统架构、关键开发技术以及最新开发工具的全面介绍。课程涵盖了从系统架构层面的核心概念如HAL、系统库、应用框架、应用程序层,到应用开发的关键组件,包括Dalvik/ART虚拟机、UI设计、Intent通信机制、资源管理、权限系统、进程与线程处理、AndroidStudio/Gradle集成、应用调试与测试,以及And
- 2025系统架构师---管道/过滤器架构风格
喜欢猪猪
java开发语言
引言在分布式系统与数据密集型应用主导技术演进的今天,管道/过滤器架构风格(PipesandFiltersArchitectureStyle)凭借其数据流驱动、组件解耦与并行处理能力,成为处理复杂数据转换任务的核心范式。从Unix命令行工具到实时金融交易引擎,从图像处理流水线到物联网边缘计算,管道/过滤器架构通过将系统拆分为独立处理单元(过滤器)与数据传递通道(管道),实现了功能模块
- 将RHEL/Rocky LINUX 9 KVM启用虚拟化的命令拆解为原子命令
openlabx.org.cn
linux运维KVM
RHEL/RockyLINUXKVM启用虚拟化的命令比较复杂,可能不是每个工程师都能够看懂。要将给定的命令拆解为原子的Linux命令,我们需要展开循环和花括号扩展。原命令如下:bashfordrvinqemunetworknodedevnwfiltersecretstorageinterface;dosystemctlstartvirt${drv}d{,-ro,-admin}.socketdone
- 17.RestClient实现数据聚合功能带过滤条件
卷土重来…
windows
@OverridepublicMap>filters(RequestParamsparam){Map>res=newHashMapbrandList=getAggByName(aggregations,"brandAgg");ListcityList=getAggByName(aggregations,"cityAgg");ListstarList=getAggByName(aggregation
- MySQL 锁机制全解析:从表锁到间隙锁,谁都能看懂的知识!
dr李四维
笔记mysql数据库锁死锁乐观锁悲观锁意向锁
目录一、锁的基本概念二、MySQL中的锁类型1.表级锁(TableLock)2.行级锁(RowLock)3.间隙锁(GapLock)4.意向锁(IntentLock)5.乐观锁(OptimisticLock)6.悲观锁(PessimisticLock)7.死锁(Deadlock)8.自增锁(Auto-IncrementLock)9.元数据锁(MetadataLock,MDL)三、锁的对比与选择四、
- Android跳转到通话记录界面,android 用Intent打开系统程序
weixin_39883670
1.打开设置主界面Intentintent=newIntent(android.provider.Settings.ACTION_SETTINGS);//系统设置startActivityForResult(intent,0);2.打开网络设置界面(其他设置中的界面同理)Intentintent=newIntent(android.provider.Settings.ACTION_WIFI_SET
- Android跳转到手机系统各个设置界面的方法及其列表
枸杞泡水
Android-基础知识android手机系统设置
android调用拨号盘拨打电话publicvoiddialPhoneNumber(StringphoneNumber){Intentintent=newIntent(Intent.ACTION_DIAL);intent.setData(Uri.parse("tel:"+phoneNumber));if(intent.resolveActivity(getPackageManager())!=nu
- GDPU移动应用开发(安卓)期末考试复习资料
第一部分一、选择题应用的基本构建块是:A.ActivityB.FragmentC.ServiceD.Intent答案:A.Activity下列哪个不是AndroidManifest.xml文件的主要用途?A.声明应用权限B.声明应用组件C.定义应用的图标和名称D.编写应用的业务逻辑答案:D.编写应用的业务逻辑在Android中、用于存储持久化数据的SQLite数据库、默认存储在哪个目录下?A./s
- adb logcat 命令详解 log过滤
liao277218962
adblogcat命令行adblogcat
作者:万境绝尘转载请著名出处eclipse自带的LogCat工具太垃圾了,开始用adblogcat在终端查看日志;1.解析adblogcat的帮助信息在命令行中输入adblogcat--help命令,就可以显示该命令的帮助信息;[plain]viewplaincopyoctopus@octopus:~$adblogcat--helpUsage:logcat[options][filterspecs
- Vue.js 过滤器详解
步行cgn
Vuevue.js前端javascript
Vue.js过滤器详解下面我将详细讲解Vue.js中过滤器的语法和使用注意事项,并提供一个完整的演示页面。过滤器基本概念在Vue.js中,过滤器(Filters)是用于文本格式化的功能,可以在双花括号插值和v-bind表达式中使用。过滤器通过管道符(|)指示,主要用于简单的文本转换。{{message|capitalize}}{{message|filterA|filterB}}{{date|fo
- 【Android知识笔记】ARouter / Navigation / EventBus
川峰
Android知识笔记ARouterNavigationEventBus
注:本文主要基于过去AndroidView体系的路由学习笔记整理,不包括最新的JetpackCompose路由体系,如您需了解关于JetpackCompose中的导航路由,请参考JetpackCompose中的导航路由一文。传统路由方式//显性意图startActivity(newIntent(this,HomeActivity.
- 使用 Nginx 默认配置为 iServer 开启 HTTPS(不安装扩展模块 sub_filters)
陈陈CHENCHEN
SuperMapnginxhttps
作者:陈陈敏CHEN目录1、环境介绍2、确认Nginx配置3、具体配置4、完整配置5、验证配置导语:在之前的文章中,我们推荐使用扩展模块sub_filters来为iServer开启HTTPS。然而,sub_filters模块需要额外安装才能使用,而多数客户因为项目原因不愿重新安装Nginx。在这种情况下,我们该如何为iServer启用HTTPS呢?下面让我来为大家介绍如何配置吧!1、环境介绍iSe
- 【GateWay】和权限验证
秋の花
gatewayjava
【GateWay】网关详解和权限验证一、Gateway核心概念与架构二、路由断言(RoutePredicates)详解三、过滤器(Filters)机制四、权限认证的核心理论模型五、SpringCloudGateway+Security+OAuth2集成方案六、OAuth2.0集成一、Gateway核心概念与架构SpringCloudGateway是基于Spring5.0、SpringBoot2.0
- Android MVI架构模式详解
追光者、Z
kotlinMVI模式androidandroidkotlin
MVI概念MVI(Model-View-Intent)是一种Android应用架构模式,旨在通过单向数据流和不可变性来简化应用的状态管理。MVI的核心思想是将用户操作、状态更新和界面渲染分离,确保应用的状态可预测且易于调试。MVI的核心组件Model(模型):表示应用的状态。Model是不可变的,通常是一个数据类,包含所有需要展示的信息。例如,一个加载数据的界面可能包含Loading、Succes
- 推荐项目:基于MVI架构的Kotlin应用示例
卓桢琳Blackbird
推荐项目:基于MVI架构的Kotlin应用示例去发现同类优质开源项目:https://gitcode.com/项目介绍该项目是一个精心设计的开源示例,旨在展示如何在Android应用中实施Model-View-Intent(MVI)架构模式。这个代码库是针对MitchTabian的视频课程而创建的,提供了一种清晰且可维护的方式来构建现代Android应用程序。通过观看配套课程或直接探索代码,您将学
- Kotlin中快速实现MVI架构
bubiyoushang888
kotlin开发语言android
MVI(Model-View-Intent)是一种现代的架构模式,广泛应用于Android开发中,以提高代码的可维护性和可测试性。本文将详细介绍如何在Kotlin中快速实现MVI架构,帮助开发者更好地管理应用的状态和交互。一、MVI架构简介MVI架构的核心思想是单向数据流和不可变状态。MVI模式主要包含以下三个部分:Model:表示应用的状态。View:负责展示Model,并接收用户输入。Inte
- 【技术支持】Android11 中获取应用列表
daban2008
技术支持android-studioandroidjetpack
getInstalledApplications方式获取的比较杂乱,有时无法获取第三方应用建议使用queryIntentActivitiesPackageManagerpm=getPackageManager();//Listapps=pm.getInstalledApplications(PackageManager.GET_META_DATA);Intentintent=newIntent(I
- PotPlayer 安装 madVR、LAV Filters 以提升解码能力和视频音频效果
SunsPlanter
办公软件学习
PotPlayer自带的解码器并不是最好,如下两张截图都是出自TOPGUN:Maverick较暗、灰蒙蒙的一张,是安装插件之前明亮的一张,是安装插件之后详细安装参考https://www.bilibili.com/video/BV1UV5qzuE74?spm_id_from=333.788.videopod.sections&vd_source=1e12faf053b82a0210ff4ab21c
- 【AAOS】【源码分析】用户管理(三)-- 用户启动
IT_码农
AndroidAutomativeAAOSCarUserManager
用户首次启动大体流程用户启动过程中,涉及很多模块,其中UserController模块负责居中调度创建并启动主用户:系统在首次启动时自动创建User0,在SystemReady时,由SystemServer启动系统用户(User0),主用户是以Background方式运行,在用户启动过程中发送一系列广播通知,包括:android.intent.action.USER_STARTED:表示用户已经启
- mp4 video转为ros image message
wendox
UAVROS
有时候我们需要把自己录制的视频使用ros广播出去,这时需要读取这个视频的文件,然后将image转换为sensor_image,可以使用cv_bridge完成图像转换的工作importsys,timeimportnumpyasnpfromscipy.ndimageimportfiltersimportcv2importroslibimportrospyfromsensor_msgs.msgimpor
- fmc接口定义_FMC接口标准
【实例简介】FMC标准接口说明,英文版现在Xilinx、Alter等主流FPGA厂商大多使用FMC接口,其对应有很多FMC扩展子板(Thispageleftblankintentionally.ANSI/VITA57.1-2008ApprovedAmericanNationANSIAmericanNationalStandardforfpgamezzaninecard(FMC)StandardSe
- android 页面跳转代码
wu_android
androidjava
Activity跳转页面代码Intentintent=newIntent();intent.setClass(MainActivity.this,activity_second.class);startActivity(intent);其中activity_second为要跳转的页面要在AndroidManifest.xml写activity,不然会报错例如>其中"com.example.repo
- 插入表主键冲突做更新
a-john
有以下场景:
用户下了一个订单,订单内的内容较多,且来自多表,首次下单的时候,内容可能会不全(部分内容不是必须,出现有些表根本就没有没有该订单的值)。在以后更改订单时,有些内容会更改,有些内容会新增。
问题:
如果在sql语句中执行update操作,在没有数据的表中会出错。如果在逻辑代码中先做查询,查询结果有做更新,没有做插入,这样会将代码复杂化。
解决:
mysql中提供了一个sql语
- Android xml资源文件中@、@android:type、@*、?、@+含义和区别
Cb123456
@+@?@*
一.@代表引用资源
1.引用自定义资源。格式:@[package:]type/name
android:text="@string/hello"
2.引用系统资源。格式:@android:type/name
android:textColor="@android:color/opaque_red"
- 数据结构的基本介绍
天子之骄
数据结构散列表树、图线性结构价格标签
数据结构的基本介绍
数据结构就是数据的组织形式,用一种提前设计好的框架去存取数据,以便更方便,高效的对数据进行增删查改。正确选择合适的数据结构,对软件程序的高效执行的影响作用不亚于算法的设计。此外,在计算机系统中数据结构的作用也是非同小可。例如常常在编程语言中听到的栈,堆等,就是经典的数据结构。
经典的数据结构大致如下:
一:线性数据结构
(1):列表
a
- 通过二维码开放平台的API快速生成二维码
一炮送你回车库
api
现在很多网站都有通过扫二维码用手机连接的功能,联图网(http://www.liantu.com/pingtai/)的二维码开放平台开放了一个生成二维码图片的Api,挺方便使用的。闲着无聊,写了个前台快速生成二维码的方法。
html代码如下:(二维码将生成在这div下)
? 1
&nbs
- ImageIO读取一张图片改变大小
3213213333332132
javaIOimageBufferedImage
package com.demo;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import javax.imageio.ImageIO;
/**
* @Description 读取一张图片改变大小
* @author FuJianyon
- myeclipse集成svn(一针见血)
7454103
eclipseSVNMyEclipse
&n
- 装箱与拆箱----autoboxing和unboxing
darkranger
J2SE
4.2 自动装箱和拆箱
基本数据(Primitive)类型的自动装箱(autoboxing)、拆箱(unboxing)是自J2SE 5.0开始提供的功能。虽然为您打包基本数据类型提供了方便,但提供方便的同时表示隐藏了细节,建议在能够区分基本数据类型与对象的差别时再使用。
4.2.1 autoboxing和unboxing
在Java中,所有要处理的东西几乎都是对象(Object)
- ajax传统的方式制作ajax
aijuans
Ajax
//这是前台的代码
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <% String path = request.getContextPath(); String basePath = request.getScheme()+
- 只用jre的eclipse是怎么编译java源文件的?
avords
javaeclipsejdktomcat
eclipse只需要jre就可以运行开发java程序了,也能自动 编译java源代码,但是jre不是java的运行环境么,难道jre中也带有编译工具? 还是eclipse自己实现的?谁能给解释一下呢问题补充:假设系统中没有安装jdk or jre,只在eclipse的目录中有一个jre,那么eclipse会采用该jre,问题是eclipse照样可以编译java源文件,为什么呢?
&nb
- 前端模块化
bee1314
模块化
背景: 前端JavaScript模块化,其实已经不是什么新鲜事了。但是很多的项目还没有真正的使用起来,还处于刀耕火种的野蛮生长阶段。 JavaScript一直缺乏有效的包管理机制,造成了大量的全局变量,大量的方法冲突。我们多么渴望有天能像Java(import),Python (import),Ruby(require)那样写代码。在没有包管理机制的年代,我们是怎么避免所
- 处理百万级以上的数据处理
bijian1013
oraclesql数据库大数据查询
一.处理百万级以上的数据提高查询速度的方法: 1.应尽量避免在 where 子句中使用!=或<>操作符,否则将引擎放弃使用索引而进行全表扫描。
2.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 o
- mac 卸载 java 1.7 或更高版本
征客丶
javaOS
卸载 java 1.7 或更高
sudo rm -rf /Library/Internet\ Plug-Ins/JavaAppletPlugin.plugin
成功执行此命令后,还可以执行 java 与 javac 命令
sudo rm -rf /Library/PreferencePanes/JavaControlPanel.prefPane
成功执行此命令后,还可以执行 java
- 【Spark六十一】Spark Streaming结合Flume、Kafka进行日志分析
bit1129
Stream
第一步,Flume和Kakfa对接,Flume抓取日志,写到Kafka中
第二部,Spark Streaming读取Kafka中的数据,进行实时分析
本文首先使用Kakfa自带的消息处理(脚本)来获取消息,走通Flume和Kafka的对接 1. Flume配置
1. 下载Flume和Kafka集成的插件,下载地址:https://github.com/beyondj2ee/f
- Erlang vs TNSDL
bookjovi
erlang
TNSDL是Nokia内部用于开发电信交换软件的私有语言,是在SDL语言的基础上加以修改而成,TNSDL需翻译成C语言得以编译执行,TNSDL语言中实现了异步并行的特点,当然要完整实现异步并行还需要运行时动态库的支持,异步并行类似于Erlang的process(轻量级进程),TNSDL中则称之为hand,Erlang是基于vm(beam)开发,
- 非常希望有一个预防疲劳的java软件, 预防过劳死和眼睛疲劳,大家一起努力搞一个
ljy325
企业应用
非常希望有一个预防疲劳的java软件,我看新闻和网站,国防科技大学的科学家累死了,太疲劳,老是加班,不休息,经常吃药,吃药根本就没用,根本原因是疲劳过度。我以前做java,那会公司垃圾,老想赶快学习到东西跳槽离开,搞得超负荷,不明理。深圳做软件开发经常累死人,总有不明理的人,有个软件提醒限制很好,可以挽救很多人的生命。
相关新闻:
(1)IT行业成五大疾病重灾区:过劳死平均37.9岁
- 读《研磨设计模式》-代码笔记-原型模式
bylijinnan
java设计模式
声明: 本文只为方便我个人查阅和理解,详细的分析以及源代码请移步 原作者的博客http://chjavach.iteye.com/
/**
* Effective Java 建议使用copy constructor or copy factory来代替clone()方法:
* 1.public Product copy(Product p){}
* 2.publi
- 配置管理---svn工具之权限配置
chenyu19891124
SVN
今天花了大半天的功夫,终于弄懂svn权限配置。下面是今天收获的战绩。
安装完svn后就是在svn中建立版本库,比如我本地的是版本库路径是C:\Repositories\pepos。pepos是我的版本库。在pepos的目录结构
pepos
component
webapps
在conf里面的auth里赋予的权限配置为
[groups]
- 浅谈程序员的数学修养
comsci
设计模式编程算法面试招聘
浅谈程序员的数学修养
- 批量执行 bulk collect与forall用法
daizj
oraclesqlbulk collectforall
BULK COLLECT 子句会批量检索结果,即一次性将结果集绑定到一个集合变量中,并从SQL引擎发送到PL/SQL引擎。通常可以在SELECT INTO、
FETCH INTO以及RETURNING INTO子句中使用BULK COLLECT。本文将逐一描述BULK COLLECT在这几种情形下的用法。
有关FORALL语句的用法请参考:批量SQL之 F
- Linux下使用rsync最快速删除海量文件的方法
dongwei_6688
OS
1、先安装rsync:yum install rsync
2、建立一个空的文件夹:mkdir /tmp/test
3、用rsync删除目标目录:rsync --delete-before -a -H -v --progress --stats /tmp/test/ log/这样我们要删除的log目录就会被清空了,删除的速度会非常快。rsync实际上用的是替换原理,处理数十万个文件也是秒删。
- Yii CModel中rules验证规格
dcj3sjt126com
rulesyiivalidate
Yii cValidator主要用法分析:
yii验证rulesit 分类: Yii yii的rules验证 cValidator主要属性 attributes ,builtInValidators,enableClientValidation,message,on,safe,skipOnError
 
- 基于vagrant的redis主从实验
dcj3sjt126com
vagrant
平台: Mac
工具: Vagrant
系统: Centos6.5
实验目的: Redis主从
实现思路
制作一个基于sentos6.5, 已经安装好reids的box, 添加一个脚本配置从机, 然后作为后面主机从机的基础box
制作sentos6.5+redis的box
mkdir vagrant_redis
cd vagrant_
- Memcached(二)、Centos安装Memcached服务器
frank1234
centosmemcached
一、安装gcc
rpm和yum安装memcached服务器连接没有找到,所以我使用的是make的方式安装,由于make依赖于gcc,所以要先安装gcc
开始安装,命令如下,[color=red][b]顺序一定不能出错[/b][/color]:
建议可以先切换到root用户,不然可能会遇到权限问题:su root 输入密码......
rpm -ivh kernel-head
- Remove Duplicates from Sorted List
hcx2013
remove
Given a sorted linked list, delete all duplicates such that each element appear only once.
For example,Given 1->1->2, return 1->2.Given 1->1->2->3->3, return&
- Spring4新特性——JSR310日期时间API的支持
jinnianshilongnian
spring4
Spring4新特性——泛型限定式依赖注入
Spring4新特性——核心容器的其他改进
Spring4新特性——Web开发的增强
Spring4新特性——集成Bean Validation 1.1(JSR-349)到SpringMVC
Spring4新特性——Groovy Bean定义DSL
Spring4新特性——更好的Java泛型操作API
Spring4新
- 浅谈enum与单例设计模式
247687009
java单例
在JDK1.5之前的单例实现方式有两种(懒汉式和饿汉式并无设计上的区别故看做一种),两者同是私有构
造器,导出静态成员变量,以便调用者访问。
第一种
package singleton;
public class Singleton {
//导出全局成员
public final static Singleton INSTANCE = new S
- 使用switch条件语句需要注意的几点
openwrt
cbreakswitch
1. 当满足条件的case中没有break,程序将依次执行其后的每种条件(包括default)直到遇到break跳出
int main()
{
int n = 1;
switch(n) {
case 1:
printf("--1--\n");
default:
printf("defa
- 配置Spring Mybatis JUnit测试环境的应用上下文
schnell18
springmybatisJUnit
Spring-test模块中的应用上下文和web及spring boot的有很大差异。主要试下来差异有:
单元测试的app context不支持从外部properties文件注入属性
@Value注解不能解析带通配符的路径字符串
解决第一个问题可以配置一个PropertyPlaceholderConfigurer的bean。
第二个问题的具体实例是:
 
- Java 定时任务总结一
tuoni
javaspringtimerquartztimertask
Java定时任务总结 一.从技术上分类大概分为以下三种方式: 1.Java自带的java.util.Timer类,这个类允许你调度一个java.util.TimerTask任务; 说明: java.util.Timer定时器,实际上是个线程,定时执行TimerTask类 &
- 一种防止用户生成内容站点出现商业广告以及非法有害等垃圾信息的方法
yangshangchuan
rank相似度计算文本相似度词袋模型余弦相似度
本文描述了一种在ITEYE博客频道上面出现的新型的商业广告形式及其应对方法,对于其他的用户生成内容站点类型也具有同样的适用性。
最近在ITEYE博客频道上面出现了一种新型的商业广告形式,方法如下:
1、注册多个账号(一般10个以上)。
2、从多个账号中选择一个账号,发表1-2篇博文