关于c++中默认拷贝构造函数的理解

如题,请解释一次详细的原理
假如我定义一个类的对象a,然后在没有自己编写构造函数的情况下用语句A b=a来初始化新的对象b,会发生什么,为什么?

c++

4个回答

编译器会给你产生一个默认的拷贝构造函数,执行的是浅拷贝。所谓浅拷贝,就是直接字段的拷贝。如果对象包含一个指针,那么两个对象都指向同一个地址。

如果未提供自己的拷贝构造函数,则C++提供一个默认拷贝构造函数,就像没有提供构造函数时,C++提供默认构造函数一样。
C++提供的默认拷贝构造函数工作的方法是:完成一个成员一个成员的拷贝,如果成员是类对象,则调用其拷贝构造函数或者默认拷贝构造函数。
在默认拷贝构造函数中,拷贝的策略是逐个成员依次拷贝,但是,一个类可能会拥有资源,如果拷贝构造函数
简单地制作了一个该资源的拷贝,而不对它本身分配,就得面临一个麻烦的局面:两个对象都拥有同一个资源
。当对象析构时,该资源将经历两次资源返还(两次delete,导致出错)。

如果没有在类的定义中声明并实现自己的拷贝构造函数,编译器将会自动帮你生成一个浅拷贝的拷贝构造函数。
所谓的浅拷贝,就是将两个类中对应的元素相赋值。即将a中的成员一个一个对应地赋值给b中的成员,然后完成b的构建
如果类里面没有指针,浅拷贝和深拷贝产生的效果一样;
如果类里面有指针,浅拷贝就只会将a中指针的值赋值给b中的指针,那么就只是a与b中的指针指向同一个区域而已,
而我们的本意无疑是应该想要b中的指针指向另一个区域,这个区域中的内容和a中指向的区域的内容相等。
所以这个时候,就需要我们自己去写拷贝构造函数,完成深拷贝

如果你没有提供构造拷贝函数,编译器会为你自动合成一个

