【c++】cpp运算符重载

目录

(1)什么是运算符重载

(2)运算符重载的本质是函数调用

(3)可以与不可以重载的运算符

(4)单目运算符与双目符重载区别

(5)双目运算符重载举例

重载=操作符

用友元函数实现重载<<操作符

重载下标运算符[]

重载函数调用符()

(6)单目运算符重载举例

重载右值++运算符

用双目运算符的思想重载左值++运算符

重载箭头->运算符

(7)运算符重载步骤总结

(8)为什么不建议重载&&和||操作符

(9)运算符重载在项目开发中的应用

自定义字符串类

自定义智能指针类


(1)什么是运算符重载

⽤复数类举例

Complex c3 = c1 + c2;

原因 Complex是⽤户⾃定义类型,编译器根本不知道如何进⾏加减。

编译器给提供了⼀种机制,让⽤户⾃⼰去完成,⾃定义类型的加减操作等。

这个机制就是运算符重载机制。

(2)运算符重载的本质是函数调用

全局函数角度:c1 + c2 从全局函数角度等价于函数调用:operator+(c1, c2)。

声明:Complex operator+(Complex &c1, Complex &c2)  // 需要定义的全局函数

注意:运算符重载函数要操作私有成员时,需要在类中声明为友元函数。

使用:

  • Complex c4 = operator+(c1, c2);  // 正常像函数那样使用,函数名是"operator+"
  • Complex c4 = c1 + c2;   // 用符号使用

类成员函数角度:c1 - c2 从类成员函数角度 等价于函数调用:c1.operator-(c2)。

声明:Complex operator-(Complex &c2)  // 需要定义类成员函数

因为c1-c2等价于c1.operator-(c2),要想能使用c1-c2

就需要在c类的内部自定义Complex operator-(Complex &c_object)函数

再以重载=为例,A类,通过类成员函数重载:a1=a2,等价于a1.operator=(a2),所以需要在A类的内部自定义A operator=(Complex &A_object)函数。

示例代码

#include <iostream>class Complex
{
public:int a;int b;
public:Complex(int a, int b){this->a = a;this->b = b;}void printCom(){std::cout << a << "+" << b << "i" << std::endl;}
public:// 类成员函数实现运算符重载Complex operator*(Complex &c2){Complex temp(this->a * c2.a - this->b * c2.b, this->a * c2.b + this->b * c2.a);return temp;}
};// 全局函数实现特定对象之间的运算
Complex com_add(Complex &c1, Complex &c2)
{Complex temp(c1.a + c2.a, c1.b + c2.b);return temp;
}// 全局函数实现运算符重载
Complex operator+(Complex &c1, Complex &c2)
{Complex temp(c1.a + c2.a, c1.b + c2.b);return temp;
}int main()
{Complex c1(1, 2), c2(3, 4);// a + bi 复数运算规则// 类也是一种数据类型// 用户自定义数据类型 两个对象之间运算 C++编译器 是不知道如何进行运算// 1.定义全局函数进行计算Complex c3 = com_add(c1, c2);c3.printCom(); //4+6i// 2.运算符重载// c++编译器应该给我们程序员提供一种机制// 让自定义数据类型 有机会 进行 运算符操作 ====> 运算符重载机制// 全局函数实现Complex c4 = operator+(c1, c2);c4.printCom(); // 4+6iComplex c5 = c1 + c2;c5.printCom(); // 4+6i// 类成员函数实现Complex c6 = c1 * c2;c6.printCom(); //-5+10ireturn 0;
}

(3)可以与不可以重载的运算符

可以重载的运算符:

+-*/%^&|~
!=<>+=-=*=/=%
^=&=|=<<>>>>=<<===!=
<=>=&&||++--->*->[ ]
()newdeletenew[]delet[]

不能重载的运算符

.::.*?:sizeof

(4)单目运算符与双目符重载区别

单目运算符:比如++,像a++,++b等。

双目运算符:比如+, =, ==,像a+b,c=d,c==d等。

运算符重载函数:[返回值] operator[运算符] (参数...) { ... };

对于参数:重载函数的参数个数必须与运算符原来的个数一致。比如+号的参数就是左加数和右加数两个。那么当我们重载加号时也要保证有左右两个加数作为参数。

