วิธีการไดรฟ์ C #, C ++ หรือคอมไพเลอร์ Java เพื่อคำนวณ 1 + 2 + 3 + ... + 1000 ในเวลาคอมไพล์?


122

ในการสัมภาษณ์ล่าสุดฉันถูกถามคำถามแปลก ๆ ผู้สัมภาษณ์ถามฉันว่าฉันจะคำนวณ 1 + 2 + 3 + ... + 1000 โดยใช้คุณสมบัติคอมไพเลอร์ได้อย่างไร ซึ่งหมายความว่าฉันไม่ได้รับอนุญาตให้เขียนโปรแกรมและเรียกใช้งาน แต่ฉันควรเขียนโปรแกรมที่สามารถขับเคลื่อนคอมไพเลอร์ให้คำนวณผลรวมนี้ในขณะที่คอมไพล์และพิมพ์ผลลัพธ์เมื่อการคอมไพล์เสร็จสิ้น เขาบอกฉันว่าฉันอาจใช้ generics และคุณสมบัติ pre-processor ของคอมไพเลอร์ เป็นไปได้ที่จะใช้คอมไพเลอร์ C ++, C # หรือ Java ความคิดใด ๆ ???

คำถามนี้ไม่เกี่ยวข้องกับการคำนวณผลรวมโดยไม่มีการวนซ้ำใด ๆ ที่ถามที่นี่ นอกจากนี้ควรสังเกตว่าควรคำนวณผลรวมระหว่างการคอมไพล์ ไม่สามารถพิมพ์เฉพาะผลลัพธ์โดยใช้คำสั่งคอมไพเลอร์ C ++


อ่านข้อมูลเพิ่มเติมเกี่ยวกับคำตอบโพสต์ที่ฉันพบว่าการแก้ปัญหาที่เกิดขึ้นในระหว่างการรวบรวมโดยใช้ C ++ แม่แบบที่เรียกว่าmetaprogramming นี่เป็นเทคนิคที่ดร. เออร์วินอุนรูห์ค้นพบโดยบังเอิญระหว่างกระบวนการสร้างมาตรฐานภาษา C ++ คุณสามารถอ่านเพิ่มเติมเกี่ยวกับหัวข้อนี้ในหน้าวิกิพีเดียของ meta-การเขียนโปรแกรม ดูเหมือนว่าจะสามารถเขียนโปรแกรมใน Java โดยใช้คำอธิบายประกอบจาวา คุณสามารถดูคำตอบของ maressด้านล่าง

เป็นหนังสือที่ดีเกี่ยวกับเมตาการเขียนโปรแกรมใน C ++ เป็นอย่างใดอย่างหนึ่ง คุ้มค่าที่จะดูหากสนใจ

ไลบรารีการเขียนโปรแกรมเมตา C ++ ที่มีประโยชน์คือ MPL ของ Boost ที่ลิงค์นี้


17
#error "500500" ข้อผิดพลาดในการคอมไพล์นับว่า "เสร็จสมบูรณ์" หรือไม่
Mysticial

4
คำใบ้หมายถึงการใช้แม่แบบ C ++ เห็นได้ชัดว่าไม่เหมือนกัน แต่อันนี้ใช้สำหรับการพิมพ์ 1 ถึง 1,000 ฉันแน่ใจว่าคุณสามารถแก้ไขเพื่อเพิ่มเป็นพันได้ ... stackoverflow.com/questions/4568645/…
โจ

8
const int value = 1 + 2 + 3.... + 1000; Console.WriteLine(value);; P
George Duckett

8
บางครั้งฉันคิดว่าคำถามสัมภาษณ์บางคำถามเป็นเพียงเพื่อพิสูจน์ความฉลาดของผู้สัมภาษณ์เหนือผู้ถูกสัมภาษณ์
Chris Dwyer

4
คุณเคยขอเงินจำนวนมากก่อนที่คุณจะถูกถามคำถามนี้หรือไม่?
Salman A

คำตอบ:


118

อัปเดตทันทีด้วยความลึกของการเรียกซ้ำที่ดีขึ้น! ทำงานบน MSVC10 และ GCC โดยไม่ต้องเพิ่มความลึก :)


การเรียกซ้ำเวลาคอมไพล์อย่างง่าย + การเพิ่ม:

template<unsigned Cur, unsigned Goal>
struct adder{
  static unsigned const sub_goal = (Cur + Goal) / 2;
  static unsigned const tmp = adder<Cur, sub_goal>::value;
  static unsigned const value = tmp + adder<sub_goal+1, Goal>::value;
};

template<unsigned Goal>
struct adder<Goal, Goal>{
  static unsigned const value = Goal;
};

Testcode:

template<unsigned Start>
struct sum_from{
  template<unsigned Goal>
  struct to{
    template<unsigned N>
    struct equals;

