ES6 中的类(class)

news2025/1/11 21:56:04

前言

ES6 引入的 class 关键字具有定义类的能力。类是 ECMAScript 中新的基础性语法糖。虽然 ES6 表面上看起来可以支持正式的面向对象编程,但实际上它背后使用的依旧是原型和构造函数的概念。(笔记内容参考《JavaScript 高级程序设计(第 4 版)》章节 8.4 类)。以前在笔记《js 中原型、原型链和继承概念(详细全面)》中简要提过部分内容,这里系统看一次。

1 类的定义

定义类有两种主要方式:类声明和类表达式。两种方式都需要使用 class 关键字加大括号定义。类表达式与函数表达式类似,在它们被求值前都不能被引用。但是类声明与函数声明不同,类声明不能提升,即在声明之前引用该类都会抛出引用错误 ReferenceError类似暂时性死区,因为类受块级作用域限制,个人认为这两者是存在关系的,书中没有说明)。

// 类声明
class Person { }

// 类表达式
const Tree = class {};

// 类声明不提升
// ReferenceError: Cannot access 'C' before initialization
console.log(C);
class C {}

// function 和 class 关键字类似于 var 和 let、const 的区别。
// 函数声明受函数作用域限制,类受块作用域限制。
{
    function f() {
        console.log('f');
    }
    class Cx {}
}
f(); // 'f'

// ReferenceError: Cx is not defined
console.log(Cx);

类包含可选的构造函数方法、静态类方法、实例方法、获取函数和设置函数等。与函数构造函数一样,多数编程风格都建议类名首字母大写,以区别于通过它创建的实例。默认情况下,类定义中的代码都在严格模式下执行。

class Person {
    // 实例私有属性或者方法(ES10)
    #age;

    // 类构造函数
    constructor(name, privateAge) {
        this.name = name;
        this.#age = privateAge;
    }

    // 实例方法(原型对象上)
    say() {
        console.log(`不想告诉你我的年龄是${this.#age}`);
    }

    // 静态类方法
    static getName() {
        console.log(this.name);
    }
}
const p = new Person('tkop', 18);
console.log(p.name, p.age);
p.say();
Person.getName();

类表达式表现出来的特征与函数非常相似,例如 name 属性是一样,如果表达式声明具有 name 则首选返回该名称字符串,否则是表达式左边的值。本质就是表达式最后的值且它们都无法在表达式作用域外部访问这个标识符

const Person = class PersonName {
    identify() {
        console.log(Person.name, PersonName.name);
    }
};
let p = new Person();
p.identify();

// ReferenceError: PersonName is not defined
// p = new PersonName();

const F = function Fn() {
    console.log(F.name, Fn.name);
};

F();
let f = new F();

// ReferenceError: Fn is not defined
// f = new Fn();
// Fn();

// ========================
const Personx = class {
    identify() {
        console.log(Personx.name);
    }
};
p = new Personx();
p.identify();

var Fx = function () {
    console.log(Fx.name);
};
Fx();

2 类构造函数

实际上可以将类定义时大括号内部作用域包含的内容视为是对类(可视为特殊函数)的原型对象的初始化。内部定义的方法即为原型对象上的方法,例如 constructor 关键字用于在类定义块内部创建类的构造函数(即为 prototypeconstructor 属性)。

2.1 类的实例化

方法名 constructor 会告诉解释器在使用 new 操作符创建类的新实例时,应该调用这个函数。构造函数的定义不是必须的,不定义则相当于将构造函数定义为空函数。使用 new 操作符调用类的构造函数会执行如下操作

  1. 在内存中创建一个新对象。
  2. 这个新对象内部的 [[Prototype]] 指针被赋值为构造函数的 protorype 属性(对象原型属性指向构造函数原型对象)
  3. 构造函数内部的 this 被赋值为这个新对象(this 指向新对象)
  4. 执行构造函数内部的代码(给新对象添加属性)。
  5. 如果构造函数返回非空对象,则返回该对象;否则返回刚创建的新对象。
// 构造函数为空函数
class C1 {
    say() {
        console.log('instanceof C1');
    }
}

// 执行构造函数并为新对象添加属性
class C2 {
    constructor(arg) {
        this.instanceProperty = arg;
    }
}

let instance1 = new C1();
let instance2 = new C2('arg_input');
instance1.say();
console.log(instance2.instanceProperty);