对于双目运算符

  • 用全局函数重载时,左边形参作为运算符左操作数,右边形参是右操作数。

  • 用类成员函数重载时,只需要写一个参数即可,因为类的成员函数默认有一个this指针,有一个参数就已经被this指针包含了,this指向运算符左参数。以减号为例,两个时间类a和b相减时。如果是a - b,那么this指针指向a,反之则指向b。在声明函数时,我们只需要写右参数即可。a - b的话只需要写 int operator-(Time b);

对于单目运算符

运算符在左侧,参数在右侧,比如++a(说明:a++这种比较特殊,运算符是在右侧,这种用用特殊技巧重载)

  • 用全局函数重载时,右操作数是形参。
  • 用类成员函数重载时,单目运算符由于只有一个参数,且该参数被this所指向,那么我们无需声明任何参数即可。this所指向运算符右参数。

对于返回值:运算符的返回值类型取决于该重载函数的作用是什么。

  • 比如a + b的作用是得到一个加数,那么返回值就是a+b的值。a += b的作用是让a的值改变,既然是让参数a的值改变,那么就无需返回值。
  • 还有就是如果我们需要运算符支持多次操作那么也需要返回值。比如流插入运算符<<。我们可能需要多次进行插入,像cout << a << b << c;之类就需要返回流ostream本身以便于之后的流插入工作。

(5)双目运算符重载举例

重载=操作符

示例代码