    typedef equals<adder<Start, Goal>::value> result;
  };
};

int main(){
  sum_from<1>::to<1000>::result();
}

ผลลัพธ์สำหรับ GCC:

ข้อผิดพลาด: การประกาศ 'โครงสร้างผลรวม _ จาก <1> :: ถึง <1000> :: เท่ากับ <500500>'

ตัวอย่างเช่นอยู่บน Ideone

เอาต์พุตสำหรับ MSVC10:

error C2514: 'sum_from<Start>::to<Goal>::equals<Result>' : class has no constructors
      with
      [
          Start=1,
          Goal=1000,
          Result=500500
      ]

@hsalimi: ฉันแก้ไขคำตอบเพื่อแสดงรหัสที่ทำให้งานเสร็จ :)
Xeo

ว้าวคุณทำให้ฉันประทับใจจริงๆ :-)
Gupta

@hsalimi: ดร. เออร์วินอุนรูห์เป็นผู้คิดค้นเทคนิคนี้ในการประชุมมาตรฐาน C ++ ปี 1997 ที่สตอกโฮล์ม เขาคำนวณอนุกรมของจำนวนเฉพาะ
Dietmar Kühl

เพื่อให้ทำงานได้โดยไม่ต้องเรียกซ้ำคุณสามารถใช้สูตรของ N * (N + 1) / 2 เพื่อคำนวณผลรวม
Adam Gritt

2
@hsalimi ในกรณีที่คุณต้องการดูตัวอย่างการเขียนโปรแกรมเทมเพลตใน C ++ ที่ยอดเยี่ยมมากขึ้นฉันขอแนะนำModern C ++ Designโดย Andrei Alexandrescu
Darhuuk

89

ตัวอย่าง C # ข้อผิดพลาดในเวลาคอมไพล์

class Foo
{
    const char Sum = (1000 + 1) * 1000 / 2;
}

สร้างข้อผิดพลาดในการคอมไพล์ต่อไปนี้:

Constant value '500500' cannot be converted to a 'char' 

4
@ildjarn มีความแตกต่างระหว่างคำตอบเทมเพลต c ++ กับคำตอบนี้: สิ่งนี้ใช้ได้เฉพาะเนื่องจากการพับคงที่ในขณะที่เทมเพลตอนุญาตให้ใช้รหัส (?) โดยพลการ ยังคงเป็นความคิดที่ดีที่จะมอบหมายให้ถ่าน!
Voo

@Voo ใช่ แต่เพื่อความยุติธรรม C # ก็ไม่ได้เปรียบเทียบกับ C ++ สำหรับการเขียนโปรแกรมประเภทนี้
Marlon

3
@ มาริออนและฉันไม่คิดว่าความผิดพลาดในการออกแบบภาษา;) การเขียนโปรแกรมเมตาเทมเพลตอาจมีประสิทธิภาพทั้งหมด แต่ภาษาอื่น ๆ ยังสามารถทำสิ่งต่างๆได้เกือบทั้งหมดด้วยโซลูชันอื่น ๆ ที่ไม่มีข้อผิดพลาดทั้งหมด ฉันต้องทำงานในโปรเจ็กต์ที่ใช้เวลาหลายชั่วโมงในการคอมไพล์ (ไม่จริงทั้งหมด - มันเร็วอย่างไม่น่าเชื่อถ้าเราไม่เพิ่มขีด จำกัด อินสแตนซ์แบบเรียกซ้ำ .. มันล้มเหลวในไม่กี่วินาที) และแย่มากที่จะรักษา อาจเป็นเหตุผลว่าทำไมฉันถึงไม่ค่อยเป็นแฟนของมัน ..
Voo

@Voo: แนวทางของ FredOverflow อาศัยการพับแบบคงที่เช่นกัน สำหรับการคอมไพล์ช้าให้ตำหนิคอมไพเลอร์ของคุณไม่ใช่ภาษา (คำใบ้ - เสียงดังรวบรวม C ++ เร็ว )
ildjarn

@ildjarn Clang รวบรวมเทมเพลตที่ซับซ้อนซ้อนลึกและซับซ้อนมากอย่างรวดเร็ว? ฉันคิดว่าทุกอย่างเป็นไปได้และฉันไม่สามารถทดสอบได้อีกแล้ว (ขอบคุณพระเจ้า) แต่ฉันนึกไม่ออก ฉันกำลังพูดถึงแนวทางของ Xeo ไม่ใช่ของ Fred ที่นี่
Voo

51

ฉันควรเขียนโปรแกรมที่สามารถขับเคลื่อนคอมไพเลอร์เพื่อคำนวณผลรวมนี้ในขณะที่คอมไพล์และพิมพ์ผลลัพธ์เมื่อการคอมไพล์เสร็จสมบูรณ์