实例化时涉及的问题:

1、类实例化时传入的参数会作为构造函数的参数。如果不需要参数,则调用时类名后面的括号也是可选的。

2、构造函数执行完成后默认会返回 this 对象。构造函数返回的对象会被用作实例化的对象,如果外部没有什么引用返回的 this 对象,那么这个对象会被销毁

3、如果构造函数返回的不是默认的 this 对象,而是其他对象。则需要注意这个对象与类并没有关联(使用 instanceof 操作符检测为 false,并不是类的实例)。因为该对象的对象原型并没有被修改

class C3 {
    constructor(arg) {
        this.instanceProperty = arg || 'default';
    }
}
let instance3 = new C3('arg_input');
let instance4 = new C3;
console.log(instance3.instanceProperty, instance4.instanceProperty);

class C4 {
    constructor() {
        return { uname: '扬尘' };
    }
}
let instance5 = new C4();
console.log(instance5 instanceof C4);

注意:类构造函数与构造函数的主要区别是,类构造函数必须使用 new 操作符调用。否则会抛出错误 TypeError。类构造函数在实例化后会成为普通的实例方法(严格来说是原型对象上共享的方法,仍然需要使用 new 调用)。

class Person {}
let p1 = new Person();

// 实际上是 Person.prototype.constructor
let p2 = new p1.constructor();
console.log(p1.constructor === Person);

// TypeError: Class constructor Person cannot be invoked without 'new'
// Person();
// p1.constructor();

2.2 将类当成特殊函数

我们可以从类各方面的表现看出类就是一种特殊的函数,具有 prototype 属性,而这个原型对象也具有一个 constructor 属性指向类的本身。

class Person {}

console.log(typeof Person); // function
console.log(Person.prototype); // {constructor: ƒ}
console.log(Person === Person.prototype.constructor);

let p = new Person();
console.log(Person === p.constructor);
console.log(p instanceof Person);

至此,所有有关类定义的基本概念应该都非常明确了。但是书中却多余地给出了下面的这段描述。

p9S9OaD.jpg

之所以说多余是因为示例中获取的并不是类块中定义的 constructor 函数。前面我们提到过,在类内定义的所有内容可以视为是原型对象中的属性和方法。使用实例或者原型对象都可以访问,且 constructor 指向的是类自身。示例中的 Person.constructor 引用的当然不会是类中定义的该属性。它是类这个函数对象(将其视为一个对象)的对象原型(__proro__)上的一个属性,也就是实例化 Person 的构造函数。验证如下:

class Person {}

// 实例化的 Person 对象是函数类型
// 那用它的构造函数再次实例化一个对象也应该是函数类型;
const o = new Person.constructor();
console.log(typeof o); // 'function'

不应该将类块内的构造函数和类的 constructor 属性混为一谈,如果不理解其中的原理,就会徒增负担

描述方面也不够详细:类中定义的 constructor 方法不会被当成构造函数?在对它使用 instanceof 操作符会返回 false ?(概念描述有点含糊了)。

class Person {}

// 这才是类中的constructor 方法,上面只是类的 constructor 属性
const o = new Person.prototype.constructor();
const p = new o.constructor();

// 可以当成构造函数使用,因为指向类的本身
console.log(typeof p); // 'object'

// true
console.log(o instanceof p.constructor && p instanceof o.constructor);

3 实例、原型和类成员

类的语法可以非常方便地定义应该存在于实例上的成员、应该存在于原型上的成员,及应该存在于类本身的成员。

3.1 实例成员

实例成员是指每次使用 new 操作符调用构造函数时,内部为新创建的实例(this)添加的“自有”属性。每次创建的实例都对应唯一的成员对象,该对象上面的实例成员都不会在原型上共享。

class Person {
    constructor(id) {
        this.id = id;
        this.friends = ['扬尘'];
    }
}

let p0 = new Person(0);
let p1 = new Person(1);
console.log(p0.id, p1.id);
console.log(p0.friends === p1.friends);

// 当然可以继续添加新成员
p0.name = 'tkop';
console.log(p0.name);

3.2 原型方法与访问器

类定义语法将在类块中定义的方法作为原型方法。类方法等同于对象属性,因此可以使用字符串或者计算的值作为键。同时也支持获取和设置访问器属性,语法和行为同普通对象一样。

