Kotlin基础(八):泛型

前言

本文主要讲解kotlin泛型,主要包括泛型基础,类型变异,类型投射,星号投射,泛型函数,泛型约束,泛型在Android中的使用。


Kotlin文章列表

Kotlin文章列表: 点击此处跳转查看


目录

在这里插入图片描述


1.1 泛型基础

在 Kotlin 中,泛型是一种类型参数化的机制,它允许我们编写具有通用性的代码。使用泛型,我们可以编写可以在不同类型上工作的代码,而无需为每个具体类型编写重复的代码。

Kotlin 中的泛型使用尖括号 < > 来定义,并且在类型声明中使用角括号来指定类型参数。例如,下面是一个简单的泛型函数的例子:

fun <T> printItem(item: T) {
    println(item.toString())
}

在这个例子中,<T> 表示这个函数是一个泛型函数,并且 T 是一个类型参数。我们可以在函数内部使用 T 来表示任意类型。在调用 printItem 函数时,编译器会根据实参的类型推断出 T 的具体类型。

下面是一个使用泛型的类的例子:

class Box<T>(val item: T) {
    fun getItem(): T {
        return item
    }
}

fun main() {
    val box = Box("Hello")
    val item: String = box.getItem()
    println(item)
}

在这个例子中,Box 类使用泛型类型参数 T,并且有一个泛型函数 getItem 返回类型为 T。在 main 函数中,我们创建了一个 Box<String> 对象,并调用了 getItem 函数来获取其中的值。

除了单个类型参数外,Kotlin 还支持多个类型参数的泛型定义。例如:

class Pair<A, B>(val first: A, val second: B) {
    // ...
}

在使用泛型时,我们可以限制类型参数的上界。这可以通过使用冒号 : 加上类型约束来实现。例如,我们可以指定一个类型参数必须是某个类的子类,或者实现了某个接口。下面是一个使用类型约束的例子:

fun <T : Number> convertToInt(value: T): Int {
    return value.toInt()
}

在这个例子中,<T : Number> 指定了类型参数 T 必须是 Number 类的子类。这样,我们可以在函数体内调用 Number 类的方法,例如 toInt

这些是 Kotlin 中泛型的基础知识。通过使用泛型,我们可以编写更具有通用性和复用性的代码,提高代码的灵活性和可读性。


1.2 类型变异

在 Kotlin 中,类型变异(Type Variance)是指在泛型类型中允许子类型关系的灵活性。Kotlin 提供了几种类型变异的修饰符,用于在使用泛型时指定类型参数的变异方式。这些修饰符包括:inout 和默认(不使用修饰符)。

  1. in:逆变(Contravariant)
    使用 in 修饰符声明的类型参数只能用作输入(即参数类型),不能用作输出(即返回类型)。它允许我们使用指定的类型参数的超类作为泛型类型的实参。在函数中,逆变类型参数只能作为方法的参数类型。

    例如,考虑一个逆变的接口 Comparable

    interface Comparable<in T> {
        fun compare(other: T): Int
    }
    

    这里的 in 修饰符表示类型参数 T 可以是方法的输入类型。这意味着我们可以使用 Comparable 接口的一个子类作为方法参数的类型,例如:

    fun sort(list: List<Comparable<in String>>) {
        // 排序逻辑...
    }
    

    在这个例子中,sort 函数接收一个 List,其中的元素类型是 Comparable<in String>,也就是说,我们可以传递 Comparable<String>Comparable<Any> 等类型的实例。

  2. out:协变(Covariant)
    使用 out 修饰符声明的类型参数只能用作输出(即返回类型),不能用作输入(即参数类型)。它允许我们使用指定的类型参数的子类作为泛型类型的实参。在函数中,协变类型参数只能作为方法的返回类型。

    例如,考虑一个协变的接口 Producer

    interface Producer<out T> {
        fun produce(): T
    }
    

    这里的 out 修饰符表示类型参数 T 可以是方法的输出类型。这意味着我们可以将 Producer 接口的一个超类赋值给类型为 Producer<out String> 的变量,例如:

    fun getProducer(): Producer<out String> {
        // 返回 Producer 的一个子类实例
    }
    

    在这个例子中,getProducer 函数的返回类型是 Producer<out String>,也就是说,我们可以将其返回值赋给类型为 Producer<String>Producer<Any> 等类型的变量。

  3. 默认
    如果没有指定任何修饰符,则类型参数既可以用作输入(参数类型),也可以用作输出(返回类型)。这种情况下,类型参数是不变的(Invariant)。

    例如,考虑一个不变的类 Box

    class Box<T>(val item: T) {
        fun getItem(): T {
               return item
           }
    }
    // 这里的类型参数 `T` 没有任何修饰符,因此它是不变的。在这种情况下,我们只能使用确切的类型作为泛型类型的实参。
    val box: Box<String> = Box("Hello")
    