เคล็ดลับยอดนิยมในการพิมพ์หมายเลขระหว่างการคอมไพล์คือการพยายามเข้าถึงสมาชิกที่ไม่มีอยู่จริงของเทมเพลตที่สร้างอินสแตนซ์ด้วยหมายเลขที่คุณต้องการพิมพ์:

template<int> struct print_n {};

print_n<1000 * 1001 / 2>::foobar go;

คอมไพเลอร์ก็พูดว่า:

error: 'foobar' in 'struct print_n<500500>' does not name a type

สำหรับตัวอย่างที่น่าสนใจมากขึ้นของเทคนิคนี้ให้ดูที่การแก้ปัญหาแปดราชินีที่รวบรวมเวลา


คุณสามารถปล่อยให้print_nอยู่เฉยๆดูคำตอบของฉัน
Xeo

2
@David แต่ Gauss ต้องการวิธีที่ชาญฉลาดเขาไม่มีคอมพิวเตอร์เพื่อทำวิธีที่โง่อย่างรวดเร็ว
Daniel Fischer

31

เนื่องจากไม่มีการระบุทั้งคอมไพเลอร์หรือภาษาในคำถามสัมภาษณ์ฉันจึงกล้าส่งวิธีแก้ปัญหาใน Haskell โดยใช้ GHC:

{-# LANGUAGE TemplateHaskell #-}
{-# OPTIONS_GHC -ddump-splices #-}
module Main where

main :: IO ()
main = print $(let x = sum [1 :: Int .. 1000] in [| x |])

รวบรวมมัน:

$ ghc compsum.hs
[1 of 1] Compiling Main             ( compsum.hs, compsum.o )
Loading package ghc-prim ... linking ... done.
<snip more "Loading package ..." messages>
Loading package template-haskell ... linking ... done.
compsum.hs:6:16-56: Splicing expression
    let x = sum [1 :: Int .. 1000] in [| x |] ======> 500500
Linking compsum ...

และเรามีโปรแกรมการทำงานด้วย


20

ชีวิตจะง่ายขึ้นมากด้วย C ++ 11 ซึ่งเพิ่มconstexprฟังก์ชันสำหรับการคำนวณเวลาในการคอมไพล์แม้ว่าตอนนี้จะรองรับเฉพาะ gcc 4.6 หรือใหม่กว่า

constexpr unsigned sum(unsigned start, unsigned end) {
    return start == end ? start :
        sum(start, (start + end) / 2) +
        sum((start + end) / 2 + 1, end);
}

template <int> struct equals;
equals<sum(1,1000)> x;

มาตรฐานต้องการเพียงคอมไพเลอร์เพื่อรองรับความลึกของการเรียกซ้ำ 512 ดังนั้นจึงยังคงต้องหลีกเลี่ยงความลึกของการเรียกซ้ำเชิงเส้น นี่คือผลลัพธ์:

$ g++-mp-4.6 --std=c++0x test.cpp -c
test.cpp:8:25: error: aggregate 'equals<500500> x' has incomplete type and cannot be defined

แน่นอนคุณสามารถใช้สูตร:

constexpr unsigned sum(unsigned start, unsigned end) {
    return (start + end) * (end - start + 1) / 2;
}

// static_assert is a C++11 assert, which checks
// at compile time.
static_assert(sum(0,1000) == 500500, "Sum failed for 0 to 1000");

1
+1 ลืมไปconstexprชั่วขณะ บางทีฉันอาจจะชอบเทมเพลตมากเกินไป :(
Xeo

นี่เป็นการใช้ constexpr ที่ดีในการตอบคำถาม (ดูการใช้งาน Adder): kaizer.se/wiki/log/post/C++_constexpr_foldr
Matt

สูตรนั้นสามารถล้น; ขั้นตอนสุดท้ายคือ/ 2การจัดการกับunsignedผลลัพธ์ที่เป็นไปได้ทั้งหมดค่าที่คุณขยับถูกต้องจะต้องกว้าง n + 1 บิต แต่ไม่ใช่ เป็นไปได้ที่จะจัดเรียงสูตรใหม่เพื่อหลีกเลี่ยงสิ่งนั้นเช่น clang ทำสำหรับช่วงตัวแปรรันไทม์: godbolt.org/z/dUGXqgแสดงให้เห็นว่าเสียงดังกราวรู้สูตรรูปแบบปิดและใช้เพื่อเพิ่มประสิทธิภาพลูtotal += i
Peter Cordes

14

ใน java ฉันคิดเกี่ยวกับการใช้การประมวลผลคำอธิบายประกอบ เครื่องมือ apt จะสแกนซอร์สไฟล์ก่อนที่จะแยกวิเคราะห์ไฟล์ต้นฉบับไปยังคำสั่ง javac

ระหว่างการคอมไพล์ไฟล์ต้นฉบับผลลัพธ์จะถูกพิมพ์ออกมา:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.METHOD})
public @interface MyInterface {

    int offset() default 0;

    int last() default 100;
}

โรงงานโปรเซสเซอร์:

public class MyInterfaceAnnotationProcessorFactory implements AnnotationProcessorFactory {

    public Collection<String> supportedOptions() {
        System.err.println("Called supportedOptions.............................");
        return Collections.EMPTY_LIST;
    }

    public Collection<String> supportedAnnotationTypes() {
        System.err.println("Called supportedAnnotationTypes...........................");
        return Collections.singletonList("practiceproject.MyInterface");
    }

    public AnnotationProcessor getProcessorFor(Set<AnnotationTypeDeclaration> set, AnnotationProcessorEnvironment ape) {
        System.err.println("Called getProcessorFor................");
        if (set.isEmpty()) {
            return AnnotationProcessors.NO_OP;
        }
        return new MyInterfaceAnnotationProcessor(ape);
    }
}

ตัวประมวลผลคำอธิบายประกอบที่แท้จริง:

public class MyInterfaceAnnotationProcessor implements AnnotationProcessor {

    private AnnotationProcessorEnvironment ape;
    private AnnotationTypeDeclaration atd;

    public MyInterfaceAnnotationProcessor(AnnotationProcessorEnvironment ape) {
        this.ape = ape;
        atd = (AnnotationTypeDeclaration) ape.getTypeDeclaration("practiceproject.MyInterface");
    }

    public void process() {
        Collection<Declaration> decls = ape.getDeclarationsAnnotatedWith(atd);
        for (Declaration dec : decls) {
            processDeclaration(dec);
        }
    }

    private void processDeclaration(Declaration d) {
        Collection<AnnotationMirror> ams = d.getAnnotationMirrors();
        for (AnnotationMirror am : ams) {
            if (am.getAnnotationType().getDeclaration().equals(atd)) {
                Map<AnnotationTypeElementDeclaration, AnnotationValue> values = am.getElementValues();
                int offset = 0;
                int last = 100;
                for (Map.Entry<AnnotationTypeElementDeclaration, AnnotationValue> entry : values.entrySet()) {
                    AnnotationTypeElementDeclaration ated = entry.getKey();
                    AnnotationValue v = entry.getValue();
                    String name = ated.getSimpleName();
                    if (name.equals("offset")) {
                        offset = ((Integer) v.getValue()).intValue();
                    } else if (name.equals("last")) {
                        last = ((Integer) v.getValue()).intValue();
                    }
                }
                //find the sum
                System.err.println("Sum: " + ((last + 1 - offset) / 2) * (2 * offset + (last - offset)));
            }
        }
    }
}

จากนั้นเราสร้างไฟล์ต้นฉบับ คลาสง่ายๆที่ใช้คำอธิบายประกอบ MyInterface:

 @MyInterface(offset = 1, last = 1000)
public class Main {

    @MyInterface
    void doNothing() {
        System.out.println("Doing nothing");
    }

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) {
        // TODO code application logic here
        Main m = new Main();
        m.doNothing();
        MyInterface my = (MyInterface) m.getClass().getAnnotation(MyInterface.class);
        System.out.println("offset: " + my.offset());
        System.out.println("Last: " + my.last());
    }
}