Csdn user default icon
上传中...
上传图片
插入图片
抄袭、复制答案,以达到刷声望分或其他目的的行为,在CSDN问答是严格禁止的,一经发现立刻封号。是时候展现真正的技术了!
其他相关推荐
关于c++中默认拷贝构造函数的理解
如题,请解释一次详细的原理 假如我定义一个类的对象a,然后在没有自己编写构造函数的情况下用语句A b=a来初始化新的对象b,会发生什么,为什么?
编译器合成默认拷贝构造函数问题
看《深度探索C++对象模型》遇到的问题。 P49里的例子对于bitwise copy semantics展现了memberwise的拷贝,但是后面P50下面说bitwise copy semantics不需要合成拷贝构造函数。 在P52页的两个例子中,第一个说没合成,我的理解也就是没有初始化int和char\*,但之后的第二个合成了拷贝构造函数,不仅初始化了string,更初始化了int. 我又用vs测试了下,发现可以运行 ``` C++ class B { public: int cc = 10; }; int main() { B b; cout << b.cc << endl; B c = b; cout << c.cc << endl; } ```
C++11的问题:类的默认移动构造函数会做什么?
类默认的移动构造函数是否是这样: * 如果是简单值、静态数组、指针,就复制 * 无视指针指向的内容 * 如果是直接的对象,就调用其移动构造函数 是这样吗? 是否意味着,假如类及其成员的类都未写移动构造函数,那么当需要移动的时候,实际效果与默认拷贝构造函数是一样的?
A a = A(); 会调用拷贝构造函数吗?
![图片说明](https://img-ask.csdn.net/upload/201902/17/1550376802_766935.png) 有如下代码,为什么运行结果只调用了默认构造函数却没有调用拷贝构造函数, A() 会生成一个临时地 A 对象, 将这个临时的对象作为 a 的初始值,不是应该调用拷贝构造函数吗?请指教 还有就是这种调用:![图片说明](https://img-ask.csdn.net/upload/201902/17/1550377780_14473.png) 为什么什么输出也没有?
构造函数的调用是拷贝还是赋值?
a代表默认构造函数,b代表拷贝构造函数,c代表赋值运算符,d代表析构函数 请问以下代码的调用顺序对应哪个选项? ``` A f(A& a){ return a; }  int main(int, char**){  A a;  A b = f(a);  return 0;  } ``` A.abdd B.acdd C.abbddd D.abcddd
关于等号重载中构造函数中形参默认函数的问题,其中不懂为什么
#define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std; //等号运算符的重载 class Array { public: //int strlen(char *) Array(char *t1=0) { this->length=strlen(t1); this->p = new char [length +1]; strcpy(p,t1); } Array(const Array &t2) { this->length=t2.length; this->p = new char [length +1]; strcpy(p,t2.p); } Array& operator=(Array &obj1) { //先释放旧的内存 if (this->p != NULL) { delete[] p; length = 0; } //2 根据obj1分配内存大小 this->length = obj1.length; this->p = new char [length+1]; //把obj1赋值 strcpy(p, obj1.p); return *this; } ~Array() { if(*p !=NULL) { delete [] p; length=0; } } void printf() { cout<<this->p; } private: int length; char *p; protected: }; int main() { Array a1("dasfsf"); Array a2=a1;//调用赋值构造函数,必须重新定义,要不然是浅拷贝,出错 a2.printf(); cout<<endl; Array a4("dasda"); Array a3; a3=a4;//浅拷贝,需重新定义等号。 a3.printf(); cout<<endl; system("pause"); return 0; } 源码如上,其中编译正确,却运行不了,如果把类中的构造函数的形参的默认赋值等于零,则正确,但是不知道为什么呢?
没有默认构造的函数怎么传参
``` #ifndef OBJ_H #define OBJ_H #include <iostream> #include <string> using namespace std; class obj { public: explicit obj(string st) :str(st) { } void print() { cout << str << endl; } private: string str; }; #endif ``` ``` #include"obj.h" class objcontainer { public: objcontainer(string str) :o(str){} objcontainer( obj ob) { } obj *operator->() { return &o; } private: obj o; }; ``` ``` objcontainer( obj ob) { o=ob; //提示obj没有默认构造的函数,这不是拷贝函数么? } ``` ``` objcontainer( obj ob):o(ob) { //这样又对了 } ```
C++程序 求代码!!!!!!急求
设计一个用于人事管理的“人员”类。由于考虑到通用性,这里只抽象出所有类型人员都具有的属性:编号,姓名,性别,出生日期。其中“出生日期”声明为一个“日期”类内嵌子对象。用成员函数实现对人员信息的录入和显示。 要求:包括构造函数和析构函数、拷贝构造函数、内联成员函数、带默认形参值的成员函数
关于C++ 函数参数为类对象时遇到的bug
``` #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> using namespace std; class Teacher { private: static int constructor_count_null; static int constructor_count_two; static int constructor_count_copy; static int constructor_count_valuation; static int destructor_count; char* m_name; int m_age; public: Teacher() { m_name = nullptr; m_age = 0; cout << "调用默认构造函数 " << ++constructor_count_null << "次" << endl; } Teacher(char*name,int age) { m_name = new char[strlen(name)]; strcpy(m_name, name); m_age = age; cout << "调用有参构造函数 " << ++constructor_count_two << "次" << endl; } //析构函数 ~Teacher() { delete []m_name; cout << "调用析构函数 " << ++destructor_count << "次" << endl; } //拷贝构造函数 Teacher(const Teacher& t) { this->m_age = t.m_age; this->m_name = new char[strlen(t.m_name)]; cout << &m_name << endl; strcpy(m_name, t.m_name); cout << "调用拷贝构造函数 " << ++constructor_count_copy << "次" << endl; } //重载赋值号 Teacher& operator =(const Teacher&t) { //首先检测是不是重复赋值 if(this!=&t) { //然后将原来的值delete if(this->m_name!=nullptr) { delete[]m_name; m_name = nullptr; } this->m_age = t.m_age; this->m_name = new char[strlen(t.m_name)]; strcpy(m_name, t.m_name); cout << "使用重载过的赋值号 " << ++constructor_count_valuation << "次" << endl; } return *this; } }; int Teacher::constructor_count_copy = 0; int Teacher::constructor_count_null = 0; int Teacher::constructor_count_two = 0; int Teacher::constructor_count_valuation = 0; int Teacher::destructor_count = 0; //这个函数我想传入一个类 void func(Teacher t) { } int main( _In_ int argc, _In_reads_(argc) _Pre_z_ char** argv, _In_z_ char** envp ) { Teacher s("sss", 22); func(s); system("pause"); return 0; } ``` 在debug模式下,会在func退出时出现bug,而在release中不会出现。 我的问题是: 为什么我明明做了深拷贝,debug模式下函数退出,析构匿名对象时会出错?
这个程序错在哪了??求大神帮忙改正
#include<iostream> #include<cmath> using namespace std; class Point { public: Point(int xx=0,int yy=0) { x=xx; y=yy; } Point(Point &p); int GetX() {return x;} int GetY() {return y;} private: int x,y; }; Point::Point(Point &p) { x=p.x; y=p.y; cout<<"实现点的拷贝构造函数"<<endl; } class Line { public: Line() { len=0; cout<<"Line默认构造函数调用"<<endl; } Line(Point &pp1,Point &pp2); Line(int x1,int y1,int x2,int y2); Line(Line &l1); int Getlen() {return len;}; ~Line() {} private: Point p1,p2; int len; }; Line::Line(int x1,int y1,int x2,int y2):p1(x1,y1),p2(x2,y2) { cout<<"构造了一条线"<<endl; int x=static_cast<int>(p1.GetX()-p2.GetX()); int y=static_cast<int>(p1.GetY()-p2.GetY()); len=sqrt(x*x+y*y); } Line::Line(Line &l1):p1(l1.p1),p2(l1.p2) { cout<<"实现线的拷贝构造函数"<<endl; len=l1.len; } int main() { Point p1(3,7),p2(9,4); Line line(p1,p2); Line line2(line); cout<<"The length of the line is:"; cout<<line.Getlen()<<endl; cout<<"The length of the line2 is:"; cout<<line2.Getlen()<<endl; return 0; }
C++中的debug error问题 困扰我两天了 大神快来
#include<iostream> #ifndef STRINGBAD_H_ #define STRINGBAD_H_ class StringBad { private: char * str; int len; static int num_strings; //不能在类声明中初始静态成员变量 这是因为声明描述如何分配内存 但并不分配内存 public: StringBad(const StringBad & st); //复制构造函数 StringBad(const char * s); //显示构造函数 StringBad(); //默认构造函数 ~StringBad(); //析构函数 friend std::ostream & operator<<(std::ostream & os, const StringBad & st); //友元函数 重载<<运算符 StringBad & operator=(const StringBad & st); //重载赋值运算符 }; #endif #include<string.h> #include"stringbad.h" using std::cout; using std::endl; int StringBad::num_strings = 0; //初始化类静态成员 静态类成员可以在类声明之外只用单独语句进行初始化 StringBad::StringBad(const StringBad & st) //赋值构造函数 进行深度复制 { num_strings++; //更新静态成员 const int b=len = st.len; //复制私有整型成员 str = new char[len + 1]; //把指针指向创建的新地址 strcpy_s(str, strlen(st.str)+1, st.str); //然后将字符串副本复制到新地址 cout << num_strings << ": \"" << str << "\" object created\n"; cout << &str << endl; } //浅复制导致的结果是把指针(指向同一内存字符串的地址)复制过去 当析构函数调用的时候将释放同一字符串 引起非常严重的后果 //所以 可能会被析构两次 应当使用上述的深度复制 而不是系统默认的 StringBad::StringBad(const char * s) { len = std::strlen(s); //检测计算字符串长度 但不包括末尾的空字符 并对len成员进行初始化 str = new char[len + 1]; //创建动态类成员 使用new分配足够的空间保存字符串 然后将新地址赋给str成员 strcpy_s(str, strlen(s) + 1, s); //复制字符串 将第二个字符串拷贝到第一个字符串的位置 num_strings++; cout << num_strings << ": \"" << str << "\" object created\n"; cout << &str << endl; } StringBad::StringBad() { len = 4; str = new char[4]; strcpy_s(str, 4, "C++"); num_strings++; cout << num_strings << ": \"" << str << "\" default object created\n"; cout << &str << endl; } StringBad::~StringBad() { cout << "\"" << str << "\" object deleted, "; --num_strings; cout << num_strings << " left\n"; delete[]str; cout << str << endl; } std::ostream & operator<<(std::ostream & os, const StringBad & st) { os << st.str; return os; } StringBad & StringBad::operator=(const StringBad & st) { if (this == &st) //检测是否自我复制 return *this; //是的话返回并结束 delete[]str; //否则释放str 释放目前占用的内存 因为 指针会等系会指向新内存 所以这段会被浪费掉 len = st.len; //赋值私有成员 无影响 str = new char[len + 1]; //创建新内存 并把新内存地址赋给指针str 因为之前的指向的旧地址已经被释放了 所以节省了内存 strcpy_s(str, strlen(st.str) + 1, st.str); //赋值字符串到新内存当中 return *this; //并返回对象 } #include<iostream> using std::cout; #include"stringbad.h" void callme1(StringBad &); void callme2(StringBad); int main() { using std::endl; { cout << "Starting an inner block.\n"; // StringBad headline1("Celery Stalks at Midnight"); // StringBad headline2("Lettuce Prey"); StringBad sports("Spinach Leaves Bowl for Dollars"); cout << "headline1: " << headline1 << endl; cout << "headline2: " << headline2 << endl; cout << "sports: " << sports << endl; callme1(headline1); cout << "headline1: " << headline1 << endl; callme2(headline2); cout << "headline2: " << headline2 << endl; cout << "Initialize one object to another:\n"; StringBad sailor = sports; cout << "sailor: " << sailor << endl; cout << "Assign one object to another:\n"; StringBad knot; knot = headline1; cout << "knot: " << knot << endl; cout << "Exiting the block.\n"; } cout << "End of main()\n"; return 0; } void callme1(StringBad & rsb) { cout << "String passed by reference:\n"; cout << " \"" << rsb << "\"\n"; } void callme2(StringBad sb) { cout << "String passwd by value:\n"; cout << " \"" << sb << "\"\n"; }![图片说明](https://img-ask.csdn.net/upload/201709/27/1506521094_491397.png)
Creat函数为什么会错呢?
循环链表ADT模板简单应用算法设计:约瑟夫环 作者: 冯向阳时间限制: 1S章节: DS:线性表 问题描述 : 目的:使用C++模板设计循环链表的抽象数据类型(ADT)。并在此基础上,使用循环链表ADT的基本操作,设计并实现单链表的简单算法设计。 内容:(1)请使用模板设计循环链表的抽象数据类型。(由于该环境目前仅支持单文件的编译,故将所有内容都集中在一个源文件内。在实际的设计中,推荐将抽象类及对应的派生类分别放在单独的头文件中。参考网盘中的单链表ADT原型文件,自行设计循环链表的ADT。) (2)ADT的简单应用:使用该ADT设计并实现循环链表应用场合的一些简单算法设计。 应用2:编号为1,2,...,n的n个人按顺时针方向围坐一圈,每人持有一个密码(正整数)。现在给定一个随机数m>0,从编号为1的人开始,按顺时针方向1开始顺序报数,报到m时停止。报m的人出圈,同时留下他的密码作为新的m值,从他在顺时针方向上的下一个人开始,重新从1开始报数,如此下去,直至所有的人全部出列为止。请在使用循环链表ADT的基础上,设计一个算法求出圈的顺序(以编号表示)。限定人数至少为1. 参考函数原型: template<class ElemType> void Joseph(CirLinkList<ElemType> &A, int m); //约瑟夫环专用结点类型 struct node{ int number; int code; }; 输入说明 : 第一行:人数n 第二行:第一个人所持的密码 第三行:第二个人所持的密码 ... 第n+1行:第n个人所持的密码 第n+2行:给定的随机数m 输出说明 : 第一行至第n行:建立的循环链表的遍历结果(一个结点占据1行) 第n+1行:空行 第n+2行:出圈的顺序(编号与编号之间以“->”分隔) #include <iostream> #include<stdio.h> #include<stdlib.h> using namespace std; /* 循环单链表的结点定义 */ template<class ElemType> struct LinkNode { ElemType data; int node; LinkNode<ElemType> *next; //构造函数1,用于构造头结点 LinkNode(LinkNode<ElemType> *ptr = NULL) { next = ptr; } //构造函数2,用于构造其他结点 //函数参数表中的形参允许有默认值,但是带默认值的参数需要放后面 LinkNode(const ElemType &item,const ElemType &itam, LinkNode<ElemType> *ptr = NULL) { next = ptr; data = item; node=itam; } //取得结点的数据域的值 ElemType getData(){ return data; } //修改结点的next域 void SetLink( LinkNode<ElemType> *link ){ next = link; } //修改结点的data域 void SetLink( ElemType value ){ data = value; } }; //带头结点的循环单链表 template<class ElemType> class CirLinkList { private: LinkNode<ElemType> *head; // 头结点 LinkNode<ElemType> *tail; // 尾结点 public: //无参数的构造函数 CirLinkList(){head = new LinkNode<ElemType>; tail = head; head->next = head;} //带参数的构造函数 CirLinkList(const ElemType &item){head = new LinkNode<ElemType>(item); tail = head; head->next = head;} //拷贝构造函数 CirLinkList(CirLinkList<ElemType> &List); //析构函数 //~CirLinkList(){ListDestroy();} //销毁链表 void ListDestroy(); //清空链表 void ListClear(); //返回链表的长度 int ListLength() { if(head->next==head) return 0; else {LinkNode<ElemType> *p; int i=1; p=head->next; while(p->next!=head->next) { p=p->next; i++; } return i;} } //判断链表是否为空表 bool ListEmpty() { if(head->next==head) return true; else return false; } //获取循环链表头结点 LinkNode<ElemType>* GetHead() { return head;} //获取循环链表尾结点 LinkNode<ElemType>* GetTail() { return tail;} //设置链表头结点 void SetHead(LinkNode<ElemType> *p){ head = p;} //在链表的第pos个位置之后插入e元素 void ListInsert_next(int pos,ElemType e) { int j; LinkNode<ElemType> *p,*s,*r; s=(LinkNode<ElemType>*)malloc(sizeof(ElemType)); p=head->next; j=1; while(j<pos&&p) { p=p->next; j++; } r=p; s->data=e; p->next=s; s->next=r->next; } //表头插入法动态生成链表 void CreateList_Head(int n) { ElemType w; LinkNode <ElemType> *h; LinkNode <ElemType> *o; h=head; for(int i=n;i>0;i--) { o=(LinkNode<ElemType>*)malloc(sizeof(ElemType)); cin>>w; o->data=w; o->node=n-i+1; h->next=o; tail=o; tail->next=head->next; h=tail; } } //表尾插入法动态生成链表 //void CreateList_Tail(int n, ElemType *A); //遍历链表 void ListTraverse() { LinkNode<ElemType> *p=head->next; while(p->next!=head->next) { cout<<p->node<<" "<<p->data<<endl; p=p->next; } cout<<p->node<<" "<<p->data<<endl; cout<<endl; return ; } void Joseph(int m,int cd) { int j=1,i; LinkNode<ElemType> *p,*s; p=head->next; s=p; for(i=1;i<cd;i++) { while(j!=m) { s=p; p=p->next; j++;} s->next=p->next; m=p->data; cout<<p->node<<"->"; p=s->next; j=1; } cout<<p->node; cout<<endl; } }; int main() { CirLinkList<int> d; int cd; cin>>cd; d.CreateList_Head(cd); int m; cin>>m; d.ListTraverse(); d.Joseph(m,cd); return 0; } 所设链表为带头指针的循环链表 即头指针指向第一个元素,尾指针也指向第一个元素 但在初始化时 头指针和尾指针相等。
、请设计一个用于保存和处理DNA序列的类DNASequence,该类具有以下特征和功能:
3、请设计一个用于保存和处理DNA序列的类DNASequence,该类具有以下特征和功能:  一个翻译表,定义为类属性,名为transcription_table,类型是字典,用于将DNA符号A、T、G、C分别转换为对应的符号,即A到U、T到A、G到C、C到G。  一个限制酶对照表,定义为类属性,名为enz_dict,类型是字典。所谓限制酶,指的是识别特定DNA序列并在识别区内产生截断的蛋白质。本题只关注两种限制酶,一个是’EcoRI’,识别’ GAATTC’序列,'EcoRV',识别'GATATC'序列。  构造函数__init__(self, seqstring)的参数seqstring为一个字符串,代表一个DNA序列,DNASequence有一个对象属性seqstring保存该字符串。要求将参数seqstring中所有字符转换为大写形式再保存到对象属性seqstring中。  有一个对象方法transcription(self),将对象属性seqstring保存的DNA序列,逐符号翻译为对应符号的DNA序列,如”ATG”翻译为”UAC”。  一个对象方法restriction(self, enz),enz为限制酶名称,类型为字符串。该方法的功能是统计对象属性seqstring中,所给限制酶对应的DNA序列的出现次数。如果不含限制酶对应的DNA序列,返回0。  重载len运算,即重新定义特殊函数__len__(self),用于返回对象属性seqstring的长度。  注意:本题不考虑__init__参数seqstring中字符是否合法,默认所给字符都是ATGC之一。 运行示例: >>> virus = DNASequence(’atggagagccttgttcttggtgtcaa’) >>> virus.seqstring ’ATGGAGAGCCTTGTTCTTGGTGTCAA’ >>> virus.transcription() ’ UACCUCUCGGAACAAGAACCACAGUU’ >>> other_virus = DNASequence(’atgatatcggagaggatatcggtgtcaa’) >>> other_virus.restriction(’EcoRV’) 2 >>>len(virus) 26 代码框架:请将该框架拷贝出去,保存在DNASequence.py文件中。如要测试,请在另外的文件里写测试程序。DNASequence.py只能包含DNASequence类的实现代码,不能有其他的测试代码。提交答题时,只提交DNASequence.py。如有其他代码影响了老师改卷,由此造成的扣分后果,由自己承担。 class DNASequence: transcription_table = {} #翻译表 enz_dict = {} #限制酶对照表 def __init__(self, seqstring): #请在下面编写程序 #请勿修改下面的程序 def __len__(self): # 请在下面编写程序 # 请勿修改下面的程序 def restriction(self, enz): # 请在下面编写程序 # 请勿修改下面的程序 def transcription(self): # 请在下面编写程序 # 请勿修改下面的程序
c++ 如何把这段代码类改为类模板
#include <iostream> using namespace std; class JuZhen { private : int Hang; //矩阵的行数 int Lie; //矩阵的列数 int **ptr; //用于动态分配矩阵空间所定义的指针 public: JuZhen(int H=0, int L=0); //构造函数,动态分配内存存储矩阵元素 JuZhen(const JuZhen &JZ); //拷贝构造函数 ~JuZhen(); //析构函数,注意释放内存 void InPut(); //输入矩阵元素 void Show(); //显示矩阵内容 JuZhen operator+(const JuZhen &JZ); //重载操作符 '+' 实现两个矩阵相加 JuZhen operator-(const JuZhen &JZ); //重载操作符 '-' 实现两个矩阵相加 JuZhen operator*(const JuZhen &JZ); //重载操作符 '*' 实现两个矩阵相乘 JuZhen& operator=(const JuZhen &JZ); //重载赋值操作符 '=' 实现矩阵之间的赋值 }; JuZhen::JuZhen(int H, int L) { Hang = H; Lie = L; ptr = new int*[Hang]; //分配Hang个int*空间 if(ptr == NULL) //判断空间分配是否成功 { cerr<<"分配空间失败!"<<endl; exit(EXIT_FAILURE); } for(int i=0; i<Hang; i++) { ptr[i] = new int[Lie];//让每一行的指针指向一个Lie大小的空间 if(ptr[i] == NULL) { cerr<<"分配空间失败!"<<endl; exit(EXIT_FAILURE); } } for(int i=0; i<Hang; i++)//将矩阵元素默认值设置为-1 for(int j=0; j<Lie; j++) ptr[i][j] = -1; } JuZhen::JuZhen(const JuZhen &JZ) { Hang = JZ.Hang; Lie = JZ.Lie; ptr = new int*[Hang]; //分配Hang个int*空间 if(ptr == NULL) //判断空间分配是否成功 { cerr<<"分配空间失败!"<<endl; exit(EXIT_FAILURE); } for(int i=0; i<Hang; i++) { ptr[i] = new int[Lie];//让每一行的指针指向一个Lie大小的空间 if(ptr[i] == NULL) { cerr<<"分配空间失败!"<<endl; exit(EXIT_FAILURE); } } for(int i=0; i<Hang; i++) //复制矩阵元素 for(int j=0; j<Lie; j++) ptr[i][j] = JZ.ptr[i][j]; } JuZhen::~JuZhen() { for(int i=0; i<Hang; i++) delete []ptr[i]; delete []ptr; } void JuZhen::InPut() { cout<<"这是一个 "<<Hang<<" 行 "<<Lie<<" 列的矩阵."<<endl; cout<<"请输入:"<<endl; for(int i=0; i<Hang; i++) for(int j=0; j<Lie; j++) { cout<<"第"<<i+1<<"行"<<j+1<<"列元素"<<"["<<i<<"]["<<j<<"]:"; cin>>ptr[i][j]; } } void JuZhen::Show() { for(int i=0; i<Hang; i++) { for(int j=0; j<Lie; j++) cout<<ptr[i][j]<<' '; cout<<endl; } } JuZhen JuZhen::operator +(const JuZhen &JZ) { if(Hang != JZ.Hang || Lie != JZ.Lie) { cout<<"两个矩阵大小不一致,加法操作失败"<<endl; return JuZhen(); } JuZhen temp(Hang,Lie);//创建临时对象存储两矩阵的和 for(int i=0; i<Hang; i++) for(int j=0; j<Lie; j++) temp.ptr[i][j] = ptr[i][j] + JZ.ptr[i][j]; return temp; } JuZhen JuZhen::operator -(const JuZhen &JZ) { if(Hang != JZ.Hang || Lie != JZ.Lie) { cout<<"两个矩阵大小不一致,减法操作失败"<<endl; return JuZhen(); } JuZhen temp(Hang,Lie);//创建临时对象存储两矩阵的差 for(int i=0; i<Hang; i++) for(int j=0; j<Lie; j++) temp.ptr[i][j] = ptr[i][j] - JZ.ptr[i][j]; return temp; } JuZhen JuZhen::operator *(const JuZhen &JZ) { if(Lie != JZ.Hang) { cout<<"两个矩阵行列不匹配,乘法操作失败"<<endl; return JuZhen(); } JuZhen temp(Hang,JZ.Lie);//创建临时对象存储两矩阵的积 for(int i=0; i<Hang; i++) for(int j=0; j<JZ.Lie; j++) { int sum = 0; for(int k=0; k<Lie; k++) sum += ptr[i][k] * JZ.ptr[k][j]; temp.ptr[i][j] = sum; } return temp; } JuZhen& JuZhen::operator =(const JuZhen &JZ) { if(this == &JZ) //防止自身赋值 return *this; for(int i=0; i<Hang; i++)//释放原矩阵空间 delete []ptr[i]; delete []ptr; Hang = JZ.Hang; Lie = JZ.Lie; ptr = new int*[Hang]; //分配Hang个int*空间 if(ptr == NULL) //判断空间分配是否成功 { cerr<<"分配空间失败!"<<endl; exit(EXIT_FAILURE); } for(int i=0; i<Hang; i++) { ptr[i] = new int[Lie];//让每一行的指针指向一个Lie大小的空间 if(ptr[i] == NULL) { cerr<<"分配空间失败!"<<endl; exit(EXIT_FAILURE); } } for(int i=0; i<Hang; i++) //复制矩阵元素 for(int j=0; j<Lie; j++) ptr[i][j] = JZ.ptr[i][j]; return *this; } int min(int x,int y) {return (x<y)?x:y;} int main() { int h1,l1,hl; char key; cout<<"请输入矩阵的行数与列数"<<endl; cin>>h1>>l1; hl=min(h1,l1); JuZhen JZ1(h1,l1); JuZhen JZ2(h1,l1); JuZhen JZ3(h1,l1); JuZhen JZ4(l1,h1); JuZhen JZ5(hl,hl); cout<<"--------------菜单--------------"<<endl; cout<<"请输入要执行的命令(+/-/*)"<<endl; cin>>key; switch (key) { case '+': cout<<"请为第一个矩阵输入元素:"<<endl; JZ1.InPut(); cout<<"请为第二个矩阵输入元素:"<<endl; JZ2.InPut(); cout<<"矩阵一:"<<endl; JZ1.Show(); cout<<"矩阵二:"<<endl; JZ2.Show(); cout<<"JZ1和JZ2的和是:"<<endl; JZ3 = JZ1 + JZ2; JZ3.Show(); break; case '-': cout<<"请为第一个矩阵输入元素:"<<endl; JZ1.InPut(); cout<<"请为第二个矩阵输入元素:"<<endl; JZ2.InPut(); cout<<"矩阵一:"<<endl; JZ1.Show(); cout<<"矩阵二:"<<endl; JZ2.Show(); cout<<"JZ1和JZ2的差是:"<<endl; JZ3 = JZ1 - JZ2; JZ3.Show(); break; case '*': cout<<"请为第一个矩阵输入元素:"<<endl; JZ1.InPut(); cout<<"请为第二个矩阵输入元素:"<<endl; JZ4.InPut(); cout<<"矩阵一:"<<endl; JZ1.Show(); cout<<"矩阵二:"<<endl; JZ4.Show(); cout<<"JZ1和JZ2的积是:"<<endl; JZ5 = JZ1 * JZ4; JZ5.Show(); break; default: break; } }
终于明白阿里百度这样的大公司,为什么面试经常拿ThreadLocal考验求职者了
点击上面↑「爱开发」关注我们每晚10点,捕获技术思考和创业资源洞察什么是ThreadLocalThreadLocal是一个本地线程副本变量工具类,各个线程都拥有一份线程私...
Java学习的正确打开方式
在博主认为,对于入门级学习java的最佳学习方法莫过于视频+博客+书籍+总结,前三者博主将淋漓尽致地挥毫于这篇博客文章中,至于总结在于个人,实际上越到后面你会发现学习的最好方式就是阅读参考官方文档其次就是国内的书籍,博客次之,这又是一个层次了,这里暂时不提后面再谈。博主将为各位入门java保驾护航,各位只管冲鸭!!!上天是公平的,只要不辜负时间,时间自然不会辜负你。 何谓学习?博主所理解的学习,它是一个过程,是一个不断累积、不断沉淀、不断总结、善于传达自己的个人见解以及乐于分享的过程。
程序员必须掌握的核心算法有哪些?
由于我之前一直强调数据结构以及算法学习的重要性,所以就有一些读者经常问我,数据结构与算法应该要学习到哪个程度呢?,说实话,这个问题我不知道要怎么回答你,主要取决于你想学习到哪些程度,不过针对这个问题,我稍微总结一下我学过的算法知识点,以及我觉得值得学习的算法。这些算法与数据结构的学习大多数是零散的,并没有一本把他们全部覆盖的书籍。下面是我觉得值得学习的一些算法以及数据结构,当然,我也会整理一些看过
大学四年自学走来,这些私藏的实用工具/学习网站我贡献出来了
大学四年,看课本是不可能一直看课本的了,对于学习,特别是自学,善于搜索网上的一些资源来辅助,还是非常有必要的,下面我就把这几年私藏的各种资源,网站贡献出来给你们。主要有:电子书搜索、实用工具、在线视频学习网站、非视频学习网站、软件下载、面试/求职必备网站。 注意:文中提到的所有资源,文末我都给你整理好了,你们只管拿去,如果觉得不错,转发、分享就是最大的支持了。 一、电子书搜索 对于大部分程序员...
《奇巧淫技》系列-python!!每天早上八点自动发送天气预报邮件到QQ邮箱
将代码部署服务器,每日早上定时获取到天气数据,并发送到邮箱。 也可以说是一个小人工智障。 思路可以运用在不同地方,主要介绍的是思路。
Python 植物大战僵尸代码实现(2):植物卡片选择和种植
这篇文章要介绍的是: - 上方植物卡片栏的实现。 - 点击植物卡片,鼠标切换为植物图片。 - 鼠标移动时,判断当前在哪个方格中,并显示半透明的植物作为提示。
死磕YOLO系列,YOLOv1 的大脑、躯干和手脚
YOLO 是我非常喜欢的目标检测算法,堪称工业级的目标检测,能够达到实时的要求,它帮我解决了许多实际问题。 这就是 YOLO 的目标检测效果。它定位了图像中物体的位置,当然,也能预测物体的类别。 之前我有写博文介绍过它,但是每次重新读它的论文,我都有新的收获,为此我准备写一个系列的文章来详尽分析它。这是第一篇,从它的起始 YOLOv1 讲起。 YOLOv1 的论文地址:https://www.c
知乎高赞:中国有什么拿得出手的开源软件产品?(整理自本人原创回答)
知乎高赞:中国有什么拿得出手的开源软件产品? 在知乎上,有个问题问“中国有什么拿得出手的开源软件产品(在 GitHub 等社区受欢迎度较好的)?” 事实上,还不少呢~ 本人于2019.7.6进行了较为全面的 回答 - Bravo Yeung,获得该问题下回答中得最高赞(236赞和1枚专业勋章),对这些受欢迎的 Github 开源项目分类整理如下: 分布式计算、云平台相关工具类 1.SkyWalk
记一次腾讯面试:进程之间究竟有哪些通信方式?如何通信? ---- 告别死记硬背
有一次面试的时候,被问到进程之间有哪些通信方式,不过由于之前没深入思考且整理过,说的并不好。想必大家也都知道进程有哪些通信方式,可是我猜很多人都是靠着”背“来记忆的,所以今天的这篇文章,讲给大家详细着讲解他们是如何通信的,让大家尽量能够理解他们之间的区别、优缺点等,这样的话,以后面试官让你举例子,你也能够顺手拈来。 1、管道 我们来看一条 Linux 的语句 netstat -tulnp | gr...
20行Python代码爬取王者荣耀全英雄皮肤
引言 王者荣耀大家都玩过吧,没玩过的也应该听说过,作为时下最火的手机MOBA游戏,咳咳,好像跑题了。我们今天的重点是爬取王者荣耀所有英雄的所有皮肤,而且仅仅使用20行Python代码即可完成。 准备工作 爬取皮肤本身并不难,难点在于分析,我们首先得得到皮肤图片的url地址,话不多说,我们马上来到王者荣耀的官网: 我们点击英雄资料,然后随意地选择一位英雄,接着F12打开调试台,找到英雄原皮肤的图片
网络(8)-HTTP、Socket、TCP、UDP的区别和联系
TCP/IP协议是传输层协议,主要解决数据如何在网络中传输,而HTTP是应用层协议,主要解决如何包装数据。 一、TCP与UDP的不同 1. 是否需要建立连接。 UDP在传送数据之前不需要先建立连接;TCP则提供面向连接的服务; 2. 是否需要给出确认 对方的传输层在收到UDP报文后,不需要给出任何确认,而 TCP需要给出确认报文,要提供可靠的、面向连接的传输服务。 3.虽然UDP不提供可靠交...
简明易理解的@SpringBootApplication注解源码解析(包含面试提问)
欢迎关注文章系列 ,关注我 《提升能力,涨薪可待》 《面试知识,工作可待》 《实战演练,拒绝996》 欢迎关注我博客,原创技术文章第一时间推出 也欢迎关注公 众 号【Ccww笔记】,同时推出 如果此文对你有帮助、喜欢的话,那就点个赞呗,点个关注呗! 《提升能力,涨薪可待篇》- @SpringBootApplication注解源码解析 一、@SpringBootApplication 的作用是什
防劝退!数据结构和算法难理解?可视化动画带你轻松透彻理解!
大家好,我是 Rocky0429,一个连数据结构和算法都不会的蒟蒻… 学过数据结构和算法的都知道这玩意儿不好学,没学过的经常听到这样的说法还没学就觉得难,其实难吗?真难! 难在哪呢?当年我还是个小蒟蒻,初学数据结构和算法的时候,在忍着枯燥看完定义原理,之后想实现的时候,觉得它们的过程真的是七拐八绕,及其难受。 在简单的链表、栈和队列这些我还能靠着在草稿上写写画画理解过程,但是到了数论、图...
西游记团队中如果需要裁掉一个人,会先裁掉谁?
2019年互联网寒冬,大批企业开始裁员,下图是网上流传的一张截图: 裁员不可避免,那如何才能做到不管大环境如何变化,自身不受影响呢? 我们先来看一个有意思的故事,如果西游记取经团队需要裁员一名,会裁掉谁呢,为什么? 西游记团队组成: 1.唐僧 作为团队teamleader,有很坚韧的品性和极高的原则性,不达目的不罢休,遇到任何问题,都没有退缩过,又很得上司支持和赏识(直接得到唐太宗的任命,既给
开挂的人生!那些当选院士,又是ACM/IEEE 双料Fellow的华人学者们
昨日,2019年两院院士正式官宣,一时间抢占了各大媒体头条。 朋友圈也是一片沸腾,奔走相告,赶脚比自己中了大奖还嗨皮! 谁叫咱家导师就是这么厉害呢!!! 而就在最近,新一年度的IEEE/ACM Fellow也将正式公布。 作为学术届的顶级荣誉,不自然地就会将院士与Fellow作比较,到底哪个含金量更高呢? 学术君认为,同样是专业机构对学者的认可,考量标准不一,自然不能一概而论。 但...
聊聊C语言和指针的本质
坐着绿皮车上海到杭州,24块钱,很宽敞,在火车上非正式地聊几句。 很多编程语言都以 “没有指针” 作为自己的优势来宣传,然而,对于C语言,指针却是与生俱来的。 那么,什么是指针,为什么大家都想避开指针。 很简单, 指针就是地址,当一个地址作为一个变量存在时,它就被叫做指针,该变量的类型,自然就是指针类型。 指针的作用就是,给出一个指针,取出该指针指向地址处的值。为了理解本质,我们从计算机模型说起...
Python语言高频重点汇总
Python语言高频重点汇总 GitHub面试宝典仓库——点这里跳转 文章目录Python语言高频重点汇总**GitHub面试宝典仓库——点这里跳转**1. 函数-传参2. 元类3. @staticmethod和@classmethod两个装饰器4. 类属性和实例属性5. Python的自省6. 列表、集合、字典推导式7. Python中单下划线和双下划线8. 格式化字符串中的%和format9.
究竟你适不适合买Mac?
我清晰的记得,刚买的macbook pro回到家,开机后第一件事情,就是上了淘宝网,花了500元钱,找了一个上门维修电脑的师傅,上门给我装了一个windows系统。。。。。。 表砍我。。。 当时买mac的初衷,只是想要个固态硬盘的笔记本,用来运行一些复杂的扑克软件。而看了当时所有的SSD笔记本后,最终决定,还是买个好(xiong)看(da)的。 已经有好几个朋友问我mba怎么样了,所以今天尽量客观...
代码详解:如何用Python快速制作美观、炫酷且有深度的图表
全文共12231字,预计学习时长35分钟生活阶梯(幸福指数)与人均GDP(金钱)正相关的正则图本文将探讨三种用Python可视化数据的不同方法。以可视化《2019年世界幸福报告》的数据为例,本文用Gapminder和Wikipedia的信息丰富了《世界幸福报告》数据,以探索新的数据关系和可视化方法。《世界幸福报告》试图回答世界范围内影响幸福的因素。报告根据对“坎特里尔阶梯问题”的回答来确定幸...
程序员一般通过什么途径接私活?
二哥,你好,我想知道一般程序猿都如何接私活,我也想接,能告诉我一些方法吗? 上面是一个读者“烦不烦”问我的一个问题。其实不止是“烦不烦”,还有很多读者问过我类似这样的问题。 我接的私活不算多,挣到的钱也没有多少,加起来不到 20W。说实话,这个数目说出来我是有点心虚的,毕竟太少了,大家轻喷。但我想,恰好配得上“一般程序员”这个称号啊。毕竟苍蝇再小也是肉,我也算是有经验的人了。 唾弃接私活、做外...
(经验分享)作为一名普通本科计算机专业学生,我大学四年到底走了多少弯路
今年正式步入了大四,离毕业也只剩半年多的时间,回想一下大学四年,感觉自己走了不少弯路,今天就来分享一下自己大学的学习经历,也希望其他人能不要走我走错的路。 (一)初进校园 刚进入大学的时候自己完全就相信了高中老师的话:“进入大学你们就轻松了”。因此在大一的时候自己学习的激情早就被抛地一干二净,每天不是在寝室里玩游戏就是出门游玩,不过好在自己大学时买的第一台笔记本性能并不是很好,也没让我彻底沉...
如何写一篇技术博客,谈谈我的看法
前言 只有光头才能变强。 文本已收录至我的GitHub精选文章,欢迎Star:https://github.com/ZhongFuCheng3y/3y 我一直推崇学技术可以写技术博客去沉淀自己的知识,因为知识点实在是太多太多了,通过自己的博客可以帮助自己快速回顾自己学过的东西。 我最开始的时候也是只记笔记,认为自己能看得懂就好。但如果想验证自己是不是懂了,可以写成技术博客。在写技术博客的...
字节跳动面试官这样问消息队列:分布式事务、重复消费、顺序消费,我整理了一下
你知道的越多,你不知道的越多 点赞再看,养成习惯 GitHub上已经开源 https://github.com/JavaFamily 有一线大厂面试点脑图、个人联系方式和人才交流群,欢迎Star和完善 前言 消息队列在互联网技术存储方面使用如此广泛,几乎所有的后端技术面试官都要在消息队列的使用和原理方面对小伙伴们进行360°的刁难。 作为一个在互联网公司面一次拿一次Offer的面霸...
面试还搞不懂redis,快看看这40道面试题(含答案和思维导图)
Redis 面试题 1、什么是 Redis?. 2、Redis 的数据类型? 3、使用 Redis 有哪些好处? 4、Redis 相比 Memcached 有哪些优势? 5、Memcache 与 Redis 的区别都有哪些? 6、Redis 是单进程单线程的? 7、一个字符串类型的值能存储最大容量是多少? 8、Redis 的持久化机制是什么?各自的优缺点? 9、Redis 常见性...
大学四年自学走来,这些珍藏的「实用工具/学习网站」我全贡献出来了
知乎高赞:文中列举了互联网一线大厂程序员都在用的工具集合,涉及面非常广,小白和老手都可以进来看看,或许有新收获。
互联网公司的裁员,能玩出多少种花样?
裁员,也是一门学问,可谓博大精深!以下,是互联网公司的裁员的多种方法:-正文开始-135岁+不予续签的理由:千禧一代网感更强。95后不予通过试用期的理由:已婚已育员工更有责任心。2通知接下来要过苦日子,让一部分不肯同甘共苦的员工自己走人,以“兄弟”和“非兄弟”来区别员工。3强制996。员工如果平衡不了工作和家庭,可在离婚或离职里二选一。4不布置任何工作,但下班前必须提交千字工作日报。5不给活干+...
【设计模式】单例模式的八种写法分析
网上泛滥流传单例模式的写法种类,有说7种的,也有说6种的,当然也不排除说5种的,他们说的有错吗?其实没有对与错,刨根问底,写法终究是写法,其本质精髓大体一致!因此完全没必要去追究写法的多少,有这个时间还不如跟着宜春去网吧偷耳机、去田里抓青蛙得了,一天天的....
《面试宝典》:检验是否为合格的初中级程序员的面试知识点,你都知道了吗?查漏补缺
欢迎关注文章系列,一起学习 《提升能力,涨薪可待篇》 《面试知识,工作可待篇》 《实战演练,拒绝996篇》 也欢迎关注公 众 号【Ccww笔记】,原创技术文章第一时间推出 如果此文对你有帮助、喜欢的话,那就点个赞呗,点个关注呗! 《面试知识,工作可待篇》-Java笔试面试基础知识大全 前言 是不是感觉找工作面试是那么难呢? 在找工作面试应在学习的基础进行总结面试知识点,工作也指日可待,欢...
关于研发效能提升的思考
研发效能提升是最近比较热门的一个话题,本人根据这几年的工作心得,做了一些思考总结,由于个人深度有限,暂且抛转引入。 三要素 任何生产力的提升都离不开这三个因素:人、流程和工具,少了其中任何一个因素都无法实现。 人,即思想,也就是古人说的“道”,道不同不相为谋,是制高点,也是高层建筑的基石。 流程,即方法,也是古人说的“法”。研发效能的提升,也就是要提高投入产出比,既要增加产出,也要减...
微博推荐算法简述
在介绍微博推荐算法之前,我们先聊一聊推荐系统和推荐算法。有这样一些问题:推荐系统适用哪些场景?用来解决什么问题、具有怎样的价值?效果如何衡量? 推荐系统诞生很早,但真正被大家所重视,缘起于以”facebook”为代表的社会化网络的兴起和以“淘宝“为代表的电商的繁荣,”选择“的时代已经来临,信息和物品的极大丰富,让用户如浩瀚宇宙中的小点,无所适从。推荐系统迎来爆发的机会,变得离用户更近: 快...
GitHub 标星 1.6w+,我发现了一个宝藏项目,作为编程新手有福了!
大家好,我是 Rocky0429,一个最近老在 GitHub 上闲逛的蒟蒻… 特别惭愧的是,虽然我很早就知道 GitHub,但是学会逛 GitHub 的时间特别晚。当时一方面是因为菜,看着这种全是英文的东西难受,不知道该怎么去玩,另一方面是一直在搞 ACM,没有做一些工程类的项目,所以想当然的以为和 GitHub 也没什么关系(当然这种想法是错误的)。 后来自己花了一个星期看完了 Pyt...
Python爬虫爬取淘宝,京东商品信息
小编是一个理科生,不善长说一些废话。简单介绍下原理然后直接上代码。 使用的工具(Python+pycharm2019.3+selenium+xpath+chromedriver)其中要使用pycharm也可以私聊我selenium是一个框架可以通过pip下载 pip installselenium -ihttps://pypi.tuna.tsinghua.edu.cn/simple/ ...
阿里程序员写了一个新手都写不出的低级bug,被骂惨了。
这种新手都不会范的错,居然被一个工作好几年的小伙子写出来,差点被当场开除了。
Java工作4年来应聘要16K最后没要,细节如下。。。
前奏: 今天2B哥和大家分享一位前几天面试的一位应聘者,工作4年26岁,统招本科。 以下就是他的简历和面试情况。 基本情况: 专业技能: 1、&nbsp;熟悉Sping了解SpringMVC、SpringBoot、Mybatis等框架、了解SpringCloud微服务 2、&nbsp;熟悉常用项目管理工具:SVN、GIT、MAVEN、Jenkins 3、&nbsp;熟悉Nginx、tomca...
恕我直言,牛逼哄哄的MongoDB你可能只会30%
MongoDB 闪亮登场自我介绍MongoDB 是一个基于分布式文件存储的数据库。由 C++ 语言编写。旨在为 WEB 应用提供可扩展的高性能数据存储解决方案。MongoDB 是一个介于...
一文带你看清 HTTP 所有概念
上一篇文章我们大致讲解了一下 HTTP 的基本特征和使用,大家反响很不错,那么本篇文章我们就来深究一下 HTTP 的特性。我们接着上篇文章没有说完的 HTTP 标头继续来介绍(此篇文章会介绍所有标头的概念,但没有深入底层) HTTP 标头 先来回顾一下 HTTP1.1 标头都有哪几种 HTTP 1.1 的标头主要分为四种,通用标头、实体标头、请求标头、响应标头,现在我们来对这几种标头进行介绍 通用...
相关热词 c#开发的dll注册 c#的反射 c# grid绑定数据源 c#多线程怎么循环 c# 鼠标左键 c# char占位符 c# 日期比较 c#16进制转换为int c#用递归求顺序表中最大 c#小型erp源代码
立即提问