#include <iostream>
#include <stdlib.h>
#include <cstring>class Name
{
public:Name(const char *p, int index){m_len = strlen(p);m_p = (char *)malloc(m_len + 1);strcpy(m_p, p);this->index = index;}Name(const Name &name){std::cout << "copy generate func" << std::endl;m_len = name.m_len;m_p = (char *)malloc(m_len + 1);strcpy(m_p, name.m_p);this->index = 0;}/*重载=运算符首选因为obj3 = obj1; 所以obj3.operator=(obj1)再因为有链式编程 obj1 = obj2 = obj3的需要,所以需要返回引用*/Name &operator=(Name &name){std::cout << "operator= func" << std::endl;if (this->m_p != NULL){free(m_p);m_p = NULL;m_len = 0;}this->m_len = name.m_len;this->m_p = (char *)malloc(m_len + 1);;strcpy(m_p, name.m_p);// 返回左操作参数的引用std::cout << getIndex() << std::endl;return *this;}~Name(){if (m_p != NULL){free(m_p);m_p = NULL;m_len = 0;}}
public:void printInfo(){std::cout << "&(*this)=" << this << ", name=" << this->m_p << ", index=" << this->index<< std::endl;// &(*this) = this的值}void setIndex(int index){this->index=index;}int getIndex(){return index;}
protected:
private:char *m_p;int m_len;int index;
};void objplaymain()
{std::cout << "-------------A--------------" << std::endl;Name obj1("hello world", 1);Name obj2 = obj1;  // 调用拷贝构造函数obj2.setIndex(2);obj2.printInfo(); std::cout << "-------------B--------------" << std::endl;Name obj3("obj3", 3);obj3 = obj1;  // 调用运算符重载函数  // obj3.operator=(obj1)obj3.printInfo(); std::cout << "-------------C--------------" << std::endl;Name obj4("obj4", 4);obj1 = obj4 = obj3;  // 调用运算符重载函数//思考obj1 = obj4 = obj3;的运行过程!//从右到左,先obj4 = obj3,同时根据类中=重载定义,返回的是obj4//然后obj1 = obj4,返回obj1
}int main()
{objplaymain();return 0;
}

运行结果

-------------A--------------
copy generate func
&(*this)=0x7ffd72212180, name=hello world, index=2
-------------B--------------
operator= func
3
&(*this)=0x7ffd72212170, name=hello world, index=3
-------------C--------------
operator= func
4
operator= func
1

用友元函数实现重载<<操作符

当⽆法修改左操作数的类时,使⽤友元函数进⾏重载

istream 和 ostream 是 C++ 的预定义流类,cin 是 istream 的对象,cout 是 ostream 的对象。

运算符 << 由ostream 重载为插⼊操作,⽤于输出基本类型数据。

运算符 >> 由 istream 重载为提取操作,⽤于输⼊基本类型数据。

⽤友员函数重载 > ,输出和输⼊⽤户⾃定义的数据类型。

由于编译器ostream源码我们是拿不到的,所以不能通过设置成员函数来重载<<运算符,这里可以通过友元函数实现。

示例代码

#include <iostream>class Complex
{
private:int a;int b;
public:// friend void operator<<(std::ostream &os_obj, Complex &c1);friend std::ostream &operator<<(std::ostream &os_obj, Complex &c1);
public:Complex(int a, int b){this->a = a;this->b = b;}void printCom(){std::cout << a << "+" << b << "i" << std::endl;}
};/*
void operator<<(std::ostream &std::cout, Complex &c1)
{std::cout << c1.a << "+" << c1.b << "i" << std::endl;
}
*/std::ostream &operator<<(std::ostream &os_obj, Complex &c1)
{os_obj << c1.a << "+" << c1.b << "i ";return os_obj;
}int main()
{// 1.编译器按照已有数据类型输出int a = 10;std::cout << a << std::endl;// 2.对于自定义数据类型,需重载运算符输出// 这里用友元函数重载了<<运算符// 使得既能输出基础数据类型,也能输出自定义类型,也能支持链式编程Complex c1(1, 2);Complex c2(3, 4);std::cout << c1;std::cout << std::endl;std::cout << c1 << "hello world" << std::endl;std::cout << c1 << c2 << "xxxxxxx" << std::endl;return 0;
}

运行结果

10
1+2i 
1+2i hello world
1+2i 3+4i xxxxxxx

重载下标运算符[]

设 x 是类 X 的一个对象,则表达式x [ y ] 可被解释为 x . operator [ ] ( y )

示例代码

#include <iostream>class Data
{
public:Data(int value = 0){this->value = value;}
public:int value;    
};class Test
{
public:Test(int a = 0, int b = 0){this->a = a;this->b = b;}
public:// 重载[],使得能传递整数类型int operator[](int index){return a * index;}// 重载[],使得能传递Test类型int operator[](const Test &t){return b * t.a;}// 重载[],使得能传递Data类型,且能链式编程Test &operator[](const Data &d){std::cout << d.value << "\t";return *this;}
private:int a;int b;
};int main()
{Test test(1, 2);// []传递整型std::cout << test[3] << std::endl;// []传递Test类型Test test2(3, 4);std::cout << test[test2] << std::endl;// []传递Data类型Data data1(3), data2(18), data3(99);test[data1][data2][data3];return 0;
}

运行结果

3
6
3	18	99

重载函数调用符()

设 x 是类 X 的一个对象,则表达式x ( arg1, arg2, … )可被解释为x . operator () (arg1, arg2, … )

#include <iostream>
#include <string>// 类的声明
class F
{
public:double operator()(double x, double y);std::string operator()(std::string x, std::string y);
};// 类的具体实现
double F::operator()(double x, double y)
{return x * x + y * y;
}std::string F::operator()(std::string x, std::string y)
{std::string ret = "(";if (x.size() == 0)ret += " ";elseret += x[0];if (y.size() == 0)ret += " ";elseret += y[0];ret += ")";return ret;
}int main()
{F f;std::cout << f(5.2, 2.5) << std::endl;; // 33.29std::cout << f.operator()(5.2, 2.5) << std::endl;; // 33.29std::cout << f("hello", "world") << std::endl;; // (hw)std::cout << f("", "xxxxx") << std::endl;; // ( x)return 0;
}

(6)单目运算符重载举例

重载右值++运算符

右值++运算符,++b,可被解释为operator ++ ( b )

示例代码

#include <iostream>
using namespace std;class Complex
{
private:int a;int b;public:Complex(int a, int b){this->a = a;this->b = b;}void print_value(){cout << a << "+" << b << "i" << endl;}public:// 右值++重载用成员函数实现Complex &operator++(){std::cout << "internal reload" << std::endl;this->a++;this->b++;return *this;}// 右值--重载用全局函数实现friend Complex &operator--(Complex &c);
};Complex &operator--(Complex &c)
{std::cout << "external reload" << std::endl;c.a--;c.b--;return c;
}int main()
{Complex c1(1, 2);Complex temp = ++c1; // internal reloadc1.print_value();    // 2+3itemp.print_value();  // 2+3i--c1;   // external reloadc1.print_value();    // 1+2ireturn 0;
}

运行结果

internal reload
2+3i
2+3i
external reload
1+2i

用双目运算符的思想重载左值++运算符

++作为单目运算符,直接重载的话,实现的是右值++,要重载左值++,技巧在于用一个占位符作为运算符重载函数的第二个参数,这样原本的变量就变成了左操作数。

示例代码

#include <iostream>class Complex
{
private:int a;int b;// 重载右值++friend Complex &operator++(Complex &c1);// 重载左值++friend Complex operator++(Complex &c1, int);
public:
Complex(int a = 0, int b = 0){this->a = a;this->b = b;}void printCom(){std::cout << a << "+" << b << "i" << std::endl;}
public:// 重载右值--Complex &operator--(){this->a--;this->b--;return *this;}// 用占位符实现重载左值++Complex operator--(int){Complex temp = *this;this->a -= 10;this->b -= 10;return temp;}
};// 重载右值++
Complex &operator++(Complex &c1)
{c1.a++;c1.b++;return c1;
}// 重载左值++
// 用占位符区分右值++和左值++
// 这里添加一个占位符,实现左值++重载
Complex operator++(Complex &c1, int)
{Complex temp = c1;c1.a += 10;c1.b += 10;return temp;
}int main()
{Complex c1(1, 2);// 为了区分左值++和右值++正确被重载// 右值++的变化量为1,左值++的变化量为10// 重载左值++操作符Complex temp = c1++;//Complex temp = operator++(c1, 3);c1.printCom(); // 11+12itemp.printCom(); // 1+2i// 重载右值++操作符temp = ++c1;//temp = operator++(c1);c1.printCom(); // 12+13itemp.printCom(); // 12+13i// 重载左值--操作符temp = c1--;c1.printCom(); // 2+3itemp.printCom(); // 12+13i// 重载右值--操作符temp = --c1;c1.printCom(); // 1+2itemp.printCom(); // 1+2ireturn 0;
}

重载箭头->运算符

->一般用在指针当中。对它的重载稍微特殊一点,设 x 是类 X 的一个对象,则对于表达式x -> y :

  • 需要先看x ->部分,对应运算符重载函数x. operator ->(),如果该函数返回指针,比如返回p,则x ->就相当于返回p->,然后x -> y就相当于p->y,就是指针访问成员y,这里y可以是成员变量,比如p->a,也可以是成员函数,比如p->action()。
  • 如果函数x. operator ->()返回的不是指针,而是一个对象(比如m,假设m是类M的一个对象),则继续对该对象调用其重载了的箭头运算符(意味着类M也必须要对->进行重载,否则会报错),直到返回的是一个指针,假设为p,则最后就是在调用p->y。

示例代码

#include <iostream>class firstClass 
{
public:firstClass* operator->() {std::cout << "firstClass ->() is called!" << std::endl;return this;}void action() {std::cout << "firstClass action() is called!" << std::endl;return;}
};class myClass 
{firstClass firstObj;
public:firstClass& operator->() {std::cout << "myClass ->() is called!" << std::endl;return firstObj;}void action() {std::cout << "myClass action() is called!" << std::endl;return;}
};int main() 
{myClass obj;obj->action();return 0;
}

运行结果

myClass ->() is called!
firstClass ->() is called!
firstClass action() is called!

(7)运算符重载步骤总结

运算符重载步骤

1)要承认操作符重载是一个函数,写出函数名称。

2)根据操作数,写出函数参数。