ตัวประมวลผลคำอธิบายประกอบถูกคอมไพล์เป็นไฟล์ jar จากนั้นใช้เครื่องมือ apt เพื่อคอมไพล์ซอร์สไฟล์เป็น:

apt -cp "D:\Variance project\PracticeProject\dist\practiceproject.jar" -factory practiceproject.annotprocess.MyInterfaceAnnotationProcessorFactory "D:\Variance project\PracticeProject2\src\practiceproject2\Main.java"

ผลลัพธ์ของโครงการ:

Called supportedAnnotationTypes...........................
Called getProcessorFor................
Sum: 5000
Sum: 500500

9

นี่คือการใช้งานที่ทำงานภายใต้ VC ++ 2010 ฉันต้องแบ่งการคำนวณออกเป็น 3 ขั้นตอนเนื่องจากคอมไพเลอร์บ่นเมื่อเทมเพลตเรียกคืนมากกว่า 500 ครั้ง

template<int t_startVal, int t_baseVal = 0, int t_result = 0>
struct SumT
{
    enum { result = SumT<t_startVal - 1, t_baseVal, t_baseVal + t_result +
        t_startVal>::result };
};

template<int t_baseVal, int t_result>
struct SumT<0, t_baseVal, t_result>
{
    enum { result = t_result };
};

template<int output_value>
struct Dump
{
    enum { value = output_value };
    int bad_array[0];
};