通过使用逆变、协变和不变的修饰符,Kotlin 中的泛型提供了更灵活的类型参数关系,以便更好地处理不同的使用场景。


1.3 类型投射

在 Kotlin 中,类型投射(Type Projections)是指在使用泛型类型时对泛型参数的灵活性处理。类型投射允许我们在使用泛型类型时指定类型参数的上界或下界,从而在特定的上下文中对泛型类型进行限制。

Kotlin 中有三种类型投射的方式:

  1. 上界投射(Upper Bounds Projection):使用 out 关键字
    当我们只需要从泛型类型中获取值,而不需要修改该值时,我们可以使用上界投射。使用 out 关键字,我们可以指定类型参数的上界。这样,我们可以将泛型类型的实例赋值给类型参数的超类型。

    例如,考虑一个定义了 Producer 接口的泛型类:

    interface Producer<out T> {
        fun produce(): T
    }
    

    在这个例子中,out 关键字指定了类型参数 T 的上界,表示我们只能从 Producer 实例中获取 T 类型的值,而不能修改它。

  2. 下界投射(Lower Bounds Projection):使用 in 关键字
    当我们只需要向泛型类型中传递值,而不需要获取值时,我们可以使用下界投射。使用 in 关键字,我们可以指定类型参数的下界。这样,我们可以将泛型类型的实例赋值给类型参数的子类型。

    例如,考虑一个定义了 Consumer 接口的泛型类:

    interface Consumer<in T> {
        fun consume(item: T)
    }
    

    在这个例子中,in 关键字指定了类型参数 T 的下界,表示我们只能向 Consumer 实例中传递类型为 T 或其子类型的值,而不能传递超类型的值。

  3. 星投射(Star Projection):使用 * 关键字
    当我们既不需要从泛型类型中获取值,也不需要向泛型类型中传递值时,我们可以使用星投射。使用星投射,我们可以在使用泛型类型时忽略类型参数。

    例如,考虑一个定义了 Box 类的泛型类:

    class Box<T>(val item: T)
    

    在某些情况下,我们可能不关心具体的类型参数,而只是希望使用一个泛型类型的实例。这时,我们可以使用星投射来表示不关心具体类型参数的情况。

    val box: Box<*> = Box("Hello")
    

    在这个例子中,Box<*> 表示一个未知的类型参数,我们可以使用该实例,但无法获取其中的具体类型。

类型投射允许我们在使用泛型类型时灵活地指定类型参数的上界或下界,从而更好地适应不同的使用场景。


1.4 星号投射

在 Kotlin 中,星号投射(Star Projection)是一种用于处理泛型类型的特殊语法。它通常用于在某些情况下,你可能不需要或无法知道泛型类型的确切参数。星号投射允许你使用未知类型的泛型参数,并且在代码中只能执行一些有限的操作。

星号投射使用星号 (*) 表示未知类型。它可以应用于泛型类、接口和方法。

下面是一个星号投射的示例:

假设有一个简单的泛型类 Box,定义如下:

class Box<T>(private val item: T) {
    fun getItem(): T {
        return item
    }
}

现在,假设你有一个泛型类 Container,它包含一个 Box 的列表:

class Container<T>(private val boxes: List<Box<T>>) {
    fun getFirstItem(): T {
        return boxes[0].getItem()
    }
}

假设你有一个 Container 的实例,但是你并不知道 Container 的泛型类型参数 T 是什么。这时,你可以使用星号投射来处理未知类型。

fun processContainer(container: Container<*>) {
    val item = container.getFirstItem()
    // 在这里,由于我们不知道 Container 的泛型类型参数 T 是什么,只能执行一些有限的操作,如打印或进行简单处理。
    println("First item: $item")
}

在上面的示例中,Container<*> 表示一个具有未知泛型类型参数的 Container 实例。我们只能调用返回 T 类型的方法(在这里是 getFirstItem()),但无法向其中添加新的元素(因为我们不知道它的确切类型)。