3)根据业务,完善函数返回值(看函数是返回引用 还是指针 还是元素),及实现函数业务。

4)根据情况考虑用全局函数还是类成员函数方法实现。

(8)为什么不建议重载&&和||操作符

&&和||是C++中非常特殊的操作符,&&和||内置实现了短路规则。

操作符重载是靠函数重载来完成的,操作数作为函数参数传递,C++的函数参数都会被求值,无法实现短路规则。

以下示例代码演示了可以实现&&的运算符的重载,但无法实现短路规则。

#include <iostream>class Test
{
private:int i;
public:Test(int i = 0){this->i = i;}
public:Test operator+(const Test &obj){Test ret(0);std::cout << "+ reload func" << std::endl;ret.i = i + obj.i;return ret;}bool operator&&(const Test &obj){std::cout << "&& reload func" << std::endl;return i && obj.i;}
};int main()
{// 1.查看&&的短路规则int a1 = 0;int a2 = 1;// 注意:&&操作符的结合顺序是从左向右if (a1 && a2++) //有一个是假,则不在执行下一个表达式的计算{/* */}std::cout << a2 << std::endl; // 1// 2.重载&&后无法实现短路规则Test t1 = 0;Test t2 = 1;// if( t1 && (t1 + t2) )// t1 && t1.operator+(t2)// t1.operator&&( t1.operator+(t2) )// && || 重载他们 不会产生短路效果if (t1 && (t1 + t2)) { ; }// t1.operator+(t2) && t1;//(t1.operator+(t2)).operator&&(t1);// 两个函数都被执行了,而且是先执行了+// 说明没有实现短路规则return 0;
}