enum
{
    value1 = SumT<400>::result,                // [1,400]
    value2 = SumT<400, 400, value1>::result,   // [401, 800]
    value3 = SumT<200, 800, value2>::result    // [801, 1000]
};

Dump<value3> dump;

เมื่อคุณรวบรวมสิ่งนี้คุณจะเห็นผลลัพธ์นี้จากคอมไพเลอร์ดังนี้:

1>warning C4200: nonstandard extension used : zero-sized array in struct/union
1>          Cannot generate copy-ctor or copy-assignment operator when UDT contains a 
zero-sized array
1>          templatedrivensum.cpp(33) : see reference to class template 
instantiation 'Dump<output_value>' being compiled
1>          with
1>          [
1>              output_value=500500
1>          ]

เป็นความคิดที่ดีมากในการทำลายมันลงฉันคิดว่าฉันจะรวมสิ่งนั้นไว้ในคำตอบของฉันอย่างใด +1 :)
Xeo

9

ฉันรู้สึกผูกพันที่จะต้องให้รหัส C นี้เนื่องจากยังไม่มีใคร:

#include <stdio.h>
int main() {
   int x = 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+32+33+34+35+36+37+38+39+40+
           41+42+43+44+45+46+47+48+49+50+51+52+53+54+55+56+57+58+59+60+
           61+62+63+64+65+66+67+68+69+70+71+72+73+74+75+76+77+78+79+80+
           81+82+83+84+85+86+87+88+89+90+91+92+93+94+95+96+97+98+99+100+     
           101+102+103+104+105+106+107+108+109+110+111+112+113+114+115+116+117+118+119+120+
           121+122+123+124+125+126+127+128+129+130+131+132+133+134+135+136+137+138+139+140+
           141+142+143+144+145+146+147+148+149+150+151+152+153+154+155+156+157+158+159+160+
           161+162+163+164+165+166+167+168+169+170+171+172+173+174+175+176+177+178+179+180+
           181+182+183+184+185+186+187+188+189+190+191+192+193+194+195+196+197+198+199+200+
           201+202+203+204+205+206+207+208+209+210+211+212+213+214+215+216+217+218+219+220+
           221+222+223+224+225+226+227+228+229+230+231+232+233+234+235+236+237+238+239+240+
           241+242+243+244+245+246+247+248+249+250+251+252+253+254+255+256+257+258+259+260+
           261+262+263+264+265+266+267+268+269+270+271+272+273+274+275+276+277+278+279+280+
           281+282+283+284+285+286+287+288+289+290+291+292+293+294+295+296+297+298+299+300+
           301+302+303+304+305+306+307+308+309+310+311+312+313+314+315+316+317+318+319+320+
           321+322+323+324+325+326+327+328+329+330+331+332+333+334+335+336+337+338+339+340+
           341+342+343+344+345+346+347+348+349+350+351+352+353+354+355+356+357+358+359+360+
           361+362+363+364+365+366+367+368+369+370+371+372+373+374+375+376+377+378+379+380+
           381+382+383+384+385+386+387+388+389+390+391+392+393+394+395+396+397+398+399+400+
           401+402+403+404+405+406+407+408+409+410+411+412+413+414+415+416+417+418+419+420+
           421+422+423+424+425+426+427+428+429+430+431+432+433+434+435+436+437+438+439+440+
           441+442+443+444+445+446+447+448+449+450+451+452+453+454+455+456+457+458+459+460+
           461+462+463+464+465+466+467+468+469+470+471+472+473+474+475+476+477+478+479+480+
           481+482+483+484+485+486+487+488+489+490+491+492+493+494+495+496+497+498+499+500+
           501+502+503+504+505+506+507+508+509+510+511+512+513+514+515+516+517+518+519+520+
           521+522+523+524+525+526+527+528+529+530+531+532+533+534+535+536+537+538+539+540+
           541+542+543+544+545+546+547+548+549+550+551+552+553+554+555+556+557+558+559+560+
           561+562+563+564+565+566+567+568+569+570+571+572+573+574+575+576+577+578+579+580+
           581+582+583+584+585+586+587+588+589+590+591+592+593+594+595+596+597+598+599+600+
           601+602+603+604+605+606+607+608+609+610+611+612+613+614+615+616+617+618+619+620+
           621+622+623+624+625+626+627+628+629+630+631+632+633+634+635+636+637+638+639+640+
           641+642+643+644+645+646+647+648+649+650+651+652+653+654+655+656+657+658+659+660+
           661+662+663+664+665+666+667+668+669+670+671+672+673+674+675+676+677+678+679+680+
           681+682+683+684+685+686+687+688+689+690+691+692+693+694+695+696+697+698+699+700+
           701+702+703+704+705+706+707+708+709+710+711+712+713+714+715+716+717+718+719+720+
           721+722+723+724+725+726+727+728+729+730+731+732+733+734+735+736+737+738+739+740+
           741+742+743+744+745+746+747+748+749+750+751+752+753+754+755+756+757+758+759+760+
           761+762+763+764+765+766+767+768+769+770+771+772+773+774+775+776+777+778+779+780+
           781+782+783+784+785+786+787+788+789+790+791+792+793+794+795+796+797+798+799+800+
           801+802+803+804+805+806+807+808+809+810+811+812+813+814+815+816+817+818+819+820+
           821+822+823+824+825+826+827+828+829+830+831+832+833+834+835+836+837+838+839+840+
           841+842+843+844+845+846+847+848+849+850+851+852+853+854+855+856+857+858+859+860+
           861+862+863+864+865+866+867+868+869+870+871+872+873+874+875+876+877+878+879+880+
           881+882+883+884+885+886+887+888+889+890+891+892+893+894+895+896+897+898+899+900+
           901+902+903+904+905+906+907+908+909+910+911+912+913+914+915+916+917+918+919+920+
           921+922+923+924+925+926+927+928+929+930+931+932+933+934+935+936+937+938+939+940+
           941+942+943+944+945+946+947+948+949+950+951+952+953+954+955+956+957+958+959+960+
           961+962+963+964+965+966+967+968+969+970+971+972+973+974+975+976+977+978+979+980+
           981+982+983+984+985+986+987+988+989+990+991+992+993+994+995+996+997+998+999+1000;
  printf("%d\n", x);
}

