operator[],识别读操作和写操作
body, table{font-family: 微软雅黑; font-size: 10pt}
table{border-collapse: collapse; border: solid gray; border-width: 2px 0 2px 0;}
th{border: 1px solid gray; padding: 4px; background-color: #DDD;}
td{border: 1px solid gray; padding: 4px;}
tr:nth-child(2n){background-color: #f8f8f8;}
#include<iostream>
#include<string.h>
using namespace std;
class String
{ //23种设计模式之代理模式
class charProxy;
public:
String():_pstr(new char[2])
{
cout<<"String()"<<endl;
initRefcount();
}
String(const char *);
String(const String&);
String& operator=(const String&);
~String()
{
cout<<"~String()"<<endl;
decreaseRefcount();
if(0==useCount())
{
delete []_pstr;
}
}
int useCount()
{
return _pstr[strlen(_pstr)+1];
}
charProxy operator[](int index); //区分写操作,这里要返回一个临时对象就不能是一个引用了
// char& operator[](int idx)const; //区分读操作,必须加const关键字,不然和前面函数构成重载
private:
char* _pstr;
void initRefcount()
{
_pstr[strlen(_pstr)+1] = 1;
}
void increaseRefcount()
{
++_pstr[strlen(_pstr)+1];
}
void decreaseRefcount()
{
--_pstr[strlen(_pstr)+1];
}
class charProxy
{
public: //这些函数一定要是公有
charProxy(String& str,int index):_str(str),_index(index)
{
cout<<"charProxy(String&,int)"<<endl;
}
char& operator=(char ch); //嵌套类里面,也不能返回对象引用;因为嵌套类对象都是临时的
operator char()
{ //类型转换函数,由Charproxy转向char
cout<<"operator char()"<<endl;
return _str._pstr[_index];
}
private:
String& _str;
int _index;
};
friend ostream& operator<<(ostream&,const String&);
friend istream& operator>>(istream&,String&);
};
#if 0
//operator[]()不能区分读操作还是写操作
char & String::operator[](int idx)const //不能区分读写操作,废弃
{
cout<<"operator [](int)const"<<endl;
return _pstr[idx];
}
#endif
String::String(const char* tmp)
{
cout<<"String(const char*)"<<endl;
_pstr = new char[strlen(tmp)+2];
strcpy(_pstr,tmp);
initRefcount(); //引用计数设置为1
}
String::String(const String& lhs)
{
cout<<"String(const String&)"<<endl;
_pstr = lhs._pstr;
increaseRefcount();
}
String& String::operator=(const String& lhs)
{
cout<<"String& operator=(const String&)"<<endl;
decreaseRefcount();
if(0==useCount())
{
delete []_pstr;
}
_pstr = lhs._pstr; // 浅拷贝
increaseRefcount();
return *this;
}
ostream& operator<<(ostream& os,const String& rhs)
{
cout<<"ostream& operator<<(ostream& os,const String&)"<<endl;
os<<rhs._pstr<<endl;
return os;
}
istream& operator>>(istream& is,String& s)
{
cout<<"istream& operator>>(istream&,String&)"<<endl;
is>>s._pstr;
return is;
}
|
String::charProxy String::operator[](int index)
{
cout<<"charProxy operator[](int)"<<endl;
return charProxy(*this,index); //返回临时嵌套类对象
}
char& String::charProxy::operator=(char ch)
{
cout<<"char& operator=(char)"<<endl;
if(_index>=0&&_index<strlen(_str._pstr))
{
if(1==_str.useCount()) //当对象没有共享内存时
{
_str._pstr[_index] = ch; //修改
return _str._pstr[_index];
}
else
{
_str.decreaseRefcount(); //有共享内存
char* _strtmp = new char[strlen(_str._pstr)+2];
strcpy(_strtmp,_str._pstr);
_str = _strtmp; //这里调用了构造函数和赋值构造函数,操作完之后调用析构函数
_str[_index] = ch; //这里再次调动[]重载函数,进行修改
_str.initRefcount();
return _str._pstr[_index];
}
}
}
int main()
{
String s1("hello,world");
String s2(s1);
cout<<"读操作:"<<s1[1]<<endl;
cout<<s1.useCount()<<endl;
cout<<"--------------------------"<<endl;
cout<<"写操作:"<<endl;
s1[1] = 'm';
cout<<"--------------------------"<<endl;
cout<<s1[1]<<endl;
cout<<s1.useCount()<<endl;
return 0;
}
//String(const char*)
//String(const String&)
//charProxy operator[](int)
//charProxy(String&,int)
//operator char()
//读操作:e
//2
//--------------------------
//写操作:
//charProxy operator[](int)
//charProxy(String&,int)
//char& operator=(char)
//String(const char*)
//String& operator=(const String&)
//~String()
//charProxy operator[](int)
//charProxy(String&,int)
//char& operator=(char)
//--------------------------
//charProxy operator[](int)
//charProxy(String&,int)
//operator char()
//m
//1
//~String()
//~String()
|
这里要识别[]操作只能借助嵌套类来底层实现,因为不可能在String类中重写=运算符;所以当遇到[]操作符的时候就转化成CharProxy的对象,必须传引用对其本身操作,在这个类中就可以重写=运算符来区分读和写,因为这里的=我们后面不会再做其他用处只是区分读写;所以这里分三步: 重写[]运算符 先转换为嵌套类对象 重写=,区分s[]和是s[]= 改: 读: 读的时候cout不能输出CharProxy对象,除非重载;采用类型转换比较方便,会根据需要自动转换 |
operator[],识别读操作和写操作的更多相关文章
- (MariaDB/MySQL)MyISAM存储引擎读、写操作的优先级
MariaDB/MySQL中使用表级锁的存储引擎(例如MyISAM.Aria(MariaDB对MyISAM引擎的改进,前身是MyISAM))在读(select).写操作(insert.delete.u ...
- flash多进程写操作
1 应用场景介绍 硬件条件:使用stm32 MCU 软件条件:协议栈应用 协议栈简单介绍如下: 类似于OSI七层模型,所涉及的协议栈包括应用层,网络层,链路层,物理层,如下图: 在 ...
- C#多线程:使用ReaderWriterLock类实现多用户读/单用户写同步
摘要:C#提供了System.Threading.ReaderWriterLock类以适应多用户读/单用户写的场景.该类可实现以下功能:如果资源未被写操作锁定,那么任何线程都可对该资源进行读操作锁定, ...
- 使用ReaderWriterLock类实现多用户读/单用户写同步
使用ReaderWriterLock类实现多用户读/单用户写同步[1] 2015-03-12 应用程序在访问资源时是进行读操作,写操作相对较少.为解决这一问题,C#提供了System.Threadin ...
- HDFS的工作原理(读和写操作)
工作原理: NameNode和DateNode,NameNode相当于一个管理者,它管理集群内的DataNode,当客户发送请求过来后,NameNode会 根据情况指定存储到哪些DataNode上,而 ...
- CI中获取读操作的结果集行数+获取写操作的影响行数
本质:读操作,用mysql_num_rows函数,写操作用mysql_affected_rows函数 mysql_num_rows() 返回结果集中行的数目.此命令仅对 SELECT 语句有效.要取得 ...
- 汇编指令-MRS(读)和MSR(写)指令操作CPSR寄存器和SPSR寄存器使用(1)
1.MSR和MRS指令介绍 MRS 指令: 对状态寄存器CPSR和SPSR进行读操作.通过读CPSR可以获得当前处理器的工作状态.读SPSR寄存器可以获得进入异常前的处理器状态(因为只有异常模式下有 ...
- python中文件的读和写操作
一.打开文件 data = open("yesterday",encoding="utf-8").read() # python默认的打字符编码是unicode ...
- 基于.net EF6 MVC5+WEB Api 的Web系统框架总结(4)-Excel文件读、写操作
Excel文件读.写可以使用Office自带的库(Microsoft.Office.Interop.Excel),前提是本机须安装office才能运行,且不同的office版本之间可能会有兼容问题.还 ...
随机推荐
- oracle修改某字段不是必输性
执行sql: ALTER TABLE table_name MODIFY 要修改的字段名字 NULL;
- 【c++ primer, 5e】【try语句块】
p172~p177:c++的try语句块和异常处理: 1.通常,与用户交互的代码和对象相加(底层的代码)是分离开的,异常由与用户交互的代码处理(底层代码抛出异常就可以了). 2.C++的runtime ...
- Linux学习笔记之CentOS6.9 防火墙的关闭以及开启
有的时候,我们需要对系统的防火墙进行操作,今天小编就给大家讲解一下如何开启以及关闭CentOS6.9系统下的防火墙. 输入:cat /etc/issue 查看版本 (一)通过service命令 s ...
- CSS Margin(外边距)
CSS Margin(外边距) 一.简介 CSS margin(外边距)属性定义元素周围的空间. margin 清除周围的(外边框)元素区域.margin 没有背景颜色,是完全透明的. margin ...
- clientWidth offsetWidth scrollWidth
网页可见区域宽: document.body.clientWidth;网页可见区域高: document.body.clientHeight;网页可见区域宽: document.body.offset ...
- 20145307第四次JAVA学习实验报告
20145307实验四 Android开发基础 实验内容 基于Android Studio开发简单的 Android应用并部署测试; 了解Android组件.布局管理器的使用: 掌握Android中事 ...
- VMware 共享目录不显示的解决办法
centos mount -t vmhgfs .host:/ /mnt/hgfs # 如果有如下提示: # Error: cannot mount filesystem: No such device ...
- yarn命令使用
yarn 常用命令 修改日期 2017.12.26 最初接触 yarn 还是在 0.17.10 版本,由于各种各样的原因,使用时没 npm 顺手, 目前 yarn 的版本已经升级为 1.3.2 各种之 ...
- Hibernate的懒加载session丢失解决方法
在web.xml加入spring提供的过滤器,延长session的生命周期 <!--Hibernate的懒加载session丢失解决方法 --> <filter> <fi ...
- IntelliJ IDEA 开发git多模块项目
1.clone主项目 填写主仓库地址 2.在项目根目录,初始化子模块,并clone源码 git submodule init git submodule update 3.定位到各个子模块根目录,并切 ...