运行结果

1
+ reload func
&& reload func

(9)运算符重载在项目开发中的应用

自定义字符串类

用来练习各种运算符重载

构造函数要求

MyString a;

MyString a(“dddd”);

MyString b = a;

常用的操作符

> != == > < =

MyString.h

#include <iostream>
using namespace std;//C中没有字符串,因此需要建一个字符串类
//C++中 我们来设计一个字符串 以零结尾的字符串
//空串 ""class MyString
{
public://用友元函数重载输入输出流friend ostream &operator<<(ostream &cout, MyString &s);friend istream &operator>>(istream &cin, MyString &s);
public://构造函数和析构函数MyString(int len = 0); //传入整数构造MyString(const char *p); //传入字符串构造MyString(const MyString &s); //传入字符串类对象构造~MyString();public://重载等号操作符 =MyString& operator=(const char *p);MyString& operator=(const MyString &s);//重载下标操作符 []public:char& operator[](int index);//重载 == != 操作符public:bool operator==(const char *p) const;bool operator==(const MyString &s) const;bool operator!=(const char *p) const;bool operator!=(const MyString &s) const;//重载 大于小于操作符
public:int operator<(const char *p) const;int operator>(const char *p) const;int operator<(const MyString &s) const;int operator>(const MyString &s) const;
public://字符串类技巧:把指针露出来char *c_str(){return m_p;}const char*c_str2(){return m_p;}int getlen(){return m_len;}
private:int m_len;char *m_p;
};

MyString.cpp

