Kotlin高仿微信-第2篇-登录

news2025/1/16 3:48:17

Kotlin高仿微信-项目实践58篇详细讲解了各个功能点,包括:注册、登录、主页、单聊(文本、表情、语音、图片、小视频、视频通话、语音通话、红包、转账)、群聊、个人信息、朋友圈、支付服务、扫一扫、搜索好友、添加好友、开通VIP等众多功能。

Kotlin高仿微信-项目实践58篇,点击查看详情

效果图:

实现代码:

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:app="http://schemas.android.com/apk/res-auto">

    <androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:id="@+id/login_root_layout"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:background="@drawable/wc_base_bg">


        <include
            layout="@layout/wc_base_top_title"/>

        <androidx.appcompat.widget.AppCompatImageView
            android:id="@+id/login_icon"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginTop="120dp"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/login_root_layout"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            android:src="@drawable/wc_logo3"/>

        <TextView
            android:id="@+id/login_account_tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginStart="20dp"
            android:layout_marginTop="60dp"
            android:text="@string/user_account"
            android:textSize="16sp"
            android:textColor="@color/color_171717"
            android:layout_marginBottom="@dimen/distance_40"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/login_icon"/>

        <EditText
            android:id="@+id/login_account"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginEnd="20dp"
            android:layout_marginStart="70dp"
            android:paddingLeft="10dp"
            android:paddingVertical="8dp"
            android:inputType="textEmailAddress"
            app:layout_constraintEnd_toStartOf="@+id/login_account_tv"
            app:layout_constraintTop_toTopOf="@+id/login_account_tv"
            app:layout_constraintBottom_toBottomOf="@+id/login_account_tv"
            android:background="@drawable/base_edittext_selector"
            android:hint="@string/login_user_tip"
            android:textColorHint="@color/gray"
            android:textSize="18sp"
            android:text=""
            android:textColor="@color/gray_text" />


        <TextView
            android:id="@+id/login_password_tv"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginStart="20dp"
            android:layout_marginTop="50dp"
            android:text="@string/login_password"
            android:textSize="16sp"
            android:textColor="@color/black"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toBottomOf="@+id/login_account_tv" />

        <EditText
            android:id="@+id/login_password"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginEnd="20dp"
            android:layout_marginStart="70dp"
            android:paddingLeft="10dp"
            android:paddingVertical="10dp"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toEndOf="@+id/login_password_tv"
            app:layout_constraintTop_toTopOf="@+id/login_password_tv"
            app:layout_constraintBottom_toBottomOf="@+id/login_password_tv"
            android:hint="@string/login_password_tip"
            android:textColorHint="@color/gray"
            android:textSize="16sp"
            android:inputType="textPassword"
            android:background="@drawable/base_edittext_selector"
            android:text=""
            android:textColor="@color/gray_text" />


        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_register"
            android:layout_width="140dp"
            android:layout_height="wrap_content"
            android:background="@drawable/wc_base_green_selector"
            android:layout_marginTop="80dp"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintEnd_toStartOf="@+id/btn_login"
            app:layout_constraintTop_toBottomOf="@+id/login_password_tv"
            android:text="@string/register"
            android:textColor="@color/white"
            android:textSize="18sp"
            android:textStyle="bold"/>


        <androidx.appcompat.widget.AppCompatButton
            android:id="@+id/btn_login"
            android:layout_width="140dp"
            android:layout_height="wrap_content"
            android:background="@drawable/wc_base_green_selector"
            app:layout_constraintTop_toTopOf="@+id/btn_register"
            app:layout_constraintBottom_toBottomOf="@+id/btn_register"
            app:layout_constraintStart_toEndOf="@+id/btn_register"
            app:layout_constraintEnd_toEndOf="parent"
            android:textColor="@color/white"
            android:text="@string/login"
            android:textSize="18sp"
            android:textStyle="bold"/>


    </androidx.constraintlayout.widget.ConstraintLayout>
</layout>

/**
 * Author : wangning
 * Email : maoning20080809@163.com
 * Date : 2022/4/19 14:41
 * Description : 登录页面
 */