class Person {
    constructor() {
        this.say = () => console.log('instance_say');
    }
    say() {
        console.log('prototype_say');
    }
    [Symbol.for('symbolKey')]() {
        console.log('symbolKey');
    }
    ['conputed' + 'Key']() {
        console.log('conputedKey');
    }
    set name(newName) {
        this._name = 'get_' + newName;
    }
    get name() {
        return this._name;
    }
}

let p = new Person();
p.say(); // 'instance_say'
Person.prototype.say(); // 'prototype_say'
p.conputedKey(); // 'conputedKey'
p[Symbol.for('symbolKey')](); // 'symbolKey'
p.name = 'tkop';
console.log(p.name); // 'get_tkop'

可以将方法定义在类构造函数中或者类块中,但是不能在类块中给原型添加原始值或者对象作为成员数据

// Uncaught SyntaxError: Unexpected identifier 'name'
class Person {
    name: 'Jack';
    // 这个会变为实例成员(公有字段声明)
    // name = 'Jack';
}

3.3 静态类方法和实例私有属性

可以在类上定义静态方法。这些方法通常用于执行不特定于实例的操作,也不要求存在类的实例。静态成员在类定义中使用 static 关键字作为前缀。静态成员由类调用,其中的 this 引用类自身静态方法非常合适作为实例工厂。类似于 Promise.resolve() 和 Promise.reject()。

class Person {
    constructor(age) {
        this.age = age;
        this.say = () => console.log('instance_say');
    }
    say() {
        console.log('prototype_say', this === Person.prototype);
    }
    static say() {
        console.log('Person_say', this === Person);
    }
    static create() {
        // 创建返回一个随机年龄属性值的 Person 实例
        // return new this(Math.floor(Math.random() * 100));
        return new Person(Math.floor(Math.random() * 100));
    }
}

let p = new Person();
p.say(); // 'instance_say'
Person.prototype.say(); // 'prototype_say'  true
Person.say(); // 'Person_say' true

p = Person.create();
console.log(p); // Person { age: 88, say: [Function (anonymous)] }

JavaScript 以前没有私有成员的概念,只是利用闭包结合 Symbol 或者 WeakMap 等类型模拟实现了私有属性的概念。但是在 ECMAScript2019 中新增了在类块中使用 “#” 定义私有成员的语法(私有字段声明)。

class Person {
    #address;
    constructor(name, address) {
        this.name = name;
        this.#address = address;
    }
    say() {
        console.log(this.#address);
    }
}

let p = new Person('扬尘', '个人隐私-家庭地址');
p.say(); // '个人隐私-家庭地址'
console.log(Reflect.ownKeys(p)); // [ 'name' ]

3.4 非函数原型和类成员

可以注意到前面的类定义并不显示支持在原型或者类上添加成员数据。但是在类定义外部,可以手动添加(其实类块内支持在类上添加成员数据,示例如下)。

class Person {
    static anotherName = 'P';
}
const p = new Person();
console.log(Person.anotherName); // 'P'

// 手动添加
class Per {
    say() {
        console.log(`${Per.greeting} ${this.name}`);
    }
}
const p = new Per();

// 在类上定义数据成员。
Per.greeting = 'My name is';

// 在原型上定义数据成员。
Per.prototype.name = 'Lucky';
p.say(); // 'My name is Lucky'

3.5 迭代器与生成器方法

类定义语法支持在原型和类本身上定义生成器方法。也可以通过添加默认的迭代器,将类实例定义为可迭代对象。这部分在迭代器和生成器部分已学习,这里只罗列代码。

class Per {
    constructor(...rest) {
        this.arg = [...rest];
    }
    *createNicknameIterator() {
        yield 'Jack';
        yield 'Jake';
        yield 'J-Dog';
    }
    static *createJobIterator() {
        yield* ['Butcher', 'Baker', 'Candlestick maker'];
    }

    // 生成器方法作为默认迭代器
    // *[Symbol.iterator]() {
    //     yield * this.arg.entries();
    // }