#include <iostream>
using namespace std;
#include "MyString.h"
#include <cstring>ostream &operator<<(ostream &cout, MyString &s)
{cout << s.m_p;return cout;
}istream &operator>>(istream &cin, MyString &s)
{cin >> s.m_p;return cin;
}// MyString::MyString(int len = 0) 
// 会error,‘MyString::MyString(int)’的第 1 个形参指定了默认实参
MyString::MyString(int len)
{if (len == 0){m_len = len;m_p = new char[m_len + 1];strcpy(m_p, "");}else{m_len = len;m_p = new char[m_len + 1];memset(m_p, 0, m_len);}
}MyString::MyString(const char *p)
{if (p == NULL){m_len = 0;m_p = new char[m_len + 1];strcpy(m_p, "");}else{m_len = strlen(p);m_p = new char[m_len + 1];strcpy(m_p, p);}
}//拷贝构造函数
// MyString s3 = s2;
MyString::MyString(const MyString &s)
{m_len = s.m_len;m_p = new char[m_len + 1];strcpy(m_p, s.m_p);
}MyString::~MyString()
{if (m_p != NULL){delete[] m_p;m_len = 0;m_p = NULL;}
}// S4 = "hello world";
MyString &MyString::operator=(const char *p)
{// 1.先释放旧内存if (m_p != NULL){delete[] m_p;m_len = 0;}// 2.根据p分配内存if (p == NULL){m_len = 0;m_p = new char[m_len + 1];strcpy(m_p, "");}else{m_len = strlen(p);m_p = new char[m_len + 1];strcpy(m_p, p);}return *this;
}// s4 = s2;
MyString &MyString::operator=(const MyString &s)
{// 1 旧内存释放掉if (m_p != NULL){delete[] m_p;m_len = 0;}// 2 根据s分配内存m_len = s.m_len;m_p = new char[m_len + 1];strcpy(m_p, s.m_p);return *this;
}char &MyString::operator[](int index)
{return m_p[index];
}// (s2 == "hello world")
bool MyString::operator==(const char *p) const
{if (p == NULL){if (m_len == 0){return true;}else{return false;}}else{if (m_len == strlen(p)){return !strcmp(m_p, p);}else{return false;}}
}bool MyString::operator!=(const char *p) const
{return !(*this == p);
}bool MyString::operator==(const MyString &s) const
{if (m_len != s.m_len){return false;}return !strcmp(m_p, s.m_p);
}bool MyString::operator!=(const MyString &s) const
{return !(*this == s);
}// (s3 < "bbbb")
int MyString::operator<(const char *p) const
{return strcmp(this->m_p, p);
}int MyString::operator>(const char *p) const
{return strcmp(p, this->m_p);
}int MyString::operator<(const MyString &s) const
{return strcmp(this->m_p, s.m_p);
}int MyString::operator>(const MyString &s) const
{return strcmp(s.m_p, m_p);
}

test.cpp

#include <iostream>
using namespace std;
#include <cstring>
#include "MyString.h"int main01()
{MyString s1(1);MyString s2("s2");// MyString s2_2 = NULL; //这句会报错MyString s2_2 = "hello world";MyString s3 = s2;MyString s4 = "s4444444444";//测试运算符重载 = [] <<s4 = s2;cout << s4 << endl; // "s2"s4 = "s2222";s4[1] = '4';cout << s4 << endl; // "s4222"return 0;
}int main02()
{MyString s1;MyString s2("s2");MyString s3 = s2;if (s2 == "aa")printf("equal\n");elseprintf("no equal\n");if (s3 == s2)printf("equal\n");elseprintf("no equal\n");return 0;
}int main03()
{MyString s1;MyString s2("s2");MyString s3 = s2;s3 = "aaaa";int tag = (s3 < "bbb");if (tag < 0)printf("s3 less than bbb\n");elseprintf("s3 more than bbb\n");MyString s4 = "aaaaffff";strcpy(s4.c_str(), "aa111");cout << s4 << endl;return 0;
}int main04()
{MyString s1(128);cout << "\n input content(Enter finish):"; // hello worldcin >> s1;cout << s1; // hello//目前没有正确处理空格//可以完善return 0;
}int main()
{// main01();// main02();// main03();main04();return 0;
}

自定义智能指针类

1问题抛出:指针使用过程中,经常会出现内存泄漏和内存多次被释放。

2 解决方案:例如:boost库的智能指针。项目开发中,要求开发者使用预先编写的智能指针类对象代替C语言中的原生指针。

3 智能指针思想

  • 工程中的智能指针是一个类模板
  • 通过构造函数接管申请的内存
  • 通过析构函数确保堆内存被及时释放
  • 通过重载指针运算符* 和 -> 来模拟指针的行为
  • 通过重载比较运算符 == 和 != 来模拟指针的比较

示例代码

#include <iostream>
using namespace std;class Test
{
public:Test(){this->a = 10;}void printT(){cout << a << endl;}
private:int a;
};class MyTestPointer
{
public:
public:MyTestPointer(){p = NULL;}MyTestPointer(Test *p){this->p = p;}~MyTestPointer(){delete p;}Test *operator->(){return p;}Test &operator*(){return *p;}protected:Test *p;
};class MyIntPointer
{
public:
public:MyIntPointer(){p = NULL;}MyIntPointer(int *p){this->p = p;}~MyIntPointer(){delete p;}int *operator->(){return p;}int &operator*(){return *p;}protected:int *p;
};void test1()
{Test *p = new Test;p->printT();delete p;MyTestPointer myp = new Test; //构造函数myp->printT(); //重载操作符 ->
};void test2()
{int *p = new int(100);cout << *p << endl;delete p;MyIntPointer myp = new int(200);cout << *myp << endl; //重载*操作符
};int main()
{test1();test2();return 0;
}