และสิ่งที่ฉันต้องทำคือตรวจสอบชุดประกอบเพื่อหาคำตอบของฉัน!

gcc -S compile_sum.c;
grep "\$[0-9]*, *-4" compile_sum.s

และฉันเห็น:

movl    $500500, -4(%rbp)

คุณลักษณะของการใช้งานเฉพาะไม่ใช่ภาษา C
ลูกสุนัข

5
คุณรู้จักคอมไพเลอร์ Cกี่ตัวที่ไม่ใช่ "การนำไปใช้งานเฉพาะ" ของ C?
Carl Walsh

@Puppy: ถ้าxเป็น global คอมไพเลอร์จะต้องใช้ (ไม่มากก็น้อย) เพื่อประเมินนิพจน์ในขณะคอมไพล์ ISO C ไม่อนุญาตตัวเริ่มต้นตัวแปรรันไทม์สำหรับ globals แน่นอนว่าการนำไปใช้งานเฉพาะสามารถส่งเสียงเรียกไปยังฟังก์ชัน static-init ที่เหมือนตัวสร้างซึ่งคำนวณที่รันไทม์และร้านค้า แต่ ISO C ช่วยให้คุณใช้ค่าคงที่ของเวลาในการคอมไพล์เป็นขนาดอาร์เรย์ (เช่นint y[x];ในนิยามโครงสร้างหรือเป็นโกลบอลอื่นเป็นต้น) ดังนั้นการนำไปใช้ในแง่ร้ายเชิงสมมุติก็ยังคงต้องรองรับ
Peter Cordes

7

ขยายความจากคำตอบของ Carl Walsh เพื่อพิมพ์ผลลัพธ์ระหว่างการรวบรวม:

#define VALUE (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+32+33+34+35+36+37+38+39+40+\
41+42+43+44+45+46+47+48+49+50+51+52+53+54+55+56+57+58+59+60+\
61+62+63+64+65+66+67+68+69+70+71+72+73+74+75+76+77+78+79+80+\
81+82+83+84+85+86+87+88+89+90+91+92+93+94+95+96+97+98+99+100+\
101+102+103+104+105+106+107+108+109+110+111+112+113+114+115+116+117+118+119+120+\
121+122+123+124+125+126+127+128+129+130+131+132+133+134+135+136+137+138+139+140+\
141+142+143+144+145+146+147+148+149+150+151+152+153+154+155+156+157+158+159+160+\
161+162+163+164+165+166+167+168+169+170+171+172+173+174+175+176+177+178+179+180+\
181+182+183+184+185+186+187+188+189+190+191+192+193+194+195+196+197+198+199+200+\
201+202+203+204+205+206+207+208+209+210+211+212+213+214+215+216+217+218+219+220+\
221+222+223+224+225+226+227+228+229+230+231+232+233+234+235+236+237+238+239+240+\
241+242+243+244+245+246+247+248+249+250+251+252+253+254+255+256+257+258+259+260+\
261+262+263+264+265+266+267+268+269+270+271+272+273+274+275+276+277+278+279+280+\
281+282+283+284+285+286+287+288+289+290+291+292+293+294+295+296+297+298+299+300+\
301+302+303+304+305+306+307+308+309+310+311+312+313+314+315+316+317+318+319+320+\
321+322+323+324+325+326+327+328+329+330+331+332+333+334+335+336+337+338+339+340+\
341+342+343+344+345+346+347+348+349+350+351+352+353+354+355+356+357+358+359+360+\
361+362+363+364+365+366+367+368+369+370+371+372+373+374+375+376+377+378+379+380+\
381+382+383+384+385+386+387+388+389+390+391+392+393+394+395+396+397+398+399+400+\
401+402+403+404+405+406+407+408+409+410+411+412+413+414+415+416+417+418+419+420+\
421+422+423+424+425+426+427+428+429+430+431+432+433+434+435+436+437+438+439+440+\
441+442+443+444+445+446+447+448+449+450+451+452+453+454+455+456+457+458+459+460+\
461+462+463+464+465+466+467+468+469+470+471+472+473+474+475+476+477+478+479+480+\
481+482+483+484+485+486+487+488+489+490+491+492+493+494+495+496+497+498+499+500+\
501+502+503+504+505+506+507+508+509+510+511+512+513+514+515+516+517+518+519+520+\
521+522+523+524+525+526+527+528+529+530+531+532+533+534+535+536+537+538+539+540+\
541+542+543+544+545+546+547+548+549+550+551+552+553+554+555+556+557+558+559+560+\
561+562+563+564+565+566+567+568+569+570+571+572+573+574+575+576+577+578+579+580+\
581+582+583+584+585+586+587+588+589+590+591+592+593+594+595+596+597+598+599+600+\
601+602+603+604+605+606+607+608+609+610+611+612+613+614+615+616+617+618+619+620+\
621+622+623+624+625+626+627+628+629+630+631+632+633+634+635+636+637+638+639+640+\
641+642+643+644+645+646+647+648+649+650+651+652+653+654+655+656+657+658+659+660+\
661+662+663+664+665+666+667+668+669+670+671+672+673+674+675+676+677+678+679+680+\
681+682+683+684+685+686+687+688+689+690+691+692+693+694+695+696+697+698+699+700+\
701+702+703+704+705+706+707+708+709+710+711+712+713+714+715+716+717+718+719+720+\
721+722+723+724+725+726+727+728+729+730+731+732+733+734+735+736+737+738+739+740+\
741+742+743+744+745+746+747+748+749+750+751+752+753+754+755+756+757+758+759+760+\
761+762+763+764+765+766+767+768+769+770+771+772+773+774+775+776+777+778+779+780+\
781+782+783+784+785+786+787+788+789+790+791+792+793+794+795+796+797+798+799+800+\
801+802+803+804+805+806+807+808+809+810+811+812+813+814+815+816+817+818+819+820+\
821+822+823+824+825+826+827+828+829+830+831+832+833+834+835+836+837+838+839+840+\
841+842+843+844+845+846+847+848+849+850+851+852+853+854+855+856+857+858+859+860+\
861+862+863+864+865+866+867+868+869+870+871+872+873+874+875+876+877+878+879+880+\
881+882+883+884+885+886+887+888+889+890+891+892+893+894+895+896+897+898+899+900+\
901+902+903+904+905+906+907+908+909+910+911+912+913+914+915+916+917+918+919+920+\
921+922+923+924+925+926+927+928+929+930+931+932+933+934+935+936+937+938+939+940+\
941+942+943+944+945+946+947+948+949+950+951+952+953+954+955+956+957+958+959+960+\
961+962+963+964+965+966+967+968+969+970+971+972+973+974+975+976+977+978+979+980+\
981+982+983+984+985+986+987+988+989+990+991+992+993+994+995+996+997+998+999+1000)

char tab[VALUE];

int main()
{
    tab = 5;
}

gcc เอาต์พุต:

test.c: In function 'main':
test.c:56:9: error: incompatible types when assigning to type 'char[500500]' fro
m type 'int'

2

คุณสามารถใช้ (และส่วนใหญ่ละเมิด) c ++ แมโคร / แม่แบบในการทำmetaprogramming AFAIK, Java ไม่อนุญาตให้มีสิ่งเดียวกัน


2
ไม่ใช่คำตอบสำหรับคำถามจริงๆ
Ikke

ฉันคิดว่าคุณถูก. ใน java คุณไม่สามารถใช้เคล็ดลับการเรียกซ้ำเทมเพลตเดียวกันได้เนื่องจากพารามิเตอร์คลาสทั่วไปไม่สามารถเป็นค่าได้ต้องเป็นคลาส
Eyal Schneider

คุณลักษณะทั่วไปของคอมไพเลอร์ C # ช่วยให้คุณสามารถคำนวณเวลาในการคอมไพล์ได้ ดูโพสต์ของ Eric Lippertเกี่ยวกับเรื่องนี้
Allon Guralnek

1

ในทางทฤษฎีคุณสามารถใช้สิ่งนี้:

#include <iostream>

template<int N>
struct Triangle{
  static int getVal()
  {
    return N + Triangle<N-1>::getVal();
  }
};