class LoginFragment : BaseDataBindingFragment<WcMainLoginBinding>() , ServiceConnection {

    override fun getLayoutRes() = R.layout.wc_main_login
    private val userViewModel : UserViewModel by viewModels()
    private var account: String = ""
    private var password: String = ""
    private var xmppService: XmppService? = null
    private var navController : NavController? = null
    //是否在注册成功页面,注册成功返回
    private var isRegisterBack:Boolean = false

    override fun onAttach(context: Context) {
        super.onAttach(context)
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }

    override fun onActivityCreated(savedInstanceState: Bundle?) {
        super.onActivityCreated(savedInstanceState)
    }

    override fun onStart() {
        super.onStart()
    }

    override fun onConfigurationChanged(newConfig: Configuration) {
        super.onConfigurationChanged(newConfig)
    }

    override fun onHiddenChanged(hidden: Boolean) {
        super.onHiddenChanged(hidden)
        if(!hidden){
            getFocus();
        }
    }

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        super.onViewCreated(view, savedInstanceState)
        initViews()
    }

    private fun initViews() {
        super.builder().setTitleContent(R.string.wc_base_top_login)
        base_top_back.setOnClickListener {
            System.exit(1)
        }
        navController = Navigation.findNavController(btn_register)
        navController?.previousBackStackEntry?.savedStateHandle?.getLiveData<Boolean>(CommonUtils.User.IS_REGISTER_BACK)?.observe(viewLifecycleOwner){
          isRegisterBack = it
        }

        var intent = Intent(requireActivity(), XmppService::class.java)
        requireActivity().bindService(intent, this , BIND_AUTO_CREATE)

        btn_register.setOnClickListener {
            Navigation.findNavController(it).navigate(R.id.action_register)
        }

        btn_login.setOnClickListener { login() }

        var accountTemp = DataStoreUtils.get(DataStoreParams.User.DS_ACCOUNT, "") as String
        var passwordTemp = DataStoreUtils.get(DataStoreParams.User.DS_PASSWORD, "") as String
        CoroutineScope(Dispatchers.Main).launch {
            delay(200)
            login_account.setText(accountTemp)
            login_password.setText(passwordTemp)
        }

        login_root_layout.setOnClickListener {
            TagUtils.d("点击根节点 Click ")
            SoftInputUtils.hideSoftInput(login_account)
        }

        userViewModel.insertUserData.observe(viewLifecycleOwner){
            //如果同步成功,则重新登录
            if(it){
                DataStoreUtils.put(DataStoreParams.User.DS_OPENFILE_REGISTER, true)
                login()
            } else {
                ToastUtils.makeText(requireActivity(), BaseUtils.getString(R.string.wc_login_failure))
                dismissLoadingDialog()
            }
        }
    }


    private var loadingUtils : BaseDialogUtils? = null
    //显示加载对话框
    private fun showLoadingDialog(){
        loadingUtils = BaseDialogUtils(requireActivity())
        loadingUtils!!.builder()
            .hideCancel()
            .hideConfirm()
            .setCancelable(true)
            .setOnLoadingClick(object : BaseDialogUtils.OnLoadingClick{
                override fun onClickCancel() {
                    ToastUtils.makeText(requireActivity(), "对话框取消按钮")
                }

                override fun onClickConfirm() {
                    ToastUtils.makeText(requireActivity(), "对话框确定按钮")
                }
            })
        loadingUtils?.show()
    }

    //隐藏加载对话框
    private fun dismissLoadingDialog(){
        loadingUtils?.dismiss()
    }

    //登录
    private fun login() {
        var isRegisterSuccess = DataStoreUtils.get(DataStoreParams.User.DS_OPENFILE_REGISTER, false) as Boolean
        TagUtils.d("login 开始 ${isRegisterSuccess}")
        account = login_account.text.toString()
        password = login_password.getText().toString()
        if (account?.length == 0 || password?.length == 0) {
            ToastUtils.makeText(requireActivity(), "帐号或密码不能为空")
            return
        } else if (!isRegisterSuccess && isRegisterBack){
            var name = DataStoreUtils.get(DataStoreParams.User.DS_NICKNAME, "") as String
            if(TextUtils.isEmpty(name)){
                name = account
            }
            //如果在openfire注册成功,没有同步到web服务器,点击登录按钮,则重新同步
            var userBean = UserBean(account = account, name = name, nickName = name,  address = "", email = "",phone = "", avatar = "", birthday = "", note = "")
            userViewModel.insertUser(userBean)
        } else {
            SoftInputUtils.hideSoftInput(login_account)
            showLoadingDialog()
            try {
                xmppService?.login(account, password)
            } catch (e : Exception) {
                e.printStackTrace()
                //ToastUtils.makeText(R.string.wc_login_failure)
                dismissLoadingDialog()
            }
        }
    }

    /*private fun initCall() {
        //在前台了,发送广播 调起权限判断弹窗
        val viop = Intent()
        val intent: Intent = requireActivity().intent
        viop.putExtra("room", intent.getStringExtra("room"))
        viop.putExtra("audioOnly", intent.getBooleanExtra("audioOnly", false))
        viop.putExtra("inviteId", intent.getStringExtra("inviteId"))
        viop.putExtra("inviteUserName", intent.getStringExtra("inviteUserName"))
        //        viop.putExtra("msgId", intent.getLongExtra("msgId", 0));
        viop.putExtra("userList", intent.getStringExtra("userList"))
        viop.action = Utils.ACTION_VOIP_RECEIVER
        viop.component = ComponentName(WcApp.getInstance().getPackageName(), VoipReceiver::class.java.getName())
        WcApp.getInstance().sendBroadcast(viop)
    }*/

    private fun processLoginResult(type : Int) {
        CoroutineScope(Dispatchers.Main).launch {
            dismissLoadingDialog()
            if(type == 1){
                ToastUtils.makeText(R.string.wc_login_success)
                /*var isFromCall = requireActivity().intent.getBooleanExtra("isFromCall", false)
                if (isFromCall) { //无权限,来电申请权限会走这里
                    initCall()
                }*/
                DataStoreUtils.get(DataStoreParams.Install.DS_FIRST_INSTALL, true)
                loginSusscess()
            } else {
                ToastUtils.makeText(R.string.wc_login_failure)
            }
        }
    }

    override fun onServiceConnected(p0: ComponentName?, binder: IBinder?) {
        TagUtils.d("LoginService 连接成功")
        var xmppBinder = binder as XmppService.XmppBinder
        xmppService = xmppBinder.service
        xmppService?.setCallback(object : XmppService.LoginCallback{
            override fun onFailure() {
                //TagUtils.d("LoginFragment登录失败")
                processLoginResult(2)
            }

            override fun onSuccess(account: String, password: String) {
                TagUtils.d("LoginFragment登录成功")
                processLoginSuccess()
            }
        })
    }

    override fun onServiceDisconnected(p0: ComponentName?) {
        TagUtils.d("LoginService onServiceDisconnected 断开连接")
    }

    private fun processLoginSuccess(){
        CoroutineScope(Dispatchers.IO).launch {
            DataStoreUtils.put(DataStoreParams.User.DS_ACCOUNT, account!!)
            DataStoreUtils.put(DataStoreParams.User.DS_PASSWORD, password!!)
            DataStoreUtils.put(DataStoreParams.User.DS_USER_EXIST, false)

            //登录成功, 保存本地信息
            var userBean = UserRepository.getUserByAccount(account!!)
            TagUtils.d("登录成功:${account} , ${password} , userBean = ${userBean}")

            // 设置用户名
            WcApp.getInstance().username  = account!!
            // 添加登录回调
            SocketManager.getInstance().addUserStateCallback(object : IUserState{
                override fun userLogin() {
                    TagUtils.d("登录视频服务器成功login")
                }

                override fun userLogout() {
                    TagUtils.d("登录视频服务器失败logout")
                }
            })
            // 连接socket:登录
            SocketManager.getInstance().connect(BaseUtils.WS, account!!, 0)

            processLoginResult(1)
        }
    }


    //登录成功
    private fun loginSusscess() {
        Navigation.findNavController(btn_login).popBackStack(R.id.main_start_dest, true)
        EventBus.getDefault().post(EventMsgBean(EventMsgBean.TYPE_LOGIN_SUCCESS))
    }

    override fun onResume() {
        super.onResume()
        getFocus()
    }

    private fun getFocus(){
        view?.isFocusableInTouchMode = true
        view?.requestFocus()
        view?.setOnKeyListener { view, keyCode, event ->
            if(event.action == KeyEvent.ACTION_UP && keyCode == KeyEvent.KEYCODE_BACK){
                System.exit(1)
            }
            true
        }
    }

}
fun login(account : String, password : String){
    CoroutineScope(Dispatchers.IO).launch {
        try {
            if(!xmppConnection.isAuthenticated){
                xmppConnection.connect()
                // 连接服务器
                xmppConnection.login(account, password)
                // 连接服务器成功,更改在线状态
                val presence = Presence(Presence.Type.available)
                xmppConnection.sendPacket(presence)
                TagUtils.d("用户登录状态:${xmppConnection.isAuthenticated} , ${xmppConnection.isConnected}")
            }
            if(xmppConnection.isAuthenticated){
                WcApp.setXmppConnection(xmppConnection)
                ChatManagerUtils.getInstance().addChatListener()
                AddFileListener.addFileListerer()
                AddFriendsListener.addSubscriptionListener()
                checkConnectionListener = CheckConnectionListener()
                xmppConnection.addConnectionListener(checkConnectionListener)

                TagUtils.d("xmpp 管理连接成功 ")

                var result = UserService.getApi().getUser(account)
                TagUtils.d("syncUser 用户返回数据:${result}")
                if(result.isSuccess){
                    //插入本地数据库
                    UserRepository.insertUser(GsonUtils.getUserBean(result.data!!))
                    //处理同步信息

                } else {
                    //如果在openfire注册成功,没有同步到web服务器,或者web服务器已经不存在该账户,只能重新创建
                    var userBean = UserBean(account = account, name = account, nickName = account,  address = "", email = "",phone = "", avatar = "", birthday = "", note = "")
                    var userStr = Gson().toJson(userBean)
                    UserRepository.insertUser(userBean)
                    UserRepository.insertUserServer(userStr)
                }

                loginCallback?.onSuccess(account, password)
            } else {
                loginCallback?.onFailure()
            }

        } catch (e: Exception){
            //ToastUtils.makeText(R.string.wc_login_failure)
            TagUtils.d("用户登录状态:${e.message}")
            e.printStackTrace()
            loginCallback?.onFailure()
        }
    }
}