    // 只返回迭代器实例
    [Symbol.iterator]() {
        return this.arg.entries();
    }
}
const p = new Per();
let nicknameIterator = p.createNicknameIterator();
console.log(nicknameIterator.next().value); // 'Jack'
console.log(nicknameIterator.next().value); // 'Jake'
console.log(nicknameIterator.next().value); // 'J-Dog'

let jobIterator = Per.createJobIterator();
console.log(jobIterator.next().value); // 'Butcher'
console.log(jobIterator.next().value); // 'Baker'
console.log(jobIterator.next().value); // 'Candlestick maker'

const p2 = new Per('401', '402', '403');
for (let [index, errorCode] of p2) {
    console.log(index + '=>' + errorCode);
}
// '0=>401'  '1=>402'  '2=>403'

4 继承

ES6 新增特性原生支持了类继承机制。虽然类继承使用的是新语法,但背后依旧使用的是原型链。

4.1 基础语法

ES6 类支持单继承。使用 extends 关键字,就可以继承任何拥有 [[Constructor]] 和原型的对象(这就包括着普通的构造函数)。派生类都会通过原型链访问到类和原型上定义的方法

// 超类
class Sclass {
    identifyPrototype(str) {
        console.log(str, this);
    }
    static identifyClass(str) {
        console.log(str, this);
    }
}

// 派生类,可以使用表达式的形式。
// const Dclass = class extends Sclass {};
class Dclass extends Sclass {}

let s0 = new Sclass();
let d0 = new Dclass();
s0.identifyPrototype('s'); // s Sclass {}
d0.identifyPrototype('d'); // d Dclass {}

Sclass.identifyClass('S'); // S [class Sclass]
Dclass.identifyClass('D'); // D class Dclass extends Sclass {}
console.log(Sclass.prototype === Dclass.prototype.__proto__); // true

4.2 构造函数、HomeObject 和 super()

派生类的方法可以通过 super 关键字引用它们的原型需要注意的是 super 是一个关键字,并且有一些特殊的语法结构。super 不是一个指向原型对象的变量super 语法总结:

  1. 这个关键字只能在派生类中使用(这里只是相对超类而言,在对象字面量中也可以使用,详细看 MDN 文档),而且仅限于类构造函数、实例方法和静态方法内部。
  2. 不能单独引用 super 关键字,要么作为函数调用,要么进行属性查询
  3. 在派生类构造函数中调用 super() 会调用父类构造函数,并将返回的实例赋值给 this。如果需要给父类构造函数传参需要手动传入
  4. 在构造函数中,不能在调用 super() 之前引用 this
  5. 如果派生类没有定义构造函数,则在实例化时默认调用 super(),且会传入所有传给派生类的参数。
  6. 派生类只要返回的是构造函数中的 this 对象,就必须调用 super()
class Sclass {
    constructor(name) {
        // SyntaxError: 'super' keyword unexpected here
        // super();
        this.name = name;
    }
    say() {
        console.log('log-in-S');
    }
}

class Dclass extends Sclass {
    constructor(name, identify) {
        // 不能再调用 super() 之前引用 this,否则会抛出 ReferenceError
        super(name);
        this.identify = identify;
    }
    say() {
        // 通过 super 调用父类原型对象上的成员
        super.say();
        console.log('log-in-D');
    }
}

let d0 = new Dclass('派生类', 'd0');
console.log(d0); // Dclass { name: '派生类', identify: 'd0' }
d0.say(); // 'log-in-S'   'log-in-D'

// ----------第5点和第6点------
class Sclass {
    constructor(name) {
        this.name = name;
    }
}

// class Dclass extends Sclass {}
class Dclass extends Sclass {
    constructor(name) {
        // super(name);
        return {};
    }
}

let d1 = new Dclass('派生类');
console.log(d1); // 对应结果分别为 Dclass { name: '派生类' }  {}

ES6 给类构造函数和静态方法添加了内部属性 [[HomeObject]],这个特性是一个指针,指向定义该方法的对象。这个指针是自动赋值的,且供 JavaScript 引擎内部访问。super 始终定义为 [[HomeObject]] 的原型。

4.3 抽象基类

抽象基类是指通过在基类的构造函数中对类实例化过程进行控制,从而达到定义特殊基类的概念。例如定义一个可供继承,但本身不能被实例化的基类。定义派生类必须具有某个方法的基类。

class Sclass {
    constructor(name) {
        if (new.target === Sclass) {
            throw new Error(`${new.target} 不能被实例化`);
        }
        if (!this.say) {
            throw new Error(`${new.target} 必须定义一个 say 方法`);
        }
    }
}

class Dclass extends Sclass {}

// Error: class Sclass{...} 不能被实例化
let s0 = new Sclass();

//  Error: class Dclass extends Sclass {} 必须定义一个 say 方法
// let d0 = new Dclass();

4.4 继承内置类型

ES6 类的继承语法使得内置引用类型的继承变得更加方便。但是有些内置类型的方法会返回新的实例。默认情况下,返回实例的类型与原始实例的类型是一致的。如果需要覆盖这个默认行为,可以覆盖 Symbol.species 访问器,这个访问器决定在创建返回的实例时使用的类

class DeriveArray extends Array {
    shuffle() {
        this.map((item, index) => {
            const j = Math.floor(Math.random() * this.length);
            [this[index], this[j]] = [this[j], this[index]];
        });
    }
    // static get [Symbol.species]() {
    //     return Array;
    // }
}

let a = new DeriveArray('A', '2', '3', '4', 'J', 'Q', 'K');
console.log(a instanceof Array); // true
console.log(a instanceof DeriveArray); // true

// DeriveArray(7) ['A', '2', '3', '4', 'J', 'Q', 'K']
console.log(a);
a.shuffle();

// DeriveArray(7) ['4', 'Q', 'J', 'A', 'K', '3', '2']
console.log(a);

let a0 = a.filter(item => !isNaN(item));
console.log(a0); // DeriveArray(3) [ '3', '2', '4' ]
console.log(a0 instanceof DeriveArray); // true

// 覆盖 Symbol.species 访问器后创建返回的实例使用的类不再跟原始实例一样
// console.log(a0 instanceof DeriveArray); // false

4.5 类混入

略。。。

很多 JavaScript 框架(特别是 React)已经抛弃混入模式,转向了组合模式(把方法提取到独立的类和辅助对象中,然后把它们组合起来,但不使用继承)。这反映了“组合胜过继承”的软件设计原则。

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

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

相关文章

TTL反相器、OC门、TS门、推挽输出、开漏输出

TTL反相器 这是一个TTL反相器,这是经过了很多工程师多种设计最终沉淀出来的电路,这个电路是比较成熟的。我们只需要对这个电路进行解析即可,不需要再去研究当初是如何设计出来的。 学过CMOS应该知道,右侧的输出级其实也是个推挽输…

为什么程序实际可用内存会远超物理内存?

背景介绍 不知道在你刚接触计算机的时候,有没有这么一个疑问:“为什么我的机器上只有两个G 的物理内存,但我却可以使用比这大得多的内存,比如 256T?”反正我当时还是挺疑惑的,不过现在我可以来告诉你这个答…

如何将销售效果最大化:从人工智能聊天机器人到即时聊天

聊天机器人的崛起 从几年前开始,客户支持是聊天机器人使用的第一个爆发点。B2C引领潮流,B2B紧随其后。市场营销和销售最近也开始加入,让聊天机器人承担起迎接和引导网站游客的任务。现在,人工智能已经进入聊天,可以说…

机器学习 第一周

目录 1. 什么是机器学习(课本给出的部分定义) 我理解的机器学习:

Java基础:IO流有哪些,各有什么特点和功能

具体操作分成面向字节(Byte)和面向字符(Character)两种方式。 如下图所示: IO流的三种分类方式 IO流的层次结构 IO流的常用基类有: 字节流的抽象基类:InputStream和OutputStream; 字符流的抽象基类:Reader和Writer…

20、单元测试

文章目录 1、JUnit5 的变化2、JUnit5常用注解3、断言(assertions)1、简单断言2、数组断言3、组合断言4、异常断言5、超时断言6、快速失败 4、前置条件(assumptions)5、嵌套测试6、参数化测试7、迁移指南 【尚硅谷】SpringBoot2零基…

JAVA性能优化实例

目录 概述 Sql性能优化 多线程 利用内存缓存 功能优化 参考博客 概述 性能优化的几个点,大致可以分为: sql优化使用多线程利用内存,缓存等,将固定不常更改的数据放入在,存取更快的内存当中功能实现逻辑优化 Sql性…

五分钟,说说Python 中多线程共享全局变量的问题

嗨害大家好鸭!我是爱摸鱼的芝士❤ 写在前面不得不看的一些P话: Python 中多个线程之间是可以共享全局变量的数据的。 但是,多线程共享全局变量是会出问题的。 假设两个线程 t1 和 t2 都要对全局变量g_num (默认是0)进行加1运算&#xff0c…

日常项目技术方案脉络

开篇引砖 软件在其生命周期中,当其进入稳定期后,大部分时间都处于迭代更新维护阶段。在这漫长的三年甚至五年的存活期内,我们需要面对林林种种大大小小的需求。今天我们就聊聊在这段期间,如何快速产出一份合格的技术方案。 方案给…

JavaScript经典教程(一)-- HTML基础部分

179:HTML基础部分(元素分类、特性、特殊元素等) 1、复习: HTML: 超文本标记语言 CSS: 层叠样式表 JavaScript: 脚本语言 http:超文本传输协议 https: 经过ssl加密的超…

如何搭建关键字驱动自动化测试框架?这绝对是全网天花板的教程

目录 1. 关键字驱动自动化测试介绍 2. 搭建关键字驱动自动化测试框架 步骤1:选择测试工具 步骤2:定义测试用例 步骤3:编写测试驱动引擎 步骤4:实现测试关键字库 步骤5:执行测试 3. 实现关键字驱动自动化测试的关…

c/c++:gets(),fgets(),puts(),fputs(),strlen(),字符串拼接函数

c/c:gets(),fgets(),puts(),fputs(),strlen(),字符串拼接函数 2022找工作是学历、能力和运气的超强结合体,遇到寒冬,大厂不招人,此时学会c的话, 我所知道的周…

C语言程序环境与预处理回顾总结

大概讲解与铺垫 首先,什么叫c语言的源代码?也就是我自己写的.c文件里面的代码,这个就叫做源代码。然后需要知道的是计算机他只认识二进制,因此他只能接收与执行二进制指令。也就是可执行的机器指令。然后我们必须得知道&#xff…

JavaWeb——tomcat(安装使用)

目录 WEB服务器-Tomcat 服务器概述 Web服务器 Tomcat Tomcat下载 安装与卸载 启动与关闭 WEB服务器-Tomcat 服务器概述 服务器硬件: 指的也是计算机,只不过服务器要比我们日常使用的计算机大很多。 服务器,也称伺服器。是提供计算服务…

【数据结构】第十一站:链式二叉树

目录 一、二叉树的创建 二、二叉树的遍历 1.前序中序后序遍历 2.层序遍历 三、二叉树的节点个数以及高度等 四、二叉树的构建和销毁 五、DFS和BFS 一、二叉树的创建 为了方便后面的讨论,我们在这里先手撕一颗二叉树 typedef int BTDateType; typedef struct …

Qt Designer

Qt Designer——即Qt设计师,是QT项目开发的可视化图形界面编辑器,通过设计师可以很方便地创建图像界面文件*.ui,然后将ui文件应用的源代码中,做到所见即所得,让界面的设计变得十分简单。下面介绍Qt Designer的简单使用…

使用PowerShell自动部署ASP.NetCore程序到IIS

asp.net core 安装asp.net core sdk https://dotnet.microsoft.com/en-us/download/dotnet/3.1 创建asp.net core项目 dotnet new webapi运行项目 访问https://localhost:5001/WeatherForecast iis配置 安装iis 以管理员身份运行powershell Enable-WindowsOptiona…

【计算机网络】学习笔记:第二章 物理层(五千字详细配图)【王道考研】

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; 给大家跳段街舞感谢支持&#xff01;ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ…

Windows中Seata连接Linux中的nacos的配置

目录 前言&#xff1a; Seata配置教程&#xff1a; file.conf: ​编辑 配置代码&#xff1a; registry.conf: ​编辑 nacos-config.sh: config.txt: 前言&#xff1a; 许多博主&#xff0c;把两个软件都安装在同一个系统中&#xff0c;但是有些人估计是装在不同的环境下…

支付系统设计:收银台设计二

文章目录 前言1. 接口校验1.1 Chains1.2 Checker1.2.1 AbstractChecker1.2.2 TokenChecker1.2.3 OrderChecker1.2.4 UserInfoChecker1.2.5 BaseInfoChecker1.2.6 SignChecker 1.3 ApiFilter 2. 下单3. 收银台首页2.1 OrderInfoResolver2.2 UserBaseInfoResolver 4. 执行流程总结…