React (三) 创建安装脚手架,类组件与函数式组件;生命周期;父子通信props;插槽;非父子通信Context

news2025/1/10 16:46:32

文章目录

  • 一、脚手架的创建与安装
    • 1. 认识脚手架
    • 2. 安装脚手架
    • 3. 创建react项目
    • 4. 项目结构
  • 二、从0编写
  • 三、组件化开发
    • 1. 什么是组件化开发
    • 2. 类组件
    • 3. render函数
    • 4. 函数式组件
  • 四、生命周期
    • 1. 挂载Mount
    • 2. 更新Update
    • 3. 卸载Unmount
    • 4. 不常用的生命周期
  • 五、父子组件通信
    • 1. 父传子
    • 2. 父传子数据的props类型限制
    • 3. 子传父
    • 4、{...object} 解构 传递对象数据---拓展
  • 六、插槽
    • 1. 组件子元素实现插槽效果
    • 2. props实现插槽效果
    • 3. 作用域插槽
  • 七、非父子组件通信Context
    • 1、Context上下文的使用(类组件)
    • 3、Context的使用(函数式组件)
      • (1) 函数组件读取Context数据---Context.consumer
      • (2) 组件读取多个Context的数据
      • (3) defaultValue
      • (4) ContextAPI总结

一、脚手架的创建与安装

1. 认识脚手架

  每个项目的基本工程化结构是相似的;既然相似,就没有必要每次都从零开始搭建,完全可以使用一些工具,帮助我们生成基本的工程化模板;
   脚手架(scaffold)就是一种工具,帮我们可以快速生成一个通用的项目目录结构,并将所需的工程环境配置好。让项目从搭建到开发,再到部署,整个流程变得快速和便捷;

2. 安装脚手架

React的脚手架:create-react-app, 简称cra
(1)提前安装node环境.(这个可参考之前安装Vue的记录配置node环境)
(2)执行命令安装脚手架:npm install create-react-app -g;
  运行create-react-app --version查看安装的版本,显示版本就说明脚手架安装成功。

在这里插入图片描述
(执行命令在powershell里也可以,在git bash里也可以)

3. 创建react项目

在对应的文件夹下执行命令create-react-app 项目名,创建项目。
注意:项目名称不能包含大写字母
在这里插入图片描述
运行项目:npm run start
在这里插入图片描述

4. 项目结构

|--public
|    |-- favucin.ico     // 标签页的icon图标
|    |-- index.html      // 入口文件
|    |-- logo192.png     // 在manifest.json文件里被调用
|    |-- logo512.png     // 在manifest.json文件里被调用
|    |-- manifest.json  // 和web app配置相关
|    |-- robots.txt     //指定本网站哪些文件可以或者无法被爬取
|--src
|    |-- App.css     // App组件相关的样式
|    |-- App.js      // App组件的代码文件
|    |-- App.test.js // App组件的测试代码文件
|    |-- index.css   // 全局的样式文件
|    |-- index.js    // 整个应用程序的入口文件
|    |-- logo.svg    // 启动项目时的react图标
|    |-- reportWebVitals.js  //
|    |-- setupTest.js    //测试初始化文件
|-- package.json        // 对整个应用程序的描述:应用名称、版本号、一些依赖包

logo192.png, logo512.png,manifest.json文件都与PWA相关,PWA(国内应用较少)了解即可。
在这里插入图片描述

二、从0编写

将src下的文件都删掉,运行项目,提示缺少index.js文件,新建src/index.js

import React from "react"
import ReactDOM from "react-dom/client" // 旧版是从react里导入ReactDOM

class App extends React.Component {
  constructor() {
    super()
    this.state = {
      msg: 'HelloWorld'
    }
  }
  render () {
    const { msg } = this.state
    return (
      <h2>{msg}</h2>
    )
  }
}
// 这里的#root是index.html文件里的
const root = ReactDOM.createRoot(document.querySelector('#root'))
root.render(<App />)

与之前写的一样,都是创建类组件,然后渲染到桌面上。
可将App组件拆分到App.jsx文件里:
在这里插入图片描述

