周末了,分享个有趣的编程问题。。。

C语言与CPP编程 2024-06-02 10:00

击上方“C语言与CPP编程”,选择“关注/置顶/星标公众号

干货福利,第一时间送达!

大家好,我是飞宇,周末快乐,分享个有趣的编程问题



这就是一个求阶乘的问题,大家刚刚开始学编程的时候应该都写过这样的程序。

一个求阶乘的问题,还能玩出什么样的花儿来?

我在回答区看到了一个非常有趣的回答,把各种版本的代码都举例了出来,一个比一个还离谱,后面的代码,甚至让怀疑是我学过的代码吗?


分享给大家图个乐,下面是回答正文,答主:小白白。


数学家版本:

(简单利索,深藏功与名)

#include 
#include 
int main()
{
    std::cout << std::tgamma(20 + 1) << std::endl;
}

语言学家版本:

(语言学家,你懂得,恨不得把所有语法特性都派上用场)

#include 
#include 

template<std::size_t...I> constexpr auto foo(std::index_sequence) return ((I+1) * ...); }

int main() 
{
    std::cout << foo(std::make_index_sequence<20>()) << std::endl;
}

历史学家版本:

(void main() 有没有嗅到浓厚的历史气息?)

#include 
void main(void) {
    int i;
    long long j;
    for(i = 1, j = 1;i <= 20; j *= i++);
        printf("%lld", j); 
}

敏捷开发上线1.0版本:

(可以说是非常敏捷了)

#include 
int main() {
    //printf("%d", 1*2*3*4*5*6*7*8*9*10);
    printf("%lld", (long long)1*2*3*4*5*6*7*8*9*10*11*12*13*14*15*16*17*18*19*20);
}

面向对象专家版本:

(好家伙,一个简单的问题,愣是祭出了接口、继承、虚函数、虚继承、智能指针等一大堆东西出来,这很面向对象)

#include 
#include 
#include 

struct IBaseInterface {
    virtual ~IBaseInterface() = 0;
};
inline IBaseInterface::~IBaseInterface() = default;

struct IDataProvider : virtual public IBaseInterface {
    virtual int first() = 0;
    virtual int last() = 0;
    virtual int next(int v) = 0;
};

struct ICalculator : virtual public IBaseInterface {
    virtual long long calc(IDataProvider *) = 0;
};

struct IPrinter : virtual public IBaseInterface {
    virtual void print(const std::string &) = 0;
};

struct ISerializer : virtual public IBaseInterface {
    virtual std::string serialize(long long value) = 0;
};

struct IRunnable : virtual public IBaseInterface {
    virtual void run() = 0;
};

class Foo : virtual public IRunnable {
    std::shared_ptr m_dp;
    std::shared_ptr m_c;
    std::shared_ptr m_s;
    std::shared_ptr m_p;
public:
    Foo(std::shared_ptr dp, std::shared_ptr c, std::shared_ptr s, std::shared_ptr p) : m_dp(std::move(dp)), m_c(std::move(c)), m_s(std::move(s)),m_p(std::move(p)) {}
    void run() override { return  m_p->print(m_s->serialize(m_c->calc(m_dp.get()))); }
};

class DefaultSerializer : virtual public ISerializer {
public:
    std::string serialize(long long value) override { return std::to_string(value); }
};

class StreamPrinter : virtual public IPrinter {
    std::ostream &m_os;
public:
    explicit StreamPrinter (std::ostream &os) : m_os(os) {}
    void print(const std::string &s)  override { m_os << s << std::endl; }
};

class MultiplyAccumulateCalculator : virtual public ICalculator {
public:
    long long calc(IDataProvider *dp) override {
        int i = dp->first();
        long long j = i;
        do
            j *= (i = dp->next(i));
        while(i != dp->last());
        return j;
    }
};

int main() {
    struct MyDataProvider : virtual public IDataProvider {
        int first() override { return 1; }
        int last() override { return 20; }
        int next(int v) override { return v+1; }
    };
    Foo foo(std::make_shared(), std::make_shared(), std::make_shared(), std::make_shared(std::cout));
    foo.run();
}

提前优化的并行版本:

(一看就是精通底层技术的大佬,把CPU拿捏得死死的)

#include 
#include 