end

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

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

相关文章

Web3的可持续性:构建环境友好的去中心化系统

引言 随着全球对可持续发展和环境问题的日益关注&#xff0c;Web3技术作为一种新型的互联网模式&#xff0c;也开始受到社区和开发者的关注。但很少有人关注到Web3对环境可持续性的潜在影响。本文将探讨Web3如何构建一个环境友好的去中心化系统&#xff0c;以及这如何促进一个…

偏微分方程算法之五点菱形差分法

目录 一、研究目标 二、理论推导 三、算例实现 四、结论 一、研究目标 上个专栏我们介绍了双曲型偏微分方程的主要算法及实现。从今天开始&#xff0c;我们在新的专栏介绍另一种形式偏微分方程-椭圆型的解法。 研究目标选取经典的二维椭圆型方程&#xff08;也称泊松Poisso…

马斯克突击访华;谷歌 Python 基础团队全数被裁;丨 RTE 开发者日报 Vol.195

开发者朋友们大家好&#xff1a; 这里是 「RTE 开发者日报」 &#xff0c;每天和大家一起看新闻、聊八卦。我们的社区编辑团队会整理分享 RTE&#xff08;Real Time Engagement&#xff09; 领域内「有话题的 新闻 」、「有态度的 观点 」、「有意思的 数据 」、「有思考的 文…

python基础学习之写入csv文件

前言 在Python编程中&#xff0c;经常会遇到要将数据存到csv文件中&#xff0c;今天来详细讲一下。 一.功能目的 将以下数据存到csv文件中。 data [ {name: Alice, age: 25, city: New York}, {name: Bob, age: 30, city: Los Angeles}, {name: Charlie, age: 35, city:…

共享模型之无锁——Unsafe

文章目录 概述Unsafe CAS 操作 名字虽然叫Unsafe,但并不是线程不安全&#xff0c;而是因为他会操作内存&#xff0c;操作线程&#xff0c;不建议开发人员使用。 概述 Unsafe 对象提供了非常底层的&#xff0c;操作内存、线程的方法&#xff0c;Unsafe 对象不能直接调用&#x…

群狼调研(长沙商业咨询)广告效果测评的关键指标

广告效果测评涉及多个关键指标&#xff0c;以下是其中一些常见的指标&#xff1a; 1.广告到达&#xff08;Ad Reach&#xff09;&#xff1a;衡量广告在目标受众中的覆盖范围和到达程度。它可以包括广告曝光的人数、频次、覆盖率等方面的评估。 2.广告认知&#xff08;Ad Aware…

基于Python和Selenium的BOSS直聘Python岗位数据分析系统的设计与实现

基于Python和Selenium的BOSS直聘Python岗位数据分析系统的设计与实现 Design and Implementation of Python-based Selenium-powered BOSS Direct Recruitment Python Job Data Analysis System 完整下载链接:基于Python和Selenium的BOSS直聘Python岗位数据分析系统的设计与实…

H5 录音功能

Recorder: html5 js 录音 mp3 wav ogg webm amr g711a g711u 格式&#xff0c;支持pc和Android、iOS部分浏览器、Hybrid App&#xff08;提供Android iOS App源码&#xff09;、微信&#xff0c;提供ASR语音识别转文字 H5版语音通话聊天示例 DTMF编码解码 git 地址&#xff1a…

DS:单链表的实现

欢迎各位来到 Harper.Lee 的编程学习小世界&#xff01; 博主主页传送门&#xff1a;Harper.Lee的博客 我将在这里分享我的学习过程等心得 创作不易&#xff0c;码字不易&#xff0c;兄弟们养成先赞后看的好习惯哦&#xff01; 想一同进步的uu&#xff0c;可以来后来找我哦&…

Selenium IDE 常见错误笔记