需要注意的是,使用星号投射通常是在处理未知泛型类型的情况下使用的,而且通常涉及一些限制,因为编译器无法确切知道类型的信息。因此,在使用星号投射时,必须小心谨慎,确保你只执行合法且安全的操作。


1.5 泛型函数

1.5.1 泛型函数的简介

在 Kotlin 中,你可以创建泛型函数,这些函数允许你在调用时指定参数类型,并且在函数定义中可以使用这些类型参数进行通用的操作。泛型函数能够增加代码的重用性和类型安全性,因为它们可以适用于多种类型而不必为每个类型编写多个函数。

Kotlin 中定义泛型函数使用尖括号 (<>) 来声明泛型类型参数,并将它们放在函数名之前。下面是一个简单的示例:

// 在函数名前声明泛型类型参数 <T>
fun <T> printItem(item: T) {
    println(item)
}

在这个示例中,我们定义了一个名为 printItem 的泛型函数,它接收一个类型参数为 T 的参数 item,并简单地打印它。在函数体内,你可以像使用普通类型一样使用类型参数 T

现在,你可以在调用 printItem 函数时指定参数的类型,编译器会根据传入的参数类型自动推断出类型参数 T 的具体类型:

printItem("Hello, World!") // 调用时,Kotlin 会推断 T 为 String 类型
printItem(42) // 调用时,Kotlin 会推断 T 为 Int 类型
printItem(true) // 调用时,Kotlin 会推断 T 为 Boolean 类型

除了单个类型参数,你也可以在函数中定义多个泛型类型参数,这使得泛型函数更加灵活:

// 定义一个泛型函数,接收两个类型参数,并返回它们的 Pair
fun <T, U> createPair(first: T, second: U): Pair<T, U> {
    return Pair(first, second)
}

fun main() {
    val pair1 = createPair(10, "Hello")
    val pair2 = createPair(true, 3.14)
    
    println(pair1) // 输出: (10, Hello)
    println(pair2) // 输出: (true, 3.14)
}

以上就是 Kotlin 中定义泛型函数的基本语法和用法。泛型函数在处理通用算法或容器类型时特别有用,因为它们能够适用于不同类型的数据,提供更灵活和类型安全的代码。


1.5.2 kotlin泛型函数与java泛型函数对比

Kotlin 和 Java 都支持泛型函数,但它们在语法和使用上有一些不同。下面我们将对 Kotlin 泛型函数和 Java 泛型函数进行对比:

  1. 语法差异:

    Kotlin 泛型函数:

    fun <T> genericFunction(item: T): T {
        // 函数体
    }
    

    Java 泛型函数:

    public <T> T genericFunction(T item) {
        // 函数体
    }
    

    Kotlin 使用 <T> 放在函数名前来声明泛型类型,而 Java 则使用 <T> 放在返回类型之前。

  2. 类型推断:

    Kotlin 泛型函数:

    val result = genericFunction("Hello")
    // 在这里,Kotlin 可以推断出泛型类型 T 为 String
    

    Java 泛型函数:

    String result = genericFunction("Hello");
    // Java 不能自动推断泛型类型,需要显式指定类型
    

    Kotlin 的类型推断机制通常更加智能,能够自动推断泛型参数类型,而 Java 需要显式指定泛型类型。

  3. 通配符和星号投射:

    Kotlin 泛型函数:

    fun processList(list: List<*>) {
        // 函数体
    }
    

    Java 泛型函数:

    public void processList(List<?> list) {
        // 函数体
    }
    

    Kotlin 使用星号 * 表示未知类型(星号投射),而 Java 使用 ? 表示通配符。

  4. 多重约束:

    Kotlin 泛型函数:

    fun <T : Number> processNumber(item: T): T {
        // 函数体
    }
    

    Java 泛型函数:

    public <T extends Number> T processNumber(T item) {
        // 函数体
    }
    

    Kotlin 使用 : 来指定泛型类型的约束,而 Java 使用 extends 关键字。

总体来说,Kotlin 和 Java 的泛型函数在基本概念上是相似的,但具体的语法和使用方式有所不同。如果你熟悉 Java 的泛型,学习 Kotlin 的泛型应该相对容易,因为 Kotlin 是建立在 Java 基础之上并进行了一些改进。


1.6 泛型约束

在 Kotlin 中,你可以对泛型进行约束,以限制泛型类型参数的范围。泛型约束可以帮助你在泛型函数或泛型类中使用特定类型的功能,提高类型安全性并允许更具体的操作。

