[ThinkingDog]是一個積極向上、樂觀、熱心的人。
沉思的狗の博客
[ThinkingDog]歡迎您的光臨,請多多指教!
正整數中數字1的計數問題 [C++]
/**/
/*
*******************************************************************
purpose:
在從1到n的正數中1出現的次數
題目:輸入一個整數n,求從1到n這n個整數的十進制表示中1出現的次數。
例如輸入12,從1到12這些整數中包含1 的數字有1,10,11和12,1一共出現了5次。
求滿足f(i)=i(i<=911111111099999009)這樣的數總計有多少個
********************************************************************
*/
#include
<
iostream
>
#include
<
string
>
#include
<
time.h
>
using
namespace
std;
int
len;
int
*
perDigit;
//
每位上的數字
long
long
*
fullOne;
//
位數是n位的數,含1的總個數為fullOne[n]
long
long
*
addTopOne;
//
位數為n的數,首位為1時有多少個這樣的數addTopOne[n] 即10^(n-1)
int
curDigit;
long
long
cnt
=
0
;
long
long
fixBit1
=
0
;
int
ocuur1Cnt
=
0
;
long
long
getOneCnt(
long
long
n)
{
len
=
0
;
while
(n
>
0
)
{
//
把數字n分成十進制串對應的數字數組
perDigit[len]
=
n
%
10
;
n
=
n
/
10
;
len
++
;
}
cnt
=
0
;
fixBit1
=
0
;
//
數字n中1所在的位被小于此位的數字重復的次數
ocuur1Cnt
=
0
;
//
數字n有幾位是1
for
(
int
i
=
len
-
1
; i
>=
0
; i
--
)
{
//
從十進制數高位到低位
curDigit
=
perDigit[i];
if
(ocuur1Cnt
>
0
)
{
fixBit1
=
fixBit1
*
10
+
ocuur1Cnt
*
curDigit;
}
if
(curDigit
>
0
)
{
if
(curDigit
==
1
)
{
cnt
+=
curDigit
*
fullOne[i];
ocuur1Cnt
++
;
}
else
{
cnt
+=
curDigit
*
fullOne[i]
+
addTopOne[i];
}
}
}
return
cnt
+
fixBit1
+
ocuur1Cnt;
}
void
HowManyOne(
long
long
topNum)
{
clock_t start, finish;
//
記錄計算開始結束時間
start
=
clock();
len
=
20
;
//
最長20位十進制數
perDigit
=
new
int
[len];
fullOne
=
new
long
long
[len];
addTopOne
=
new
long
long
[len];
fullOne[
0
]
=
0
;
addTopOne[
0
]
=
1
;
cnt
=
1
;
for
(
int
i
=
1
; i
<
len; i
++
)
{
//
初始化信息
fullOne[i]
=
fullOne[i
-
1
]
*
10
+
cnt;
cnt
*=
10
;
addTopOne[i]
=
cnt;
}
long
long
stack[
1000
];
//
存儲數據段, [from, to]及計算方向,每次分別存入from,to,dir
long
long
lRel[
1000
];
//
符合f(i)==i表達式的i的數組
int
pStack
=
0
, pRel
=
0
;
//
stack與lRel當前長度或下一次存儲位置或棧頂
long
long
from, to, dir;
//
當前要驗證的一段數據的始終與驗證方向,驗證方向為0x01(向上) 0x10(向下) 0x11(向上向下均可以)
long
long
fn, dist;
//
fn當前一個數字n對應的1到n所有數字的十進制中的1的總個數;dist臨時變量(from與to的差)
stack[
0
]
=
1
;
stack[
1
]
=
topNum;
stack[
2
]
=
3
;
//
0x11
pStack
=
3
;
int
maxP
=
0
;
while
(pStack
>
0
)
{
//
從stack中取出一段數據,驗證其中的i是否滿足f(i)==i
dir
=
stack[
--
pStack];
to
=
stack[
--
pStack];
from
=
stack[
--
pStack];
if
((dir
&
0x01
)
!=
0
)
{
//
UP 從from開始向to的方向計算 f(i)==i
while
(from
<=
to)
{
fn
=
getOneCnt(from);
if
(fn
>
from)
{
from
=
fn;
}
else
if
(fn
<
from)
{
from
++
;
break
;
}
else
{
lRel[pRel
++
]
=
fn;
from
++
;
}
}
}
if
((dir
&
0x10
)
!=
0
)
{
//
down 從to開始向from的方向計算 f(i)==i
while
(from
<=
to)
{
fn
=
getOneCnt(to);
if
(fn
<
to)
{
to
=
fn;
}
else
if
(fn
>
to)
{
to
--
;
break
;
}
else
{
lRel[pRel
++
]
=
fn;
to
--
;
}
}
}
if
(to
-
from
<
2
)
{
//
這一段己經很小,直接計算完
for
(;from
<=
to;from
++
)
{
if
(from
==
getOneCnt(from))
{
lRel[pRel
++
]
=
fn;
}
}
}
else
{
//
當前段向上向下己計算完,二分后入棧,再計算
dist
=
(to
-
from)
>>
1
;
dist
=
from
+
dist;
stack[pStack
++
]
=
from;
stack[pStack
++
]
=
dist;
stack[pStack
++
]
=
0x10
;
stack[pStack
++
]
=
dist
+
1
;
stack[pStack
++
]
=
to;
stack[pStack
++
]
=
0x01
;
}
}
finish
=
clock();
int
i
=
pRel;
while
(i
>
0
)
//
輸出符合f(i)==i的所有i
cout
<<
lRel[
--
i]
<<
endl;
cout
<<
"
time:
"
<<
((
double
)(finish
-
start))
<<
"
ms
"
<<
endl;
cout
<<
"
total:
"
<<
pRel
<<
endl;
}
void
Test_HowManyOne()
{
HowManyOne(911111111099999009LL);
}
輸出為:
199981
199982
199983
199984
199985
199986
199987
199990
199989
199988
200000
200001
1599981
1599982
1599983
1599984
1599985
1599986
1599987
1599988
1599990
1599989
2600000
2600001
13199998
35199981
35199982
35199983
35199984
35199985
35199986
35199987
35199990
35199989
35199988
35000001
35000000
35200000
35200001
117463825
500199981
500199982
500199983
500199984
500199985
500199986
500199987
500199990
500199989
500199988
500200000
500200001
501599981
501599982
501599983
501599984
501599985
501599986
501599987
501599988
501599990
501599989
502600000
502600001
513199998
535199981
535199982
535199983
535199984
535199985
535199986
535199987
535199990
535199989
535199988
535000001
535000000
500000001
500000000
535200000
535200001
1111111110
1
time: 0ms
total: 83
發表于 2011-05-25 17:51
沉思的狗
閱讀(1277)
評論(3)
編輯
收藏
評論
#
re: 正整數中數字1的計數問題 [C++]
借鑒了
http://blog.csdn.net/ljsspace/archive/2011/05/22/6437981.aspx
http://topic.csdn.net/u/20110523/12/12a128ad-8a0a-4eb9-b4a1-9cda06f23e39.html?seed=1136147581&r=73511306#r_73511306
冰封空間
評論于 2011-05-25 22:10
回復
更多評論
#
re: 正整數中數字1的計數問題 [C++]
樓主為什么沒有main函數呢?
sf
評論于 2011-05-26 15:39
回復
更多評論
#
re: 正整數中數字1的計數問題 [C++]
@sf
void Test_HowManyOne()
這個函數改成main就可以了
冰封空間
評論于 2011-05-27 09:18
回復
更多評論
新用戶注冊
刷新評論列表
只有注冊用戶
登錄
后才能發表評論。
網站導航:
博客園
IT新聞
Chat2DB
C++博客
博問
管理
<
2011年5月
>
日
一
二
三
四
五
六
24
25
26
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
導航
BlogJava
首頁
發新隨筆
發新文章
聯系
聚合
管理
統計
隨筆: 115
文章: 1
評論: 86
引用: 0
常用鏈接
我的隨筆
我的文章
我的評論
我的參與
最新評論
留言簿
(5)
給我留言
查看公開留言
查看私人留言
隨筆檔案
(115)
2015年1月 (1)
2011年5月 (12)
2011年4月 (2)
2010年9月 (2)
2010年8月 (4)
2009年9月 (1)
2009年6月 (1)
2009年3月 (1)
2008年6月 (1)
2008年1月 (2)
2007年7月 (2)
2007年6月 (2)
2007年5月 (4)
2007年4月 (1)
2007年1月 (1)
2006年12月 (1)
2006年11月 (2)
2006年10月 (2)
2006年9月 (3)
2006年8月 (6)
2006年7月 (1)
2006年6月 (2)
2006年5月 (10)
2006年4月 (50)
2006年3月 (1)
網址
http://blog.csdn.net/Unagain
v_JULY_v
搜索
積分與排名
積分 - 210835
排名 - 266
最新評論
1.?re: 使用Policy文件來設置Java的安全策略[未登錄]
ss
--啊啊
2.?re: Jni中C++和Java的參數傳遞
老大,Long 是J啊,不是L啊,可害苦我了,趕緊改回來吧;
--cnhua5
3.?re: Jni中C++和Java的參數傳遞
樓主,在jni里返回String和C++里獲取的為什么不一樣,比如在java里看到的值是57891234,在C++里顯示的是5789@,這是為什么?。?
--chr
4.?re: 螺旋數字與坐標
對我的項目很有幫助。
謝謝
--cs221313
5.?re: Jni中C++和Java的參數傳遞
long的符號表寫錯了,作為初學者亞歷山大啊
--hhhhhh
閱讀排行榜
1.?Jni中C++和Java的參數傳遞 (63524)
2.?本地計算機上的 MSSQLSERVER 服務啟動后又停止了。一些服務自動停止,如果它們沒有什么可做的,例如“性能日志和警報”服務。[用批處理解決](22460)
3.?使用Policy文件來設置Java的安全策略(10507)
4.?一個簡單的十六進制計算器(出自Win程序設計)(8747)
5.?VC++6.0 全部默認快捷鍵(6212)
評論排行榜
1.?Upload Server (HTTP 上傳服務JAVA程序) 速度極快(11)
2.?Jni中C++和Java的參數傳遞 (10)
3.?垃圾軟件反刪除批處理文件 (7)
4.?剛寫的八皇后問題 - 遞歸 (隨便你定義幾個皇后了)JAVA(4)
5.?火車運煤問題(4)
Powered by:
博客園
模板提供:
滬江博客
Copyright ©2025 沉思的狗
[ThinkingDog]是一個積極向上、樂觀、熱心的人。
主站蜘蛛池模板:
免费在线一级毛片
|
亚洲一卡二卡三卡四卡无卡麻豆
|
在线毛片片免费观看
|
91亚洲国产成人精品下载
|
成**人免费一级毛片
|
成在线人视频免费视频
|
亚洲精品国产专区91在线
|
国产免费无遮挡精品视频
|
两个人看的www免费视频中文
|
亚洲日本香蕉视频
|
亚洲AV日韩精品一区二区三区
|
免费A级毛片无码视频
|
色屁屁www影院免费观看视频
|
亚洲综合小说久久另类区
|
全免费一级毛片在线播放
|
两个人看的www高清免费观看
|
亚洲一久久久久久久久
|
久久精品国产亚洲沈樵
|
免费的涩涩视频在线播放
|
精品国产麻豆免费人成网站
|
精品无码专区亚洲
|
亚洲美女激情视频
|
久久亚洲AV永久无码精品
|
女人让男人免费桶爽30分钟
|
91福利免费视频
|
中出五十路免费视频
|
曰批免费视频播放在线看片二
|
一级做α爱过程免费视频
|
亚洲大尺码专区影院
|
亚洲成色www久久网站夜月
|
又粗又大又硬又爽的免费视频
|
免费观看美女用震蛋喷水的视频
|
一区二区免费国产在线观看
|
亚洲欧美日本韩国
|
亚洲人成黄网在线观看
|
亚洲高清视频在线观看
|
亚洲综合伊人久久大杳蕉
|
免费欧洲美女牲交视频
|
啦啦啦www免费视频
|
成人免费福利电影
|
野花高清在线电影观看免费视频
|