double foo(int x) {
    __m128 a = {1.0f2.0f3.0f4.0f};
    __m128 b = {4.0f4.0f4.0f4.0f};
    __m128 c = {1.0f1.0f1.0f1.0f};
    for(int i = 0; i < x / 4; ++i, a = _mm_add_ps(a, b))
        c = _mm_mul_ps(c, a);
    for(int i = x % 4; i < 4; ++i)
        a[i] = 1.0f;
    c = _mm_mul_ps(c, a);
    return (double)c[0] * (double)c[1] * (double)c[2] * (double)c[3];
}

int main() {
    std::cout << foo(20) << std::endl;
}

黑魔法版本:

(能看懂这段代码的,都不是普通人!)

#include 
#include 
#include 
#include 
int main() {
 std::vector<int> v(std::atoi(std::end(__DATE__) - (__LINE__) / 2) - 1); // 2021年,第六行
 std::iota(v.begin(), v.end(), 1);
 std::cout << std::accumulate(v.begin(), v.end(), 1ull, std::multiplies<>()) << std::endl;
}

“宏孩儿”元编程版:

(当年看各种C++框架中,排山倒海一样的宏定义,简直令人发指)

#include 

// 由于boost.preprocessor仅提供255以下的整数运算
// 所以使用sequence来 (十位个位)(千位百位)(十万位万位) 的方式来表示大整数。

// 不进位加法:(77)(66)(55) + (44)(33)(22) = (121)(99)(77)
#define PP_ADD_N_N_CARRY_OP(R, DATA, I, ELEM) (BOOST_PP_ADD(BOOST_PP_SEQ_ELEM(I, DATA), ELEM))
#define PP_ADD_N_N_CARRY(SEQ_A, SEQ_B) BOOST_PP_SEQ_FOR_EACH_I(PP_ADD_N_N_CARRY_OP, SEQ_A, SEQ_B)

// 进位加法:(121)(99)(77) = (21)(0)(78)
// 注意SEQ_A的长度要比SEQ_B长
#define PP_ADD_N_N_OP(S, STATE, ELEM_CARRY) \
    BOOST_PP_SEQ_PUSH_FRONT( \
            BOOST_PP_SEQ_REPLACE(STATE, 0, BOOST_PP_MOD(BOOST_PP_ADD(BOOST_PP_SEQ_HEAD(STATE), ELEM_CARRY), 100)), \
            BOOST_PP_DIV(BOOST_PP_ADD(BOOST_PP_SEQ_HEAD(STATE), ELEM_CARRY), 100) \
            )

#define PP_ADD_N_N(SEQ_A, SEQ_B) BOOST_PP_SEQ_REVERSE(BOOST_PP_SEQ_FOLD_LEFT(PP_ADD_N_N_OP, BOOST_PP_SEQ_NIL(0), PP_ADD_N_N_CARRY(SEQ_A, SEQ_B)))

// 没什么好说的,X*N = X+X+X+X+X+...+X
#define PP_MUL_N_1_EXP_OP(Z, I, DATA) (DATA)
#define PP_MUL_N_1_EXP(SEQ_N, N) BOOST_PP_REPEAT(N, PP_MUL_N_1_EXP_OP, SEQ_N)
#define PP_MUL_N_1_MYOP(S, STATE, ITEM) PP_ADD_N_N(STATE, ITEM)
#define PP_MUL_N_1_FWD(EXP) BOOST_PP_SEQ_FOLD_LEFT(PP_MUL_N_1_MYOP, BOOST_PP_SEQ_HEAD(EXP), BOOST_PP_SEQ_TAIL(EXP))
#define PP_MUL_N_1(SEQ_N, N) PP_MUL_N_1_FWD(PP_MUL_N_1_EXP(SEQ_N, N))

#define FACT5 PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1((1), 2), 3), 4), 5)
#define FACT10 PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(FACT5, 6), 7), 8), 9), 10)
#define FACT15 PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(FACT10, 11), 12), 13), 14), 15)
#define FACT20 PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(FACT15, 16), 17), 18), 19), 20)
#define FACT25 PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(PP_MUL_N_1(FACT20, 21), 22), 23), 24), 25)

static_assert(false, BOOST_PP_STRINGIZE(FACT10));

真·模板元编程版本