以下是在 Kotlin 中定义泛型约束的一些方法:

  1. 上界约束(Upper Bounds):

    通过使用 : 来指定泛型类型的上界,你可以限制泛型类型参数只能是指定类或其子类。这意味着类型参数必须是指定类或其子类的实例。

    // 示例:泛型类型参数 T 必须是 Number 类或其子类的实例
    fun <T : Number> printNumberInfo(number: T) {
        println("Value: ${number.toDouble()}")
    }
    
  2. 多重约束(Multiple Bounds):

    Kotlin 支持多重约束,即指定一个泛型类型参数必须满足多个约束条件。你可以使用 where 关键字来声明多个约束条件。

    // 示例:泛型类型参数 T 必须是 Number 类或其子类,并且实现了 Comparable 接口
    fun <T> printInfo(item: T) where T : Number, T : Comparable<T> {
        println("Value: $item")
    }
    
  3. 非空约束(Not-null Constraint):

    通过使用 T : Any 来约束泛型类型参数为非空类型。这样,你可以确保泛型参数不接受可空类型。

    // 示例:泛型类型参数 T 不能为可空类型
    fun <T : Any> processItem(item: T) {
        // 处理非空类型的 item
    }
    
  4. 泛型函数中的类型约束:

    在泛型函数中,你也可以对泛型参数进行约束,就像在泛型类中一样。使用 : 和类型名称来指定约束。

    // 示例:泛型函数中的类型约束
    fun <T> processList(list: List<T>) where T : CharSequence, T : Comparable<T> {
        // 在这里,T 必须是 CharSequence 和 Comparable<T> 的子类型
        // 可以使用 CharSequence 和 Comparable<T> 接口中定义的功能
    }
    

这些约束可以帮助你在泛型函数或泛型类中获得更具体的类型信息,并确保类型的安全性。在使用约束时,应该选择最合适的约束以满足你的需求,并根据实际情况来决定是否需要单个约束还是多重约束。


1.7 泛型在Android中的使用

在 Android 开发中,Kotlin 的泛型可以在许多地方使用,例如列表适配器、网络请求、异步任务等。这里我将以列表适配器为例来演示 Kotlin 泛型在 Android 中的使用。

假设你有一个简单的数据类 Person 表示人员信息:

data class Person(val name: String, val age: Int)

现在,我们要在 Android 中展示一个 RecyclerView,显示一组人员信息。我们将使用泛型来创建一个通用的列表适配器,可以在不同的地方重用。

首先,创建一个泛型适配器类 GenericAdapter

import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.TextView
import androidx.recyclerview.widget.RecyclerView

class GenericAdapter<T>(
    private val items: List<T>,
    private val itemLayoutResId: Int,
    private val bindHolder: (item: T, view: View) -> Unit
) : RecyclerView.Adapter<GenericAdapter<T>.ViewHolder>() {

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder {
        val view = LayoutInflater.from(parent.context).inflate(itemLayoutResId, parent, false)
        return ViewHolder(view)
    }

    override fun onBindViewHolder(holder: ViewHolder, position: Int) {
        val item = items[position]
        bindHolder(item, holder.itemView)
    }

    override fun getItemCount(): Int {
        return items.size
    }

    inner class ViewHolder(itemView: View) : RecyclerView.ViewHolder(itemView)
}

在这个适配器类中,我们使用泛型 T 来表示适配器中的数据项类型。items 参数是一个泛型类型的列表,用于存储要展示的数据。itemLayoutResId 参数表示列表项的布局资源 ID,bindHolder 参数是一个函数,用于将数据绑定到列表项的视图上。

现在,我们可以在 MainActivity 中使用这个通用的列表适配器来显示人员信息:

import android.os.Bundle
import android.view.View
import android.widget.TextView
import androidx.appcompat.app.AppCompatActivity
import androidx.recyclerview.widget.LinearLayoutManager
import kotlinx.android.synthetic.main.activity_main.*

class MainActivity : AppCompatActivity() {

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        val peopleList = listOf(
            Person("John Doe", 30),
            Person("Jane Smith", 25),
            Person("Michael Johnson", 40)
        )

        recyclerView.layoutManager = LinearLayoutManager(this)
        recyclerView.adapter = GenericAdapter(peopleList, R.layout.item_person) { person, view ->
            val nameTextView = view.findViewById<TextView>(R.id.nameTextView)
            val ageTextView = view.findViewById<TextView>(R.id.ageTextView)

            nameTextView.text = person.name
            ageTextView.text = person.age.toString()
        }
    }
}