三、组件化开发

1. 什么是组件化开发

  组件化就是分而治之;如果一个页面里的所有功能和逻辑处理都放在一起,就会很难维护。如果将一个页面拆分为一个个小的功能块,有利于之后页面的管理和扩展。

用组件化的思想来构建应用:

  • 一个完整的页面可以分为很多组件;
  • 每个组件都用于实现页面的一个功能块
  • 每个组件都可以继续细分,组件本身又可以进行复用

最终,任何的应用都会被抽象成一棵组件树
在这里插入图片描述

2. 类组件

定义类组件的要求:

  • 组件的名称是大写字符开头(无论类组件还是函数组件)
  • 类组件需要继承React.Component(后边优化时,可继承Pure)
  • 类组件必须实现render函数

使用class定义一个类组件:

import React from "react";  // 导入的React是个对象,里面有React.Component属性
class App extends React.Component {
  // constructor是可选的,通常在构造函数里初始化一些数据
  constructor() {
    super()
    // 维护的是组件内部的数据
    this.state = {
      msg: 'Hello World'
    }
  }
  // 组件内必须要实现的方法
  render () {
    return <h2>{this.state.msg}</h2>
  }
}

也可以这样继承: 反正继承的都是Component

import { Component } from "react";
class App extends Component {
  render () {
    return <h2>Hello</h2>
  }
}

3. render函数

(1) render函数什么时候被调用
  页面初次加载时,函数render会被调用渲染页面。
  当propsstate发生变化时,render会再次被调用。(props后面会学,state里的数据是通过调用this.setState进行修改)

(2) 返回值有哪几类

  • React元素
    通过JSX创建的就是React元素。JSX本质上就是调用React.createElement(),创建一个React的元素。
    在这里插入图片描述
  • 数组或fragments:返回多个元素
    fragments后边再学
     // 2.组件或者fragments(后续学习)
        return ["abc", "cba", "nba"]
        return [
           <h1>h1元素</h1>,
           <h2>h2元素</h2>,
           <div>哈哈哈</div>
        ]
    

在这里插入图片描述

  • Portals (还没学):可以渲染子节点到不同的DOM子树中。
  • 字符串或数值类型:在DOM中渲染为文本节点
  • 布尔类型或null:什么都不渲染
    return "Hello World" // 界面渲染 HelloWorld
    return true          // 什么都不渲染
    

4. 函数式组件

(1) 函数组件是使用function来进行定义的函数,这个函数返回的内容和类组件中render函数返回的一致。

(2) 函数组件的特点:

  • 没有生命周期,也会被更新并挂载,但是没有生命周期函数;
  • this关键字不能指向组件实例(因为没有组件实例)
  • 没有内部状态(state),即使自己定义了state数据,每次调用state返回的数据都是初始数据。也就是无法进行状态维护
// 函数式组件
function App () {
  const state = { name: 'tom' } // 每次调用App组件,state里的数据值都是tom
  // 返回值:和类组件中的render函数返回的值类型一致。
  return <h2>Hello World</h2>
}

四、生命周期

生命周期就是从创建到销毁的这个过程;
在这里插入图片描述
在生命周期这个过程中,可以划分为很多阶段:

  • 挂载阶段(Mount): 组件第一次在Dom树中被渲染的过程
  • 更新阶段(Update):组件状态发生变化,重新更新渲染的过程
  • 卸载阶段(Unmount):组件从Dom树中被移除的过程

React为了告诉我们当前组件处于哪些阶段,会在对应的阶段调用某些函数,这些函数就是生命周期函数

  • Constructor
    如果不初始化 state 或不进行方法绑定(为事件绑定实例this),则不需要为 React 组件实现构造函数。

  • componentDidMount
    依赖于DOM的操作可以在这里进行;
    在此处发送网络请求就最好的地方;(官方建议)
    可以在此处添加一些订阅

  • componentDidUpdate
    componentDidUpdate() 会在更新后会被立即调用,首次渲染不会执行此方法。
    当组件更新后,可以在此处对 DOM 进行操作;

  • componentWillUnmount
    该生命周期函数会在组件卸载及销毁之前直接调用。
    在此方法中执行必要的清理操作 ( 比如 清除 timer,取消网络请求或清除在 componentDidMount() 中创建的订阅等)

