隨筆
C++博客
首頁
新隨筆
聯系
聚合
管理
posts - 6, comments - 30, trackbacks - 0
關于Visual C++2010編譯器的問題
今天使用
visual
C++ 2010
編譯
器編寫
程序
逆波蘭表達式求值
但是遭遇了以前從未看到的問題,在此
求助
各位,謝謝
頭
文件
Stack.h
1
template
<
class
T
>
2
class
Stack
3
{
4
public
:
5
Stack(
int
size
=
0
);
//
構造大小為size的堆棧
6
~
Stack()
{delete p;}
//
撤銷堆棧
7
bool
IsFull();
//
判斷堆棧是否已滿
8
bool
IsEmpty();
//
判斷堆棧是否為空
9
void
Push(T
&
x);
//
將x壓入堆棧
10
T Pop(T
&
x);
//
將棧頂元素彈出用x保存
11
private
:
12
int
Size;
//
大小
13
int
top;
//
棧頂
14
T
*
p;
//
數組指針
15
}
;
實現源文件Stack.cpp
1
#include
"
Stack.h
"
2
3
template
<
class
T
>
4
void
Stack
<
T
>
::Push(T
&
x)
5
{
6
p[top
++
]
=
x;
//
將x保存于棧頂,棧頂+1
7
}
8
9
template
<
class
T
>
10
T Stack
<
T
>
::Pop(T
&
x)
11
{
12
x
=
p[
--
top];
//
x保存棧頂元素,棧頂-1
13
return
x;
14
}
15
16
template
<
class
T
>
17
Stack
<
T
>
::Stack(
int
size)
18
{
19
Size
=
size;
//
棧大小為size
20
top
=
0
;
//
棧頂top指向0
21
p
=
new
T[Size];
//
為p開辟Size大小的空間
22
}
23
24
template
<
class
T
>
25
bool
Stack
<
T
>
::IsFull()
26
{
27
return
top
==
Size;
28
}
//
判斷堆棧是否已滿
29
30
template
<
class
T
>
31
bool
Stack
<
T
>
::IsEmpty()
32
{
33
return
top
==
0
;
34
}
//
判斷堆棧是否為空
main.cpp
1
#include
<
iostream
>
2
#include
<
stdio.h
>
3
#include
"
Stack.h
"
4
const
int
MaxSize
=
40
;
//
由于定義數組時開辟空間大小
5
using
namespace
std;
6
7
/**/
/*
8
*逆波蘭表達式求值函數
9
*/
10
float
LiBolan(
char
*
array)
{
//
array存儲表達式
11
Stack
<
float
>
sta(MaxSize);
//
建立大小為Maxsize的堆棧
12
int
i
=
0
;
13
if
(array[
0
]
==
'
.
'
)
{
14
cout
<<
"
表達式錯誤:小數表示錯誤
"
<<
endl;
15
return
0
;
16
}
17
while
(array[i]
!=
'
\0
'
)
{
//
當表達式沒有結束
18
if
(array[i]
==
'
.
'
)
{
//
數字第一個字符為小數點或運算符后是小數點
19
cout
<<
"
表達式錯誤:小數表示錯誤
"
<<
endl;
20
return
0
;
21
}
22
if
(i
==
0
&&
array[i]
==
'
-
'
)
{
//
判斷第一個是否負數,后面的沒有必要判斷,如果是
23
i
++
;
//
處理下一個字符
24
float
tem
=
0
;
25
while
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
)
{
26
tem
=
(
float
)array[i]
-
48
+
tem
*
10
;
//
先計算該負數的絕對值
27
i
++
;
28
}
29
if
(array[i]
==
'
.
'
)
{
//
若存在小數
30
i
++
;
//
忽略該位
31
if
(
!
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
))
{
//
小數點后不是數字
32
cout
<<
"
表達式錯誤:小數表示錯誤
"
<<
endl;
33
return
0
;
34
}
35
}
36
Stack
<
char
>
s(
7
);
//
用于存儲小數字符
37
while
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
)
{
38
s.Push(array[i]);
39
i
++
;
40
}
41
float
item
=
0
;
42
char
x;
43
while
(
!
s.IsEmpty())
44
item
=
((
float
)s.Pop(x)
-
48
)
*
0.1
+
item
*
0.1
;
//
計算小數
45
tem
+=
item;
46
tem
=-
tem;
//
取相反數得到原數
47
sta.Push(tem);
//
將該數壓入棧中
48
}
49
if
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
)
{
//
其他時候,判斷是否為數字
50
float
tem
=
0
;
51
while
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
)
{
//
計算整數部分
52
tem
=
(
float
)array[i]
-
48
+
tem
*
10
;
53
i
++
;
54
}
55
if
(array[i]
==
'
.
'
)
{
//
若存在小數
56
i
++
;
//
忽略該位
57
if
(
!
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
))
{
//
小數點后不是數字
58
cout
<<
"
表達式錯誤:小數表示錯誤
"
<<
endl;
59
return
0
;
60
}
61
}
62
Stack
<
char
>
s(
7
);
//
用于存儲小數字符
63
while
(array[i]
>=
'
0
'
&&
array[i]
<=
'
9
'
)
{
64
s.Push(array[i]);
65
i
++
;
66
}
67
float
item
=
0
;
68
char
x;
69
while
(
!
s.IsEmpty())
70
item
=
((
float
)s.Pop(x)
-
48
)
*
0.1
+
item
*
0.1
;
//
計算小數
71
tem
+=
item;
72
sta.Push(tem);
73
}
74
if
(array[i]
==
'
+
'
||
array[i]
==
'
-
'
||
array[i]
==
'
*
'
||
array[i]
==
'
/
'
)
{
75
float
it1,it21,it22;
//
it21棧頂元素,it22倒數第二個,it1是it21,it22對應運算的值
76
char
ch
=
array[i];
77
switch
(ch)
{
78
case
'
+
'
:it21
=
sta.Pop(it21);
79
if
(sta.IsEmpty())
{
80
cout
<<
"
表達是錯誤:運算符比對應所需的運算數多
"
<<
endl;
//
彈出一個運算數后為空
81
return
0
;
82
}
83
it22
=
sta.Pop(it22);
84
it1
=
it21
+
it22;
85
sta.Push(it1);
break
;
86
case
'
-
'
:it21
=
sta.Pop(it21);
87
if
(sta.IsEmpty())
{
88
cout
<<
"
表達式錯誤:運算符比對應所需的運算數多
"
<<
endl;
89
return
0
;
90
}
91
it22
=
sta.Pop(it22);
92
it1
=
it22
-
it21;
93
sta.Push(it1);
break
;
94
case
'
*
'
:it21
=
sta.Pop(it21);
95
if
(sta.IsEmpty())
{
96
cout
<<
"
表達式錯誤:運算符比對應所需的運算數多
"
<<
endl;
97
return
0
;
98
}
99
it22
=
sta.Pop(it22);
100
it1
=
it21
*
it22;
101
sta.Push(it1);
break
;
102
case
'
/
'
:it21
=
sta.Pop(it21);
103
if
(sta.IsEmpty())
{
104
cout
<<
"
表達式錯誤:運算符比對應所需的運算數多
"
<<
endl;
105
return
0
;
106
}
107
it22
=
sta.Pop(it22);
108
it1
=
it22
/
it21;
109
sta.Push(it1);
break
;
110
default
:
break
;
111
}
112
i
++
;
113
}
114
else
115
i
++
;
116
}
117
float
value;
118
sta.Pop(value);
119
if
(
!
sta.IsEmpty())
{
120
cout
<<
"
表達式錯誤:運算數多于所需的運算符
"
<<
endl;
//
最后棧不為空
121
return
0
;
122
}
123
return
value;
124
}
125
void
main()
{
126
printf(
"
請輸入一個后綴表達式:
"
);
127
char
str[MaxSize];
128
gets(str);
129
float
value
=
LiBolan(str);
130
printf(
"
%2.2f\n
"
,value);
131
}
當用Visual C++ 6.0編譯時正確,用visualC++201編譯時,如果不分頭文件源文件,即直接把這三個文件放在man.cpp中編譯正確,當時當把他們分開后
其文件結構圖如下時編譯通不過
其問題描述如下
1>t.obj : error LNK2019: 無法解析的外部符號 "public: bool __thiscall Stack<float>::IsEmpty(void)" (
?IsEmpty@?$Stack@M@@QAE_NXZ
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: float __thiscall Stack<float>::Pop(float &)" (
?Pop@?$Stack@M@@QAEMAAM@Z
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: void __thiscall Stack<float>::Push(float &)" (
?Push@?$Stack@M@@QAEXAAM@Z
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: char __thiscall Stack<char>::Pop(char &)" (
?Pop@?$Stack@D@@QAEDAAD@Z
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: bool __thiscall Stack<char>::IsEmpty(void)" (
?IsEmpty@?$Stack@D@@QAE_NXZ
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: void __thiscall Stack<char>::Push(char &)" (
?Push@?$Stack@D@@QAEXAAD@Z
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: __thiscall Stack<char>::Stack<char>(int)" (
??0?$Stack@D@@QAE@H@Z
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>t.obj : error LNK2019: 無法解析的外部符號 "public: __thiscall Stack<float>::Stack<float>(int)" (
??0?$Stack@M@@QAE@H@Z
),該符號在函數 "float __cdecl LiBolan(char *)" (
?LiBolan@@YAMPAD@Z
) 中被引用
1>E:\Visual2010\t\Debug\t.exe : fatal error LNK1120: 8 個無法解析的外部命令
1>
1>生成失敗。
1>
1>已用時間 00:00:00.79
========== 生成: 成功 0 個,失敗 1 個,最新 0 個,跳過 0 個 ==========
posted on 2011-01-19 14:22
あ維wêiセ
閱讀(2762)
評論(12)
編輯
收藏
引用
所屬分類:
C++
FeedBack:
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 14:42 |
coolypf
Stack.cpp的內容全部放到Stack.h里面。
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 14:45 |
あ維wêiセ
@coolypf
但是按照程序編寫習慣,不是應該把他們分開嗎?
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 14:49 |
ChowZenki
如果不想寫在hpp中你可以去掉template
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 14:54 |
あ維wêiセ
去掉template后那個T就是一個未知符號了@ChowZenki
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 14:59 |
ChowZenki
@あ維wêiセ
實現寫在hpp中是template的使用約定...
具體去看看stl相關的書吧
如果要用就必須這么做
否則你只能改寫成非模板的類了
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 15:06 |
あ維wêiセ
@ChowZenki
額,是不是用了template就必須把.cpp與.h文件寫在一起呀?
謝謝你為我解決
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題[未登錄]
2011-01-19 20:12 |
codejie
建議不要C和C++混用。。。下面這個風格非常不好。。
1#include<iostream>
2#include<stdio.h>
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 20:14 |
あ維wêiセ
@codejie
關鍵是要用的gets()函數
用while(...)
{
.....
}老是不能結束
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-19 22:51 |
jmchxy
template的參數需要在使用時候特化,你不放到包含文件中,調用的文件不知道怎么特化。
template函數或類是在被調用到的時候才生成真正的函數體或類,如果調用者看不到模板的實現就不能生成真正的函數體或類
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-23 08:28 |
coderman
對于模板,編譯時和普通代碼的編譯是有區別的,普通代碼在編譯時只需要知道函數定義即可編譯通過,但模板不一樣,模板編譯有兩個階段:
1、先編譯類模板本身,看是否有語法錯誤,例如是不是少寫了分號
2、當模板實例化時再進行一次編譯,即編譯實例化的后的代碼,檢查你指定的類型能不能在模板里被支持,這時編譯器需要清楚的知道模板的具體實現,所以常規的做法是將聲明和實現都放在一個.h文件里
你可以看一下開源的代碼(例如 boost庫的 asio),里面的模板都放.h文件里
感覺樓主的代碼在安全上有點問題,作為一個stack,對它的調用者來說應該是安全的,你的代碼里應該檢查下標是否越界,指針是否為空,刪除指針時應該判空并置NULL
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-01-28 12:37 |
summerheart
vc6.共認的對標準C++支持不夠好,大概也就60%上下吧。
而Vc2010對標準C++的支持更完全。這也是新版本的亮點之一。
所以 樓主還是按標準的C++模板編寫方法寫吧。
回復
更多評論
#
re: 關于Visual C++2010編譯器的問題
2011-02-27 21:28 |
marmot
樓上說的比較對,寫模板最好用包含模型
用分離模型編譯容易出問題
Boost中很多子庫都采用的是包含模型
回復
更多評論
刷新評論列表
只有注冊用戶
登錄
后才能發表評論。
【推薦】100%開源!大型工業跨平臺軟件C++源碼提供,建模,組態!
相關文章:
關于除以2的n次方求余數的思考
中綴表達式轉為后綴表達式
關于Visual C++2010編譯器的問題
有關二叉查找樹的一些功能(BST)
鏈表實驗
網站導航:
博客園
IT新聞
BlogJava
博問
Chat2DB
管理
Copyright ©2025 あ維wêiセ Powered By:
博客園
模板提供:
滬江博客
<
2025年5月
>
日
一
二
三
四
五
六
27
28
29
30
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
1
2
3
4
5
6
7
常用鏈接
我的隨筆
我的評論
我參與的隨筆
留言簿
(1)
給我留言
查看公開留言
查看私人留言
隨筆分類
C++(5)
Java(1)
隨筆檔案
2011年2月 (1)
2011年1月 (5)
文章分類
個人總結(1)
文章檔案
2011年1月 (1)
搜索
最新評論
1.?re: 中綴表達式轉為后綴表達式
你可以考慮先將表達式中的數字字符轉換成對應的數字,成為一個元素,那么打印時加空格就好@js
--yangwei
2.?re: 中綴表達式轉為后綴表達式
是的,要犧牲一個來區分隊空和隊滿@jhayumu
--yangwei
3.?re: 中綴表達式轉為后綴表達式
(8+9*10)-4/2+3
與
(8+91*0)-4/2+3
怎么區分?
--js
4.?re: 關于除以2的n次方求余數的思考[未登錄]
好像法
--zhang
5.?re: 關于Visual C++2010編譯器的問題
樓上說的比較對,寫模板最好用包含模型
用分離模型編譯容易出問題
Boost中很多子庫都采用的是包含模型
--marmot
閱讀排行榜
1.?中綴表達式轉為后綴表達式(6031)
2.?關于除以2的n次方求余數的思考(3939)
3.?關于Visual C++2010編譯器的問題(2762)
4.?鏈表實驗(1953)
5.?有關二叉查找樹的一些功能(BST)(1922)
評論排行榜
1.?關于Visual C++2010編譯器的問題(12)
2.?中綴表達式轉為后綴表達式(5)
3.?鏈表實驗(5)
4.?關于除以2的n次方求余數的思考(5)
5.?初學java(一)(3)
国内高清久久久久久
|
97超级碰碰碰久久久久
|
日产精品久久久一区二区
|
观看 国产综合久久久久鬼色 欧美 亚洲 一区二区
|
亚洲精品WWW久久久久久
|
国产精品久久久久久一区二区三区
|
久久国产色AV免费观看
|
99久久国产精品免费一区二区
|
亚洲精品国精品久久99热一
|
久久亚洲高清综合
|
久久精品一区二区影院
|
国产亚洲美女精品久久久
|
亚洲欧洲中文日韩久久AV乱码
|
久久人人爽人人人人爽AV
|
久久国产精品99精品国产987
|
亚洲va久久久噜噜噜久久男同
|
久久精品亚洲欧美日韩久久
|
久久综合久久美利坚合众国
|
国产精品久久久久久久久软件
|
久久久久亚洲精品天堂
|
欧美日韩精品久久久免费观看
|
国产精品久久久99
|
精品久久久久久无码免费
|
久久精品国产免费观看三人同眠
|
狠狠色丁香婷综合久久
|
久久免费视频1
|
久久夜色精品国产网站
|
久久乐国产精品亚洲综合
|
国产婷婷成人久久Av免费高清
|
色综合久久精品中文字幕首页
|
99久久精品国产一区二区三区
|
国产精品成人久久久久三级午夜电影
|
国产精品美女久久久久
|
亚洲欧美日韩精品久久亚洲区
|
久久免费高清视频
|
久久久久久亚洲Av无码精品专口
|
一本色综合久久
|
久久综合成人网
|
精品视频久久久久
|
亚洲精品国产成人99久久
|
99久久国语露脸精品国产
|