错误1&#xff1a;Failed:Exceeded waiting time for new window to appear 2000ms 这个错误通常出现在第一次运行时&#xff0c;有两个原因&#xff1a; Firefox阻止了弹出式窗口&#xff0c;在浏览器设置里允许这个操作即可。 有些网站设置了反扒机制&#xff0c;脚本运行…

解决Blender导出FBX文件到Unity坐标轴错误的问题

发现Blender的模型导入到Unity里面有问题,简单研究了下发现是坐标系不同,Unity使用的是左手坐标系,Blender使用的是右手坐标系 。 下面直接将如何解决 首先忽略Blender的右手坐标系以及Z轴朝上的事&#xff0c;依照unity坐标系情况修改模型物体的旋转&#xff0c;以Blender猴…

算法工程师——算法岗的分类及要求汇总

算法岗工程师 根据 Talent Seer 人才报告显示,全球 AI 从业者总人数约有 30 万,还是供不应求,其中 AI 技术专家(具有相关领域博士学位及 3 年以上工作经验的)约有 3.65 万。 简介 对于计算机专业的毕业生而言,算法岗基本上就是 「高薪」 的代名词。 在当今 IT 行业,算…

【城市】2023深圳市定居与生活相关政策(含租房、租车)

【城市】2023深圳市定居与生活相关政策&#xff08;含租房、租车&#xff09; 文章目录 一、户籍身份1、深圳市居住登记凭证、居住证&#xff08;点击就送&#xff09;2、深圳落户&#xff08;点击就送1&#xff09; 二、人才补贴人才引进补贴&#xff08;含应届生&#xff09;…

vue3左树的全选和反选

<el-input v-model"filterText" placeholder"" style"width: 48%"/><el-button type"primary" click"handleSearch" class"ml-2">查找</el-button><el-radio-group v-model"form.choic…

django.db.utils.NotSupportedError: MySQL 8 or later is required (found 5.7.26).

环境:django 4.11 mysql 5.7 python 3.12.1 时间:20240429 说明:版本不兼容,最初使用注释源码,但是感觉这种处理很低端,所以有了这篇小作文 解决方法一: 1 找到文件:aca\Lib\site-packages\django\db\backends\base\base.py 注释第239行, 即:self.check_database_versio…

学习笔记:能量信号与功率信号(一)

目录 一、能量信号&#xff08;Energy Signal&#xff09; 二、功率信号&#xff08;Power Signal&#xff09; 三、信号关系图 四、总结 能量信号和功率信号是信号分析中两个基本的概念&#xff0c;它们主要用来描述信号在时间域中能量分布的特性&#xff0c;对于理解信号…

Faststone Capture:一触即发的效率革命【AI写作】

首先&#xff0c;这篇文章是基于笔尖AI写作进行文章创作的&#xff0c;喜欢的宝子&#xff0c;也可以去体验下&#xff0c;解放双手&#xff0c;上班直接摸鱼~ 按照惯例&#xff0c;先介绍下这款笔尖AI写作&#xff0c;宝子也可以直接下滑跳过看正文~ 笔尖Ai写作&#xff1a;…

LeetCode题练习与总结:删除有序数组中的重复项Ⅱ--80

一、题目描述 给你一个有序数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使得出现次数超过两次的元素只出现两次 &#xff0c;返回删除后数组的新长度。 不要使用额外的数组空间&#xff0c;你必须在 原地 修改输入数组 并在使用 O(1) 额外空间的条件下完…

苹果发布开源模型;盘古大模型5.0将亮相;英伟达将收购 Run:ai

苹果首次发布开源语言模型 近期&#xff0c;苹果在 Hugging Face 发布了 OpenELM 系列模型。OpenELM 的关键创新是逐层扩展策略&#xff0c;该策略可在 transformer 模型的每一层中有效地分配参数&#xff0c;从而提高准确性。 与具有统一参数分配的传统语言模型不同&#xff…

【免费Java系列】给大家出一些JavaSE基础第八天的内容案例 , 让大家更好的理解与掌握

String字符串 案例一 求取字符串的长度 public class Main {public static void main(String[] args) {String str "Hello World";String substring str.substring(6);System.out.println("截取后的字符串为&#xff1a;" substring);} }输出结果&…