(泛型编程,码不惊人死不休)

#include 
#include 
#include 

using BaseType_t = long long;
constexpr BaseType_t lgBase = 9// 注意10000*10000刚刚好小于int的取值范围
constexpr BaseType_t Base = 1000000000// 注意10000*10000刚刚好小于int的取值范围

// 大整数的表示
template struct BigInteger {
    using type = BigInteger;
};

// 连接
template<class T1class T2struct BI_Cat;
template struct BI_Cat , BigInteger> : BigInteger {};

// 左移一个单元(即*Base)
template<class Tstruct BI_SHL;
template struct BI_SHL> : BigInteger0> {};

// 去除开头的0
template<class Tstruct BI_Remove_Zeros : T {};
template struct BI_Remove_Zeros> : BI_Remove_Zeros> {};

// 填充0到N个单元
template<int X, class ISstruct BI_Fill_Impl;
template<int X, class TT...Istruct BI_Fill_Impl> : BigInteger<(I, X)...> {};
template<int Size> struct BI_Fill_Zeros : BI_Fill_Impl<0std::make_index_sequence> {};

template<class Tint Nstruct BI_Resize;
templateint N> struct BI_Resize, N> : BI_Cat<typename BI_Fill_Zerossizeof...(I)>::type, BigInteger> {};