template<>
struct Triangle<1>{
  static int getVal()
  {
    return 1;
  }
};

int main(){
   std::cout << Triangle<1000>::getVal() << std::endl;
   return 0;
}

(ตามรหัสที่ Xeo โพสต์) แต่ GCC ทำให้ฉันมีข้อผิดพลาดนี้:

triangle.c++:7: error: template instantiation depth exceeds maximum of 500 (use -ftemplate-depth-NN to increase the maximum) instantiating struct Triangle<500>

บวก pseudo-stacktrace มหาศาล


ต้องใช้ธง: -ftemplate-depth-1000
Jetti

@hsalimi: ใช่ นอกจากนี้ยังใช้ได้กับ 1,000 เมื่อคุณเพิ่มค่าสถานะ แต่มันไม่ได้พิมพ์ตามเวลาคอมไพล์และ Xeo ได้เปลี่ยนคำตอบของเขาเพื่อตอบปัญหานี้โดยเฉพาะดังนั้นฉันคิดว่าคำตอบของฉันล้าสมัย :-)
ruakh

1

การใช้ java คุณสามารถทำสิ่งที่คล้ายกันกับคำตอบ C #:

public class Cheat {
    public static final int x = (1000 *1001/2);
}

javac -Xprint Cheat.java

public class Cheat {

  public Cheat();
  public static final int x = 500500;
}

คุณสามารถทำได้ในสกาล่าโดยใช้หมายเลขพีโนเพราะคุณสามารถบังคับให้คอมไพเลอร์ทำการเรียกซ้ำได้ แต่ฉันไม่คิดว่าคุณจะทำสิ่งเดียวกันใน c # / java ได้

วิธีแก้ปัญหาอื่นที่ไม่ใช้ -Xprint แต่ยิ่งหลบ

public class Cheat {
  public static final int x = 5/(1000 *1001/2 - 500500);
}

javac -Xlint:all Cheat.java

Cheat.java:2: warning: [divzero] division by zero
  public static final int x = 5/(1000 *1001/2 - 500500);
                            ^
1 warning

โดยไม่ต้องใช้แฟล็กคอมไพเลอร์ใด ๆ เนื่องจากคุณสามารถตรวจสอบจำนวนค่าคงที่โดยพลการ (ไม่ใช่แค่ 500500) วิธีนี้ควรเป็นที่ยอมรับได้

public class Cheat {
  public static final short max = (Short.MAX_VALUE - 500500) + 1001*1000/2;
  public static final short overflow = (Short.MAX_VALUE - 500500 + 1) + 1001*1000/2;

}

Cheat.java:3: error: possible loss of precision
  public static final short overflow = (Short.MAX_VALUE - 500500 + 1) + 1001*1000/2;
                                                                  ^
  required: short
  found:    int
1 error

คุณไม่ได้ขับคอมไพเลอร์เพื่อคำนวณ 500500ขออภัย
Xeo

1
นี่อ้างอิงถึงโซลูชันทั้งสามหรือไม่ ในวิธีแก้ปัญหา 1 ฉันเอาโค้ด java มาคอมไพล์แล้วคอมไพเลอร์ก็พิมพ์ออกมา 500500 มันดูเหมือนคอมไพเลอร์ที่คำนวณ 500500 นี่ไม่ใช่คอมไพเลอร์ที่คำนวณ 500500 ได้อย่างไร
benmmurphy

ใช่จริงพอแล้วฉันกำลังพูดถึงวิธีแก้ปัญหา 2 และ 3 ฉันอ่านคำตอบนี้แล้วในการอัปเดตก่อนหน้านี้และกลับมาที่คำตอบใหม่ล่าสุดและดูเหมือนว่าจะลืมวิธีแก้ปัญหาแรกไปแล้ว
Xeo

ฉันจะบอกว่าโซลูชัน 2 และ 3 กำลังคำนวณเช่นกัน for (i = 0; i < 100000; ++i) {if (i == 1000*1000/2) print i}คุณสามารถเพิ่มจำนวนข้อของการตรวจสอบเพื่อให้คุณมีพื้นทำ ฉันมีไฟล์ java ขนาด 160MB ที่ใช้งานได้ :)
benmmurphy

1

แม้ว่าจะใช้งานได้จริงกับตัวเลขขนาดเล็ก แต่เสียงดังกล่าว ++ จะส่งกลับข้อผิดพลาดของคอมไพเลอร์ให้ฉันหากฉันใช้ sum_first โดยที่ N> 400

#include <iostream>

using namespace std;


template <int N>
struct sum_first
{
   static const int value = N + sum_first<N - 1>::value;
};

template <>
struct sum_first<0>
{
    static const int value = 0;
};

int main()
{
    cout << sum_first<1000>::value << endl;
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.