谈生命周期函数,主要指的是类组件,因为函数式组件没有生命周期函数。

1. 挂载Mount

挂载步骤:
(1) 创建组件实例(constructor)。创建组件实例会先执行对应类组件里的构造函数constructor
每执行一次<HelloWordl/>,相当于new一个class HelloWorld extends Component{}的实例 (就像java一样)
在这里插入图片描述
(2) 执行render方法,渲染界面
(3) 挂载完毕,执行生命周期函数componentDidMount

class HelloWorld extends React.Component {
  // 1.执行构造方法
  constructor() {
    super()
    console.log('HW, constructor');
    this.state = { msg: 'HelloWorld' }
  }
  // 2. 执行render函数
  render () {
    console.log('HW, render');
    let { msg } = this.state
    return (
      <h2>{msg}</h2>
    )
  }
  // 3. 挂载完成
  componentDidMount () {
    console.log('HW, componentDidMount');
  }
}

挂载阶段,依次打印:HW, constructor,HW, render,HW, componentDidMount

2. 更新Update

从图里可以看出,触发更新有三种方式,现在只说setState()
(1) setState()修改数据
(2) 重新调用render函数
(3) 更新完成,调用生命周期函数componentDidUpdate

  // 组件的DOM更新完成
  componentDidUpdate () {
    console.log('HW, componentDidUpdate');
  }

在这里插入图片描述

3. 卸载Unmount

HelloWorld组件

  // 5. 组件将从DOM树中被移除:卸载组件
  componentWillUnmount () {
    console.log('HW, componentWillUnmount');
  }

App组件:点击按钮时,隐藏组件

 render () {
   let { isShow } = this.state
   return (
     <div>
       <button onClick={e => this.changeHWShow()}>切换</button>
       {isShow && <HelloWorld />}
     </div>
   )
 }
 changeHWShow () {
  this.setState({
    isShow: !this.state.isShow
  })
}

在这里插入图片描述

4. 不常用的生命周期

shouldComponentUpdate下一篇博客会说。
在这里插入图片描述
具体查看官方文档:官方文档

五、父子组件通信

1. 父传子

  • 父组件通过属性=值的形式来传递给子组件数据
  • 子组件通过props参数获取父组件传递过来的数据(不能换名,只能叫props)

需求:父组件Content给子组件Banner传递数据:

父组件Content:

 this.state = {
   banners: ['新歌曲', '新歌单', '新MV'],
   title: '轮播图'
 }
render () {
  let { banners, title } = this.state
  return (
    <div>
       <Banner banners={banners} title={title} />
    </div>
  )
}

<Banner banners={banners} />相当于在new实例时,传递了参数。所以Banner的构造函数需要接收这个参数。

子组件Banner:

export class Banner extends Component {
  constructor(props) {
  // props接收之后传给super()
    super(props)
    console.log('Banner接收:', props);
  }
  
  render () {
  // 从props属性里可以读取父组件传递过来的值
    let { banners, title } = this.props
    return (
      <div  className='banner'>
        <h3>{title}</h3>
        {banners.map(item => {
          return <li key={item}>{item}</li>
        })}
      </div>
    )
  }
}

  其实,如果没有constructor构造函数(2-6行代码),也会默认通过props帮忙接收父组件的数据。render函数里仍然可以通过this.props接收数据。
在这里插入图片描述

2. 父传子数据的props类型限制

(1) 设置数据类型限制

// 1. 引入 prop-types
import PropTypes from 'prop-types'
class Banner extends Component {
 ...
}
// 2. 限制类型 
// 要求title是字符串,且必须传值(如果没传,且Banner组件也没有设置title默认值,即使Banner不使用title,也会报错);
// 要求banners是数组类型的数据
Banner.propTypes = {
  title: PropTypes.string.isRequired,
  banners: PropTypes.array
}

