haskell 基础题解(33)

本文介绍了一种将人民币金额从数字形式转换为汉字大写形式的方法,特别关注于零的正确处理,确保转换准确无误。文章提供了详细的转换规则,并通过Haskell代码实现了这一功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

人民币金额大写

【题目】在打印凭据,签定合同等场合,经常需要把人民币金额用汉字大写。这个需求十分常见,十分重要。但细节多,很容易出错。
请你把一个整数表示的人民币金额转为金额大写
特别要注意,金额中,的出现,既不能多,也不能少。
例如:
1000000001 -> 壹拾亿零壹
1250030001 -> 壹拾贰亿伍仟零叁万零壹
30050 -> 叁万零伍拾

总体的大规则上并不难。
西方按3位一组命名: thousand,million, billion 等
中文则4位一组命名:万,亿,万亿,亿亿,万亿亿,亿亿亿 等
如果某组为0,它的单位不读出,
中间多个连续零读为一个
结尾不能有零。。。。
细节很多

上个 haskell 码

import Data.List (dropWhileEnd)

f合并连续零 :: String -> String
f合并连续零 = foldr f [] 
    where 
    f x [] = [x]
    f x (y:ys) | x=='零' && y=='零' = y:ys
               | otherwise          = x:y:ys  

f拆组 :: Integer -> Integer -> [Integer] 
f拆组 0 _ = []
f拆组 x n = x `mod` n : f拆组 (x `div` n) n

f千内 :: Integer -> String
f千内 x = 
    let 
        u1 = ["零","壹","贰","叁","肆","伍","陆","柒","捌","玖"]
        u2 = ["","拾","佰","仟"]
        f加单位 0 _ = "零"
        f加单位 n u = u1 !! fromIntegral n ++ u
        t  = zipWith f加单位 (f拆组 x 10) u2 
        t1 = concat $ reverse t
        t2 = dropWhileEnd (=='零') t1  ---去尾零
        t3 = f合并连续零 t2
    in 
         if null t3 then "零" else t3

fRMB :: Integer -> String
fRMB x = 
    let 
        u = "" : [y : replicate (i `div` 2) '亿' | 
                        i<-[0..], 
                        let y = if even i then '万' else '亿']
        f加单位 n u | n == 0    = "零"
                    | n < 1000  = "零" ++ f千内 n ++ u
                    | otherwise = f千内 n ++ u
        t = zipWith f加单位 (f拆组 x 10000) u
        t1 = concat $ reverse t
        t2 = dropWhileEnd (=='零') t1  ---去尾零
        t3 = dropWhile (=='零') t2 ---去首零
        t4 = f合并连续零 t3
    in 
         if null t4 then "零" else t4  

main :: IO ()
main = do
    putStrLn $ fRMB 0
    putStrLn $ fRMB 2
    putStrLn $ fRMB 10
    putStrLn $ fRMB 23
    putStrLn $ fRMB 100
    putStrLn $ fRMB 108
    putStrLn $ fRMB 120
    putStrLn $ fRMB 456
    putStrLn $ fRMB 2000
    putStrLn $ fRMB 2001
    putStrLn $ fRMB 2100
    putStrLn $ fRMB 2040
    putStrLn $ fRMB 12040
    putStrLn $ fRMB 30000
    putStrLn $ fRMB 30001
    putStrLn $ fRMB 30050
    putStrLn $ fRMB 31000
    putStrLn $ fRMB 31009
    putStrLn $ fRMB 1000030001
    putStrLn $ fRMB 1000000001
    putStrLn $ fRMB 1000000070
    putStrLn $ fRMB 1200030001
    putStrLn $ fRMB 1250030001

看着挺长的。一大半都是测试,还是觉得不够。。。
(请帮忙发现反例)

多数码是处理细节的。没什么可说。 u = … 构建万级单位的那个地方,是把:
万,亿,万,亿,… 加到 空,空,亿,亿,亿亿,亿亿,亿亿亿,亿亿亿,…
的前边,写的时候就觉得别扭,直觉上应该有更简单的描述方式,但还没找到。。。。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值