//登录成功,同步数据
private fun showLoadingDialog(){
    var isFirstLogin = DataStoreUtils.get(DataStoreParams.User.DS_IS_FIRST_LOGIN, true) as Boolean
    //首次安装
    if(!isFirstLogin){
        return
    }
    var account = DataStoreUtils.getAccount()
    if(TextUtils.isEmpty(account)){
        return
    }
    var view = LayoutInflater.from(this).inflate(R.layout.wc_sync_first_data, null)

    loadingUtils = BaseDialogUtils(this)
    loadingUtils!!.builder()
        .showView(view)
        .hideCancel()
        .hideConfirm()
        .setCancelable(false)
    loadingUtils?.show()
    SyncRepository.syncFirst(account, packageName, object : ICommonCallback{
        override fun onFailure(obj: Any) {
            dismissLoadingDialog()
        }

        override fun onSuccess(obj: Any) {
            dismissLoadingDialog()
        }
    })
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/44964.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【安卓逆向】去除云注入(使用MT论坛dl的方法总结拓展)

1 需求 因为最近使用的虚拟机突然不能用了&#xff0c;被人云注入强制弹窗&#xff0c;如下图&#xff1a;&#xff08;这一看就是云注入了&#xff09; 2 大佬的方法 如图&#xff08;MT大佬分享的&#xff0c;感兴趣的朋友可以去大佬主页看看他其他文章&#xff09;&…

把backtrader改造成金融强化学习回测引擎

原创文章第119篇&#xff0c;专注“个人成长与财富自由、世界运作的逻辑&#xff0c; AI量化投资”。 继续强化学习应用于金融投资。 我们的AI量化平台&#xff0c;针对传统规则量化策略&#xff0c;进行了“积木式”的拆分&#xff0c;这种拆分的好处&#xff0c;就是最大化…

【三维目标检测】Part-A2(二)

PartA2数据和源码配置调试过程请参考上一篇博文&#xff1a;【三维目标检测】Part-A2&#xff08;一&#xff09;_Coding的叶子的博客-CSDN博客。本文主要详细介绍PartA2网络结构及其运行中间状态。 1 PointRCNN模型总体过程 Part-A2的整体结构如下图所示&#xff0c;主要包括…

据2019年中国社交电商行业研究报告称,电商正处于更新换代的时期

引言&#xff1a; 据艾瑞咨询《2019年中国社交电商行业研究报告》示,传统主流电商平台用户与交易规模增速均呈现逐渐放缓的趋势,平台亟需找到更高效、低价、高粘性的流量来源,来跳出竞争日益激烈获客成本持续攀升的困境。移动互联网时代,微信、QQ、微博、快手、抖音等社交类AP…

Wordpress模板主题中functions.php常用功能代码与常用插件(持续收集整理)

用Wordpress建站的初学者一定会需要用到的Wordpress模板主题中functions.php常用功能代码与常用插件。慢慢持续收集整理....... 目录 一、Wordpress模板主题中functions文件常用的代码 二、Wordpress自定义字段的设定与调用代码&#xff08;系统常规自定义字段&#xff09; …

【面试宝典】Java八股文之Dubbo 面试题

Dubbo 面试题1、为什么要用 Dubbo?2、Dubbo 的整体架构设计有哪些分层?3、默认使用的是什么通信框架&#xff0c;还有别的选择吗?4、服务调用是阻塞的吗?5、一般使用什么注册中心?还有别的选择吗?6、默认使用什么序列化框架&#xff0c;你知道的还有哪些?7、服务提供者能…

下沉市场投资热度提升 7天酒店打造酒店投资“极致性价比”

近日&#xff0c;7天酒店 “总裁面对面”酒店投资云沙龙活动举办&#xff0c;通过微信、抖音双平台联合直播&#xff0c;多维度探讨酒店行业的“新蓝海”机遇以及下沉市场的投资模式&#xff0c;助力更多投资人把握新的市场红利。 经济型酒店拥抱“新蓝海” 下沉市场投资热度提…

【学习笔记60】JavaScript原型链的理解

一、万物皆对象 JS中, 万物都可以都可以称为对象 1、对象概念 含义1: 一种数据格式 {key: value, key2: value2}含义2: 某一类事务的实例(某一类内容中的真实个体) 2、说明 arr1就是Array这一类内容中的某一个真实个体数组也可以算作一个对象(Array 这一类事务中的一个个体) …

天宇优配|酒企没借壳!标准股份股价上演A杀,两跌停

11月28日晚间&#xff0c;接连三日大跌的规范股份&#xff08;600302.SZ&#xff09;发布股价异动公告&#xff0c;再次否定了借壳和重组风闻。当日龙虎榜该股获净卖出774.89万元&#xff0c;闻名游资“赵老哥”常用席位中国银河绍兴现身卖一席位。另外&#xff0c;也有多家本地…

五笔会消亡吗

今天第一次看到“五笔会消亡”的说法。一看好像也没有什么消不消亡的说法&#xff0c;但是深入想一想好像的确是有一个现象90 后 00后使用五笔的应该会少很多&#xff0c;可能用的非常少。 从五笔与拼音在百度的搜索比例也可以看出&#xff0c;的确在2015~2016年间有了转折&am…

【Hack The Box】linux练习-- Paper

HTB 学习笔记 【Hack The Box】linux练习-- Paper &#x1f525;系列专栏&#xff1a;Hack The Box &#x1f389;欢迎关注&#x1f50e;点赞&#x1f44d;收藏⭐️留言&#x1f4dd; &#x1f4c6;首发时间&#xff1a;&#x1f334;2022年11月27日&#x1f334; &#x1f36…

第一期 微信云开发小程序介绍-生活智打卡

目录 1.项目介绍 1.1 开发背景 1.2 项目简介 1.2.1 雏形 1.2.2 现状 1.2.3 展望 1.3 市场分析 1.3.1 目标用户 1.3.2 市场需求分析 1.4 系统需求 1.5 竞品分析 2.产品设计 2.1产品功能 2.1.1 智打卡 2.1.2 发现 2.1.2 我的 2.2交互设计 2.2.1 智打卡流程 2.2…

BP神经网络的梯度公式推导(三层结构)

本站原创文章&#xff0c;转载请说明来自《老饼讲解-BP神经网络》bp.bbbdata.com目录 一. 推导目标 1.1 梯度公式目标 1.2 本文梯度公式目标 二. 网络表达式梳理 2.1 梳理三层BP神经网络的网络表达式 三. 三层BP神经网络梯度推导过程 3.1 简化推导目标 3.2 输出层权重…

模拟电路(详细版)--放大电路的频率效应(RC电路)

一、高通电路 1.1传输特性 AuA_uAu​ RR1jωC\frac{R}{R \frac {1} { j \omega C}}RjωC1​R​   &#xff08;补充知识&#xff1a;j是复数域中的一个旋转因子&#xff09; 详细求解思路&#xff1a;   求解AuA_uAu​就是要求输入与输出的关系。 所以AuA_uAu​ U˙oU˙…

门店数字化转型| 美容院管理系统

随着互联网信息的高速发展&#xff0c;行业数字化进程加快&#xff0c;传统美容院面临几个难题。 1、竞争激烈、拓客困难。 美业巨头迅速扩张积压中小型门店生存空间。大多数中小美容院仍旧停留在发传单、口口相传的传统渠道&#xff0c;辐射范围非常有限。 2、投资周期长、资…

Linux网络配置管理

目录 一、实验目的 二、实验软硬件要求 三、实验预习 1、利用ifconfig 命令实现ip地址、MAC地址的配置&#xff0c;并测试网络连通性 1-1查看网卡 1-2临时改写eth0网卡地址 1-3测试连通性 2、通过修改interfaces配置文件&#xff0c;分别实现ip地址的动态配置和静态配置…

【信号处理】时序数据中的稀疏辅助信号去噪和模式识别(Matlab代码实现)

目录 一、概述 二、算例及仿真 &#x1f4e2;算例一&#xff1a; &#x1f4e2;算例二&#xff1a; &#x1f4e2;算例三&#xff1a; &#x1f4e2;算例四&#xff1a; &#x1f4e2;算例五&#xff1a; &#x1f4e2;算例六&#xff1a; 三、Matlab代码实现 一、概述…

【笔试强训】Day 4

&#x1f308;欢迎来到笔试强训专栏 (꒪ꇴ꒪(꒪ꇴ꒪ )&#x1f423;,我是Scort目前状态&#xff1a;大三非科班啃C中&#x1f30d;博客主页&#xff1a;张小姐的猫~江湖背景快上车&#x1f698;&#xff0c;握好方向盘跟我有一起打天下嘞&#xff01;送给自己的一句鸡汤&#x…

初识数据结构

目录 1. 集合的框架 集合框架的重要性 数据结构的介绍 算法的介绍 容器背后对应的数据结构 2. 时间复杂度和空间复杂度 算法效率 时间复杂度 时间复杂度的概念 大O的渐进表示法 常见的时间复杂度的计算 空间复杂度 空间复杂度的概念 从本章开始又要开始新的篇章&a…

流媒体传输 - RTMP 协议报文分析

握手之后&#xff0c;连接开始对一个或多个 chunk stream 进行合并。创建的每个块都有一个唯一 id 对其进行关联&#xff0c;这个 id 叫做 chunk stream id。这些块通过网络进行传输。传递时&#xff0c;每个块必须被完全发送才可以发送下一块。在接收端&#xff0c;这些块被根…