PropTypes的其他限制:NPM:prop-types

(2) 设置数据默认值

  • 在组件内用static关键字;
  • 在组件外用defaultProps
class Banner extends Component {
  // 方式一:设置默认值
  static defaultProps = {
    title: '默认标题',
    banners: []
  }
}
// 方式二
Banner.defaultProps = {
  banners: [],
  title: "默认标题"
}

测试

为了方便观察,给banner添加了样式
Banner组件:
在这里插入图片描述

  <!--规范传值-->
  <Banner banners={banners} title={title} />
   <!--啥也不传-->
  <Banner />

在这里插入图片描述

3. 子传父

需求:
在这里插入图片描述
其实还是父组件给子组件传递一个函数,子组件调用这个函数,并将数据通过参数的形式传递给父组件。
在这里插入图片描述

4、{…object} 解构 传递对象数据—拓展

父组件的state里有一个对象数据要传递给子组件。

this.state = {
     info: { stuName: 'tom', age: 18 }
}
// props传递数据
 <Home stuName={info.stuName} age={info.age} />
 {/* 等价于 */}
 <Home {...info} /> 

六、插槽

比如导航区的组件NavBar;每个页面的导航组件结构大体一致(分为左中右三个部分),但内容不一样。
在Vue里可以通过插槽实现不同样式的导航组件。
在这里插入图片描述
React里并没有插槽这个概念。对于这种需要插槽的情况,React有两种方案可以实现:

  • 组件的children子元素
  • props属性传递React元素

1. 组件子元素实现插槽效果

App.jsx:

<!--将button,h2,i等标签传递给NavBar,NavBar标签包裹的就叫子元素-->
  <NavBar>
    <button>按钮</button>
    <h2>HelloWorld</h2>
    <i>斜体文字</i>
  </NavBar>

子组件可以通过this.props.children接收父组件传递过来的react元素。当有多个元素时,children是一个数组,包含这些元素;当只有一个元素时,children的值就是这个元素。

nav-bar.jsx:

// 子组件通过props接收react元素
 render () {
   let { children } = this.props
   console.log('children', children);
   return (
     <div className='nav-bar'>
       <div className="left">{children[0]}</div>
       <div className="center">{children[1]}</div>
       <div className="right">{children[2]}</div>
     </div>
   )
 }

在这里插入图片描述
在这里插入图片描述
如果只传button
在这里插入图片描述

缺点: <NAvBar>标签里子元素(react元素)的书写顺序决定了元素在children里的索引值。 子组件通过children的索引值获取传入的元素,容易出错。

2. props实现插槽效果

和之前props传值一样,只不过这次props传递的是react元素。
在这里插入图片描述
通过具体的属性名,可以在传入元素和获取元素时更加精准。

3. 作用域插槽

适用场景:结构由父组件决定,但是结构里用的值在子组件中。
子组件有标题titles数据,父组件传递结构,决定这些标题如何展示。比如说展示按钮
在这里插入图片描述
传递react元素:
在这里插入图片描述
问题是这样渲染出来的按钮内容都是哈哈哈, 所以需要子组件将标题内容传给父组件。

怎么传?还是通过回调函数传参的方式,itemType改成一个函数。

App.jsx:

 <TabControl
   itemType={(item) => <button>{item}</button>}
 />

子组件:

render () {
  let { titles } = this.state
  let { itemType } = this.props // 接收函数
  return (
    <div className='tab-control'>
      {titles.map((item, index) => {
        return (
          <div className='item' key={index} >
            {/* 调用函数并传参 */}
            {itemType(item)}
          </div>
        )
       })
      }
    </div >
  )
}

就很绝,甚至可以根据不同的title内容返回不同的页面结构

  getTabItem(item) {
    if (item === "流行") {
      return <span>{item}</span>    // 标签 
    } else if (item === "新款") {
      return <button>{item}</button> // 按钮
    } else {
      return <i>{item}</i>  // 斜体文字
    }
  }
...
itemType={item => this.getTabItem(item)}

七、非父子组件通信Context