// 返回较大的数值
template<int A, int B> struct int_min : std::integral_constant<int, (A {};

// 非进位加法:先把两个数的位数改成一样的然后依次相加
template<class Aclass Bclass ShouldResizestruct BI_AddNotCarry_Impl;
template struct BI_AddNotCarry_Impl , BigInteger, std::true_type> : BigInteger<(I1 + I2)...> {};

template struct BI_AddNotCarry_Impl , BigInteger, std::false_type>
     :
 BI_AddNotCarry_Impl<
         typename BI_Resize, int_min<sizeof...(I1), sizeof...(I2)>::value>::type,
         typename BI_Resize, int_min<sizeof...(I1), sizeof...(I2)>::value>::type,
         std::true_type
     >{};

template<class Aclass Bstruct BI_AddNotCarry;
template struct BI_AddNotCarry , BigInteger>
    :
 BI_AddNotCarry_Impl, BigIntegerstd::bool_constant<sizeof...(I1) == sizeof...(I2)>> {};

// 判断是否为0
template<class Ystruct BI_IsZero;
template struct BI_IsZero> : std::bool_constant<((I == 0) && ...)> {};

// 自动进位
template<class Astruct BI_Carry;
template<class Aclass Bstruct BI_Add : BI_Carry<typename BI_AddNotCarry::type> {};

template<class Modclass Divclass ShouldCalc = typename BI_IsZero
::type> struct BI_Carry_Impl;
template<class Modclass Divstruct BI_Carry_Impl : Mod {};
template<class Modclass Divstruct BI_Carry_Impl
        :
 BI_Addtypename BI_SHL
::type > {};
template struct BI_Carry>
        :
 BI_Remove_Zeros<typename BI_Carry_Impl, BigInteger<(I / Base)...>>::type> {};

// 乘以X并自动进位
template<class Aint Xstruct BI_MulX;
templateint X> struct BI_MulX , X>
        :
 BI_Carry> {};

// 计算阶乘
template<int X> struct BI_Fact : BI_MulX<typename BI_Fact-1>::type, X> {};
template<> struct BI_Fact<0> : BigInteger<1> {};

template
std::ostream &operator<<(std::ostream &out, BigInteger) {
    return ((out << std::setfill('0') << I << std::setw(lgBase)), ...);
}

int main()
{
    std::cout << typename BI_Fact<20>::type() << std::endl;
}

原回答:

https://www.zhihu.com/question/365763395/answer/971009059

不得不服,这位老哥真是人才,把 C++ 玩出这么多花样,不愧是语言学家。当然,现实工作中不会有人这样写代码的,所以大家图个乐就好,不用较真。

你们在第几个版本倒下了?

我反正看到面向对象专家版本就已经忍不住了···

EOF

你好,我是飞宇,本硕均于某中流985 CS就读,先后于百度搜索字节跳动电商以及携程等部门担任Linux C/C++后端研发工程师。

最近跟朋友一起开发了一个新的网站:编程资源网,已经收录了不少资源(附赠下载地址),如果屏幕前的靓仔/女想要学习编程找不到合适资源的话,不妨来我们的网站看看,欢迎扫码下方二维码白嫖~

同时,我也是知乎博主@韩飞宇,日常分享C/C++、计算机学习经验、工作体会,欢迎点击此处查看我以前的学习笔记&经验&分享的资源。

我组建了一些社群一起交流,群里有大牛也有小白,如果你有意可以一起进群交流。

欢迎你添加我的微信,我拉你进技术交流群。此外,我也会经常在微信上分享一些计算机学习经验以及工作体验,还有一些内推机会

加个微信,打开另一扇窗


C语言与CPP编程 C语言/C++开发,C语言/C++基础知识,C语言/C++学习路线,C语言/C++进阶,数据结构;算法;python;计算机基础等
评论
  •     IPC-2581是基于ODB++标准、结合PCB行业特点而指定的PCB加工文件规范。    IPC-2581旨在替代CAM350格式,成为PCB加工行业的新的工业规范。    有一些免费软件,可以查看(不可修改)IPC-2581数据文件。这些软件典型用途是工艺校核。    1. Vu2581        出品:Downstream     
    电子知识打边炉 2025-01-22 11:12 45浏览
  • 嘿,咱来聊聊RISC-V MCU技术哈。 这RISC-V MCU技术呢,简单来说就是基于一个叫RISC-V的指令集架构做出的微控制器技术。RISC-V这个啊,2010年的时候,是加州大学伯克利分校的研究团队弄出来的,目的就是想搞个新的、开放的指令集架构,能跟上现代计算的需要。到了2015年,专门成立了个RISC-V基金会,让这个架构更标准,也更好地推广开了。这几年啊,这个RISC-V的生态系统发展得可快了,好多公司和机构都加入了RISC-V International,还推出了不少RISC-V
    丙丁先生 2025-01-21 12:10 105浏览
  • Ubuntu20.04默认情况下为root账号自动登录,本文介绍如何取消root账号自动登录,改为通过输入账号密码登录,使用触觉智能EVB3568鸿蒙开发板演示,搭载瑞芯微RK3568,四核A55处理器,主频2.0Ghz,1T算力NPU;支持OpenHarmony5.0及Linux、Android等操作系统,接口丰富,开发评估快人一步!添加新账号1、使用adduser命令来添加新用户,用户名以industio为例,系统会提示设置密码以及其他信息,您可以根据需要填写或跳过,命令如下:root@id
    Industio_触觉智能 2025-01-17 14:14 118浏览
  • 数字隔离芯片是一种实现电气隔离功能的集成电路,在工业自动化、汽车电子、光伏储能与电力通信等领域的电气系统中发挥着至关重要的作用。其不仅可令高、低压系统之间相互独立,提高低压系统的抗干扰能力,同时还可确保高、低压系统之间的安全交互,使系统稳定工作,并避免操作者遭受来自高压系统的电击伤害。典型数字隔离芯片的简化原理图值得一提的是,数字隔离芯片历经多年发展,其应用范围已十分广泛,凡涉及到在高、低压系统之间进行信号传输的场景中基本都需要应用到此种芯片。那么,电气工程师在进行电路设计时到底该如何评估选择一
    华普微HOPERF 2025-01-20 16:50 73浏览
  •  万万没想到!科幻电影中的人形机器人,正在一步步走进我们人类的日常生活中来了。1月17日,乐聚将第100台全尺寸人形机器人交付北汽越野车,再次吹响了人形机器人疯狂进厂打工的号角。无独有尔,银河通用机器人作为一家成立不到两年时间的创业公司,在短短一年多时间内推出革命性的第一代产品Galbot G1,这是一款轮式、双臂、身体可折叠的人形机器人,得到了美团战投、经纬创投、IDG资本等众多投资方的认可。作为一家成立仅仅只有两年多时间的企业,智元机器人也把机器人从梦想带进了现实。2024年8月1
    刘旷 2025-01-21 11:15 261浏览
  •  光伏及击穿,都可视之为 复合的逆过程,但是,复合、光伏与击穿,不单是进程的方向相反,偏置状态也不一样,复合的工况,是正偏,光伏是零偏,击穿与漂移则是反偏,光伏的能源是外来的,而击穿消耗的是结区自身和电源的能量,漂移的载流子是 客席载流子,须借外延层才能引入,客席载流子 不受反偏PN结的空乏区阻碍,能漂不能漂,只取决于反偏PN结是否处于外延层的「射程」范围,而穿通的成因,则是因耗尽层的过度扩张,致使跟 端子、外延层或其他空乏区 碰触,当耗尽层融通,耐压 (反向阻断能力) 即告彻底丧失,
    MrCU204 2025-01-17 11:30 176浏览
  • 日前,商务部等部门办公厅印发《手机、平板、智能手表(手环)购新补贴实施方案》明确,个人消费者购买手机、平板、智能手表(手环)3类数码产品(单件销售价格不超过6000元),可享受购新补贴。每人每类可补贴1件,每件补贴比例为减去生产、流通环节及移动运营商所有优惠后最终销售价格的15%,每件最高不超过500元。目前,京东已经做好了承接手机、平板等数码产品国补优惠的落地准备工作,未来随着各省市关于手机、平板等品类的国补开启,京东将第一时间率先上线,满足消费者的换新升级需求。为保障国补的真实有效发放,基于
    华尔街科技眼 2025-01-17 10:44 221浏览
  • 2024年是很平淡的一年,能保住饭碗就是万幸了,公司业绩不好,跳槽又不敢跳,还有一个原因就是老板对我们这些员工还是很好的,碍于人情也不能在公司困难时去雪上加霜。在工作其间遇到的大问题没有,小问题还是有不少,这里就举一两个来说一下。第一个就是,先看下下面的这个封装,你能猜出它的引脚间距是多少吗?这种排线座比较常规的是0.6mm间距(即排线是0.3mm间距)的,而这个规格也是我们用得最多的,所以我们按惯性思维来看的话,就会认为这个座子就是0.6mm间距的,这样往往就不会去细看规格书了,所以这次的运气
    wuliangu 2025-01-21 00:15 153浏览
  • 现在为止,我们已经完成了Purple Pi OH主板的串口调试和部分配件的连接,接下来,让我们趁热打铁,完成剩余配件的连接!注:配件连接前请断开主板所有供电,避免敏感电路损坏!1.1 耳机接口主板有一路OTMP 标准四节耳机座J6,具备进行音频输出及录音功能,接入耳机后声音将优先从耳机输出,如下图所示:1.21.2 相机接口MIPI CSI 接口如上图所示,支持OV5648 和OV8858 摄像头模组。接入摄像头模组后,使用系统相机软件打开相机拍照和录像,如下图所示:1.3 以太网接口主板有一路
    Industio_触觉智能 2025-01-20 11:04 141浏览
  • 本文介绍瑞芯微开发板/主板Android配置APK默认开启性能模式方法,开启性能模式后,APK的CPU使用优先级会有所提高。触觉智能RK3562开发板演示,搭载4核A53处理器,主频高达2.0GHz;内置独立1Tops算力NPU,可应用于物联网网关、平板电脑、智能家居、教育电子、工业显示与控制等行业。源码修改修改源码根目录下文件device/rockchip/rk3562/package_performance.xml并添加以下内容,注意"+"号为添加内容,"com.tencent.mm"为AP
    Industio_触觉智能 2025-01-17 14:09 159浏览
  • 高速先生成员--黄刚这不马上就要过年了嘛,高速先生就不打算给大家上难度了,整一篇简单但很实用的文章给大伙瞧瞧好了。相信这个标题一出来,尤其对于PCB设计工程师来说,心就立马凉了半截。他们辛辛苦苦进行PCB的过孔设计,高速先生居然说设计多大的过孔他们不关心!另外估计这时候就跳出很多“挑刺”的粉丝了哈,因为翻看很多以往的文章,高速先生都表达了过孔孔径对高速性能的影响是很大的哦!咋滴,今天居然说孔径不关心了?别,别急哈,听高速先生在这篇文章中娓娓道来。首先还是要对各位设计工程师的设计表示肯定,毕竟像我
    一博科技 2025-01-21 16:17 95浏览
我要评论
0
点击右上角,分享到朋友圈 我知道啦
请使用浏览器分享功能 我知道啦