在上面的例子中,我们使用 GenericAdapter 来展示人员信息,将每个人员的姓名和年龄显示在列表项中。由于我们使用了泛型,这个通用适配器可以用于显示其他类型的数据,只需更改数据源和布局即可。
这样,我们通过泛型的使用,实现了一个通用的列表适配器,可以在 Android 中更方便地重用。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

opencv-28 自适应阈值处理-cv2.adaptiveThreshold()

什么是自适应阈值处理? 对于色彩均衡的图像&#xff0c;直接使用一个阈值就能完成对图像的阈值化处理。但是&#xff0c;有时图像的色彩是不均衡的&#xff0c;此时如果只使用一个阈值&#xff0c;就无法得到清晰有效的阈值分割结果图像。 有一种改进的阈值处理技术&#xff…

Mybatis-Plus学习笔记,包含mybatis-plus基本使用,各种插件使用等等

&#x1f600;&#x1f600;&#x1f600;创作不易&#xff0c;各位看官点赞收藏. 文章目录 Mybatis-Plus笔记1、简介2、Mybatis-Plus Demo 程序3、Mybatis-Plus 常见注解4、Mybatis-Plus 条件构造器 Wrapper5、Mybatis-Plus 插件5.1、乐观锁插件5.2、分页插件5.3、逻辑删除插件…

OJ练习第145题——并行课程 III

力扣链接&#xff1a;2050. 并行课程 III 题目描述 给你一个整数 n &#xff0c;表示有 n 节课&#xff0c;课程编号从 1 到 n 。同时给你一个二维整数数组 relations &#xff0c;其中 relations[j] [prevCoursej, nextCoursej] &#xff0c;表示课程 prevCoursej 必须在课…

Python(四十七)列表对象的创建

❤️ 专栏简介&#xff1a;本专栏记录了我个人从零开始学习Python编程的过程。在这个专栏中&#xff0c;我将分享我在学习Python的过程中的学习笔记、学习路线以及各个知识点。 ☀️ 专栏适用人群 &#xff1a;本专栏适用于希望学习Python编程的初学者和有一定编程基础的人。无…

脚手架(vue-cli)的安装详细教程

首先要下载node.js 下载 | Node.js 中文网 (nodejs.cn)https://nodejs.cn/download/ 大家根据自己的系统来选择哪个&#xff0c;我是Windows系统&#xff0c;所以选择红色箭头所指的安装包去安装&#xff01;&#xff01;&#xff01; 接下来双击安装&#xff01;&#xff01;…

面试之CurrentHashMap的底层原理

首先回答HashMap的底层原理? HashMap是数组链表组成。数字组是HashMap的主体&#xff0c;链表则是主要为了解决哈希冲突而存在的。要将key 存储到&#xff08;put&#xff09;HashMap中&#xff0c;key类型实现必须计算hashcode方法&#xff0c;默认这个方法是对象的地址。接…

【深度学习笔记】Softmax 回归

本专栏是网易云课堂人工智能课程《神经网络与深度学习》的学习笔记&#xff0c;视频由网易云课堂与 deeplearning.ai 联合出品&#xff0c;主讲人是吴恩达 Andrew Ng 教授。感兴趣的网友可以观看网易云课堂的视频进行深入学习&#xff0c;视频的链接如下&#xff1a; 神经网络和…

C++数据结构笔记(11)二叉树的#号创建法及计算叶子节点数

首先分享一段计算叶子节点数目的代码&#xff0c;如下图&#xff1a; 不难发现&#xff0c;上面的二叉树叶子节点数目为4。我们可以采用递归的方式&#xff0c;每当一个结点既没有左结点又没有右节点时&#xff0c;即可算为一个叶子结点。 int num0; //全局变量&#xff0c;代…

会议OA系统会议管理模块开发思路(layui搭建)

目录 一.为什么要进行开发 1.开发目的 2.项目流程 A.发起会议请求过程 1.首先实现我们的多选下拉框功能&#xff01; 2.时间组件功能&#xff0c;并且提交我们新增加的会议内容 3.在进行发起会议编码时遇到的问题&#xff0c;BUG 3.1.有点时候js访问不到路径 3.2在增加…

Mac/win开发快捷键、vs插件、库源码、开发中的专业名词