在组件树中,顶层Provider(父组件)提供数据,下边的作为消费者Consumer(后代组件)接收数据,或者通过指定的方式接收数据。

1、Context上下文的使用(类组件)

通过props实现父给孙组件传值,会打扰到中间的组件。而且比较麻烦。
在这里插入图片描述
组件关系 App> Home> HomeInfo

Context使用步骤:

(1) 使用React.createContext创建一个context

src/context/theme-context.js:

import React from "react";
// 1. 创建一个Context
const ThemeContext = React.createContext()
export default ThemeContext

(2) 引入上下文,并为后代提供数据

App组件中:

import ThemeContext from './contex

{/* 2.通过ThemeContext中的Provider中value属性为后代提供数据 */}
<ThemeContext.Provider value={{ color: 'red', price: '50' }}>
   <Home />
 </ThemeContext.Provider>

给子组件Home包裹标签<ThemeContext>, (这里演示非父子组件传值,所以包裹Home。也可以在Home组件里给孙组件HomeInfo包裹标签,包裹到的组件及后代组件都可以用到数据)

(3) 在组件中指定要读取的Context类型,并读取数据

HomeInfo组件:

class HomeInfo extends Component {
  render () {
    // 4. 第四步:读取数据
    console.log('上下文:', this.context); 
		...
  }
}
// 3. 第三步:context可能有很多,设置组件的要读取哪一类的context
HomeInfo.contextType = ThemeContext

在这里插入图片描述

3、Context的使用(函数式组件)

(1) 函数组件读取Context数据—Context.consumer

定义函数组件HomeBanner,并在Home组件中使用
Home组件:

  render () {
    return (
      <div>
        <h2>Home组件</h2>
        <HomeBanner />
      </div>
    )
  }

注意之前在App组件中,已经用Context的组件将Home组件包裹了。

HomeBanner组件:
使用context名.Consumer读取数据。
Context.Consumer标签需要 函数作为子元素(function as child);
这个函数接收当前的 context 值,返回一个 React 节点

import ThemeContext from "./context/theme-context"
function HomeBanner () {
  return (
    <div>
      {/* 函数式组件中使用Context共享的数据 */}
      <ThemeContext.Consumer>
        {
           {/* value就是context的值*/}
          value => {
            return <h3>颜色:{value.color}</h3>
          }
        }
      </ThemeContext.Consumer>
    </div>
  )
}

(2) 组件读取多个Context的数据

在类组件中,组件名.contextType = xxx只能指定一种context类型,如果该组件想读取多个context的数据,可以借助Context.Consumer
在这里插入图片描述

(3) defaultValue

什么时候使用默认值defaultValue呢?当组件未被Context组件包裹,但又想使用该Context的数据时。比如Profile组件
在这里插入图片描述
步骤一: 在context文件里指定默认值
在这里插入图片描述
步骤二: 组件中使用
在这里插入图片描述
控制台打印结果:{color: 'blue', price: 10}

(4) ContextAPI总结

  • React.createContext
    创建一个需要共享的Context对象:ThemeContext = React.createContext({ color: "blue", price: 10 })
    defaultValue是组件在顶层查找过程中没有找到对应的Provider,那么就使用默认值

  • Context.Provider
    每个 Context 对象都会返回一个 Provider React 组件;Provider 接收一个 value 属性,传递给消费组件(Consumer);<ThemeContext.Provider value={{ color: 'red', price: '50' }}>
    当 Provider 的 value 值发生变化时,它内部的所有消费组件都会重新渲染;

  • Class.contextType
    挂载在 class 上的 contextType 属性会被重赋值为一个由 React.createContext() 创建的 Context 对象:
    Profile.contextType = ThemeContext

  • Context.Consumer
    这里,React 组件也可以订阅到 context 变更。这能让你在 函数式组件 中完成订阅 context。
    这里需要 函数作为子元素(function as child);
    这个函数接收当前的 context 值,返回一个 React 节点;
    在这里插入图片描述

什么时候使用Context.Consumer呢?

