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