目录 触控板手势&#xff08;2/3指&#xff09; 鼠标右键 快捷键 鼠标选择后shift⬅️→改变选择 mac command⬅️&#xff1a;删除←边的全部内容 commadtab显示下栏 commandshiftz向后撤回 commandc/v复制粘贴 command ⬅️→回到行首/末 commandshift3/4截图 飞…

C语言每日一题:4.消失的数字+数字在升序数组中出现的次数+整数转换

消失的数字&#xff1a; 思路1&#xff1a;排序遍历 1.使用qsort排序数组判断当前数值1是否是数组下一个元素的数值。 2.如果是一直循环注意数组越界&#xff0c;如果不是那么当前的数组的数值1就是消失的数。 3.存在0——n的数字是第n个数没有了。循环过程中从头到尾也找不到这…

适用于虚拟环境的免费企业备份软件

多年来&#xff0c;许多行业严重依赖物理服务器提供计算资源——你可以想象到巨大的服务器机房和笨重的服务器的场景。 然而&#xff0c;随着业务快速增长&#xff0c;许多组织发现物理服务器已经无法有效利用计算资源。因此&#xff0c;为了节省成本&#xff0c;引入了虚拟服…

SQL-每日一题【627. 变更性别】

题目 Salary 表&#xff1a; 请你编写一个 SQL 查询来交换所有的 f 和 m &#xff08;即&#xff0c;将所有 f 变为 m &#xff0c;反之亦然&#xff09;&#xff0c;仅使用 单个 update 语句 &#xff0c;且不产生中间临时表。 注意&#xff0c;你必须仅使用一条 update 语句…

c++数据锁链

题目描述&#xff1a; 创建一个结构体为Node&#xff0c;具有value , next 两个属性&#xff1b; value为整型&#xff0c;用来储存结构体数值&#xff1b; next为Node类型指针&#xff0c;用来指向下一组数据地址&#xff1b; 第1组数据value 5&#xff1b; 第2组数据value …

S475支持 ModbusRTU 转 MQTT协议采集网关

6路模拟量输入和2路RS485串口是一种功能强大的通信接口&#xff0c;可以接入多种设备和系统&#xff0c;支持4-20mA输出的传感器以及开关量输入输出。本文将详细介绍6路模拟量输入和2路RS485串口的应用场景和功能&#xff0c;重点介绍其在SCADA、HMI、远程数据监控以及采集控制…

中间件安全-CVE漏洞复现-Docker+Websphere+Jetty

中间件-Docker Docker容器是使用沙盒机制&#xff0c;是单独的系统&#xff0c;理论上是很安全的&#xff0c;通过利用某种手段&#xff0c;再结合执行POC或EXP&#xff0c;就可以返回一个宿主机的高权限Shell&#xff0c;并拿到宿主机的root权限&#xff0c;可以直接操作宿主机…

大数据课程D4——hadoop的MapReduce

文章作者邮箱&#xff1a;yugongshiyesina.cn 地址&#xff1a;广东惠州 ▲ 本章节目的 ⚪ 了解MapReduce的作用和特点&#xff1b; ⚪ 掌握MapReduce的组件&#xff1b; ⚪ 掌握MapReduce的Shuffle&#xff1b; ⚪ 掌握MapReduce的小文件问题&#xff1b; ⚪…

在CentOS 7上挂载硬盘到系统的步骤及操作

目录 1&#xff1a;查询未挂载硬盘2&#xff1a;创建挂载目录3&#xff1a;检查磁盘是否被分区4&#xff1a;格式化硬盘5&#xff1a;挂载目录6&#xff1a;检查挂载状态7&#xff1a;设置开机自动挂载总结&#xff1a; 本文介绍了在CentOS 7上挂载硬盘到系统的详细步骤。通过确…

【机器学习】基础知识点的汇总与总结!更新中

文章目录 一、监督学习1.1、单模型1.1.1、线性回归1.1.2、逻辑回归&#xff08;Logistic Regression&#xff09;1.1.3、K近邻算法&#xff08;KNN&#xff09;1.1.4、决策树1.1.5、支持向量机&#xff08;SVM&#xff09;1.1.6、朴素贝叶斯 1.2、集成学习1.2.1、Boosting1&…

QTDAY4

思维导图 tcp服务器 #ifndef WIDGET_H #define WIDGET_H#include <QWidget> #include <QTcpServer> //服务器类 #include <QTcpSocket> //客户端类 #include <QMessageBox> //对话框类 #include <QList> //链表容器…