1.当使用value的组件是一个函数式组件时;
2.当组件中需要使用多个Context时;

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

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

相关文章

数据结构之队列(python)

华子目录 1.队列存储结构1.1队列基本介绍1.2队列的实现方式 2.顺序队列2.1顺序队列的介绍2.2顺序队列的简单实现2.3代码实现 3.链式队列和基本操作3.1链式队列数据入队3.2链式队列数据出队3.3队列的链式表示和实现 1.队列存储结构 1.1队列基本介绍 队列的两端都"开口&qu…

springcloud之服务集群注册与发现 Eureka

前言 1&#xff1a;对于能提供完整领域服务接口功能的RPC而言&#xff0c;例如&#xff1b;gRPC、Thrift、Dubbo等&#xff0c;服务的注册与发现都是核心功能中非常重要的一环&#xff0c;使得微服务得到统一管理。 2&#xff1a;在分布式领域中有个著名的CAP理论&#xff1b;…

Vue3 + Element plus 实现切换el-radio前二次确认

Vue3 Element plus 实现切换el-radio前二次确认 场景&#xff1a;点击切换el-radio之前判断当前内容是否有改变&#xff0c;如有改变弹窗提示切换el-radio将销毁操作&#xff0c;弹窗二次确认是否切换 问题&#xff1a; el-radio 没有提供类似于beforeUpdate这样的钩子去处理这…

探索极简计算的新边界:从Uxn虚拟机看未来编程生态

越来越多的开发者追求复杂度和功能性的极致,然而,有一个小众的编程社区选择了截然不同的道路——极简主义。Uxn虚拟机便是这一思潮的代表之一。它通过简洁的指令集和有限的硬件资源模拟,试图打造一种可以在多种设备上运行的便携性编程环境。 与主流的重型操作系统和复杂…

HTB:Legacy[WriteUP]

目录 连接至HTB服务器并启动靶机 1.How many TCP ports are open on Legacy? 2.What is the 2008 CVE ID for a vulnerability in SMB that allows for remote code execution? 3.What is the name of the Metasploit module that exploits CVE-2008-4250? 4.When expl…

VS+QT 自定义插件变成动态库加载及使用

一、前言 有个界面需要重复使用某个自定义的控件&#xff0c;希望自定义控件能够像动态库文件那样&#xff0c;添加引用lib就能使用&#xff0c;经过多次太坑后&#xff0c;总结如下 二、实现方式 ① 新建项目&#xff0c;选择"Qt Designer Custom Widget" 创建自定…

Springboot从入门到起飞-【day01】

个人主页→VON 收录专栏→Springboot从入门到起飞 一、前言 经过了近两个月的沉淀开始了新专栏的学习&#xff0c;经过深思熟虑还是决定重新学习java&#xff0c;因为基础部分东西太多太乱就不进行逐一的更新了&#xff0c;等到学完了一同进行更新。 二、Springboot简要概述 …

kafka消息队列核心内容及常见问题

目录 1. 使用消息队列的目的&#xff08;优点与缺点&#xff09; 2. 常见各消息队列对比 3. kafka介绍 3.1 kafka简介 3.2 kafka特点 3.3 kafka系统架构 3.4 设置数据可靠性 3.4.1 Topic 分区副本 3.4.2 消息确认机制 4. 常见问题&#xff08;面试题&#xff09; 4.…

Springboot 接入 WebSocket 实战

Springboot 接入 WebSocket 实战 前言&#xff1a; WebSocket协议是基于TCP的一种新的网络协议。它实现了浏览器与服务器全双工(full-duplex)通信——允许服务器主动发送信息给客户端。 简单理解&#xff1a; 1&#xff0c;常见开发过程中我们知道 Http协议&#xff0c;客户端…

LED显示器闪烁故障原因及解决方法

随着电子显示屏在各行各业的广泛应用&#xff0c;LED显示器因其高亮度、节能和灵活的宣传功能&#xff0c;成为了宣传推广的重要工具。然而&#xff0c;LED显示器在使用过程中有时会出现闪烁的现象&#xff0c;这不仅影响了显示效果&#xff0c;还可能影响用户体验。针对这一问…

