C++ 補充 & C++ 11 - 函數返回引用深入解析

函數返回引用深入解析

C++引用使用時的難點

1.當函數返回值爲引用時

若返回棧變量,不能成爲其它引用的初始值,不能作爲左值使用

2. 若返回靜態變量或全局變量

可以成爲其他引用的初始值
即可作爲右值使用,也可作爲左值使用

3. 返回形參當引用
(注:C++鏈式編程中,經常用到引用,運算符重載專題)

demo 代碼(一)

#include <iostream>
#include <stdlib.h>
#include <stdio.h>

using namespace std;

int& demo()
{
	int i = 666;
	i++;
	return i;
}

int main(void)
{
	int& i1 = demo(); /* 函數返回引用成爲其它引用的初始值 */
	printf("i1=%d \n", i1);

	system("pause");
	return 0;
}

編譯:

因爲 int i = 666 是臨時變量(執行後就會被釋放), 存放在棧上, 只是暫時這個棧沒有被調用
在這裏插入圖片描述

執行:

在這裏插入圖片描述

demo 代碼(二)

#include <iostream>
#include <stdlib.h>
#include <stdio.h>

using namespace std;

int& demo(int **addr)
{
	int i = 666;
	*addr = &i;
	printf("i 的地址: %p , i=%d\n", &i, i);

	return i;
}

int main(void)
{
	int* addr = NULL;
	int& i1 = demo(&addr); /* 函數返回引用成爲其它引用的初始值 */
	printf("addr: %p, i1=%d \n", addr, i1);


	system("pause");
	return 0;
}

編譯:

執行:

在這裏插入圖片描述

demo 代碼(三)

#include <iostream>
#include <stdlib.h>
#include <stdio.h>

using namespace std;

int& demo(int **addr)
{
	int i = 666;
	*addr = &i;
	printf("i 的地址: %p , i=%d\n", &i, i);

	return i;
}

int main(void)
{
	int* addr = NULL;
	int& i1 = demo(&addr); /* 函數返回引用成爲其它引用的初始值 */
	printf("addr: %p, i1=%d \n", addr, i1);
	i1 = 888;
	demo(&addr);
	printf("addr: %p, i1=%d \n", addr, i1);

	system("pause");
	return 0;
}

編譯:

執行:

在這裏插入圖片描述

demo 代碼(四)

#include <iostream>
#include <stdlib.h>
#include <stdio.h>

using namespace std;

int demo1()
{
	int i = 0;
	printf("i 的地址: %p, i=%d\n", &i, i);
	return i;
}

int& demo(int **addr)
{
	int i = 666;
	*addr = &i;
	printf("i 的地址: %p , i=%d\n", &i, i);
	return i;
}

int main(void)
{
	int* addr = NULL;
	int& i1 = demo(&addr); /* 函數返回引用成爲其它引用的初始值 */
	printf("addr: %p, i1=%d \n", addr, i1);
	i1 = 888;
	//demo(&addr);
	demo1();
	printf("addr: %p, i1=%d \n", addr, i1);

	system("pause");
	return 0;
}

編譯:

因爲臨時變量保存到-> 棧(調用函數或保存傳遞的形參或臨時變量) 再次調用demo1(), 會覆蓋臨時變量.

執行:

在這裏插入圖片描述

demo 代碼(五)

#include <iostream>
#include <stdlib.h>
#include <stdio.h>

using namespace std;

int demo1()
{
	int i = 0;
	printf("i 的地址: %p, i=%d\n", &i, i);
	return i;
}

int& demo(int **addr)
{
	int i = 666;
	*addr = &i;
	printf("i 的地址: %p , i=%d\n", &i, i);
	return i;
}

int main(void)
{
	int* addr = NULL;

	/* 第一種情況 函數返回局部變量引用不能成爲其他引用的初始值 */
	// int& i1 = demo(&addr);
	// i1 = 888;
	// printf("addr: %p, i1=%d \n", addr, i1);
	
	// demo(&addr);
	// demo1();
	// printf("addr: %p, i1=%d \n", addr, i1);

	/* 第二種情況 函數返回局部變量做爲左值 ? */
	demo(&addr) = 888;
	printf("1 addr: %p value: %d\n", addr, *addr);
	demo1();
	printf("2 addr: %p value: %d\n", addr, *addr);

	system("pause");
	return 0;
}

編譯:

執行:

在這裏插入圖片描述

demo 代碼(六)

#include <iostream>
#include <stdlib.h>
#include <stdio.h>

using namespace std;

int demo1()
{
	int i = 0;
	printf("i 的地址: %p, i=%d\n", &i, i);
	return i;
}

int& demo(int **addr)
{
	int i = 666;
	*addr = &i;
	printf("i 的地址: %p , i=%d\n", &i, i);
	return i;
}

int& demo_static(int** addr)
{
	static int i = 666;
	*addr = &i;
	printf("i 的地址: %p , i=%d\n", &i, i);
	return i;
}

int main(void)
{
	int* addr = NULL;

	/* 第一種情況 函數返回局部變量引用不能成爲其他引用的初始值 */
	// int& i1 = demo(&addr);
	// i1 = 888;
	// printf("addr: %p, i1=%d \n", addr, i1);
	
	// demo(&addr);
	// demo1();
	// printf("addr: %p, i1=%d \n", addr, i1);

	/* 第二種情況 函數返回局部變量做爲左值 ? */
	/*demo(&addr) = 888;
	printf("1 addr: %p value: %d\n", addr, *addr);
	demo1();
	printf("2 addr: %p value: %d\n", addr, *addr);*/

	/* 第三種情況 返回靜態變量或全局變量 */
	demo_static(&addr) = 888;
	printf("1 addr: %p value: %d\n", addr, *addr);
	demo1();
	printf("2 addr: %p value: %d\n", addr, *addr);


	system("pause");
	return 0;
}

編譯:

執行:

在這裏插入圖片描述

結語:

總的來說, 主要看這個變量的生存週期

時間: 2020-07-01

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章