【layui】多文件上传组件实现

插件预览效果&#xff1a; 需要引入layui的脚本文件layui.js和样式文件layui.css html代码&#xff1a; <div class"layui-input-block"><div class"layui-upload-list"><table class"layui-table"><colgroup><col…

18936 手串

### 思路 1. **输入处理**&#xff1a;读取输入的n, m, c&#xff0c;以及每个串珠的颜色信息。 2. **颜色位置记录**&#xff1a;使用一个字典来记录每种颜色出现的位置。 3. **检查颜色分布**&#xff1a;遍历每种颜色&#xff0c;检查其在任意连续m个串珠中是否出现超过一次…

【Flask】Flask数据库

【Flask】Flask数据库 1.概述2.使用Flask-SQLAlchemy管理数据库3.定义模型4.关系5.数据库操作创建表插入行修改行删除行查询行 1.概述 大多数的数据库引擎都有对应的 Python 包&#xff0c;包括开源包和商业包。Flask 并不限制你使用何种类型的数据库包&#xff0c;因此可以根…

Java体系中的泛型

1. 泛型 一般的类和方法&#xff0c;只能够使用基本类型&#xff0c;要么是自定义的类&#xff0c;如果要编写可以应用于多种数据类型的代码&#xff0c;这种刻板的限制对代码的约束就会很大&#xff0c;那么如何实现可应用于多种数据类型的代码&#xff0c;而不局限于单一一种…

第5篇:DDOS病毒----应急响应之Linux实战篇

现象描述 某服务器网络资源异常,感染该木马病毒的服务器会占用网络带宽&#xff0c;甚至影响网络业务正常应用。 系统分析 针对日志服务器病毒事件排查情况&#xff1a; 在开机启动项/etc/rc.d/rc.local发现可疑的sh.sh脚本&#xff0c;进一步跟踪sh.sh脚本,这是一个检测病毒…

C++从入门到起飞之——AVL树 全方位剖析!

&#x1f308;个人主页&#xff1a;秋风起&#xff0c;再归来~&#x1f525;系列专栏&#xff1a;C从入门到起飞 &#x1f516;克心守己&#xff0c;律己则安 目录 1. AVL的概念 2. AVL树的实现 2.1 AVL树的结构 2.2 AVL树的插⼊ >AVL树插⼊⼀个值的⼤概过程 &…

Rocky linux 修改ip地址, rocky服务器修改静态地址, rocky虚拟机修改ip

1. 更新yum yum update 2. 安装ifconfig yum install net-tools 3. 修改配置 vi /etc/NetworkManager/system-connections/ens33.nmconnection 将ipv4内容修改如下&#xff1a; # 自动改为手动 methodmanual # 网关为vm ware 查看网关地址 address你想改为的ip/24,网关 #dns不…

Qml 分组动画(二) 动画嵌套(自学笔记)

分组动画嵌套示例&#xff0c;直接看效果&#xff0c; 做一个踢足球的示例 下面两个Rectangle 制作渐变的天空和大地 下面这个Rectangle 用于放置足球图片&#xff0c; 由于足球图片直接从网上下载的 没有找到合适大小的图片 &#xff0c;所以用 一个矩形框作限制&#xff0c;…

闲谈Promise

预备知识 回调函数&#xff1a;当一个函数作为参数传入另一个函数中&#xff0c;并且它不会立刻执行&#xff0c;当满足一定条件之后&#xff0c;才会执行&#xff0c;这种函数称为回调函数。比如&#xff1a;定时器。异步任务&#xff1a;与之对应的概念是同步任务&#xff0…

【JVM】面试篇

1 什么是JVM&#xff1f; 1.1 定义 JVM 指的是Java虚拟机&#xff08; Java Virtual Machine &#xff09;。JVM 本质上是一个运行在计算机上的程序&#xff0c;他的职责是运行Java字节码文件&#xff0c;Java虚拟机上可以运行Java、Kotlin、Scala、Groovy等语言。 启动这个程…