วิธีการเชื่อม std :: string และ int เข้าด้วยกัน?


655

ฉันคิดว่ามันจะง่ายจริงๆ แต่ก็มีปัญหาบางอย่าง ถ้าฉันมี

std::string name = "John";
int age = 21;

ฉันจะรวมพวกเขาเพื่อรับสายเดียวได้"John21"อย่างไร


Sutter สมุนไพรมีบทความที่ดีในเรื่องนี้: "สตริง formatters ของฟาร์มแมเนอร์" เขาครอบคลุมBoost::lexical_cast, std::stringstream, std::strstream(ซึ่งเลิกใช้แล้ว) และเมื่อเทียบกับsprintf snprintf
Fred Larson

ให้ฉันเพิ่มลงในสิ่งนี้: ฉันพยายาม 'str = "hi"; STR + = 5; ศาล << str; ' และไม่เห็นผลใด ๆ ปรากฎตัวดำเนินการการโทรนี้ + = (อักขระถ่าน) และเพิ่มอักขระที่ไม่สามารถพิมพ์ได้
daveagp

คำตอบ:


1126

ตามลำดับตัวอักษร:

std::string name = "John";
int age = 21;
std::string result;

// 1. with Boost
result = name + boost::lexical_cast<std::string>(age);

// 2. with C++11
result = name + std::to_string(age);

// 3. with FastFormat.Format
fastformat::fmt(result, "{0}{1}", name, age);

// 4. with FastFormat.Write
fastformat::write(result, name, age);

// 5. with the {fmt} library
result = fmt::format("{}{}", name, age);

// 6. with IOStreams
std::stringstream sstm;
sstm << name << age;
result = sstm.str();

// 7. with itoa
char numstr[21]; // enough to hold all numbers up to 64-bits
result = name + itoa(age, numstr, 10);

// 8. with sprintf
char numstr[21]; // enough to hold all numbers up to 64-bits
sprintf(numstr, "%d", age);
result = name + numstr;

// 9. with STLSoft's integer_to_string
char numstr[21]; // enough to hold all numbers up to 64-bits
result = name + stlsoft::integer_to_string(numstr, 21, age);

// 10. with STLSoft's winstl::int_to_string()
result = name + winstl::int_to_string(age);

// 11. With Poco NumberFormatter
result = name + Poco::NumberFormatter().format(age);
  1. ปลอดภัย แต่ช้า ต้องใช้Boost (ส่วนหัวเท่านั้น) แพลตฟอร์มส่วนใหญ่ / ทั้งหมด
  2. ปลอดภัยต้องใช้ C ++ 11 ( to_string ()รวมอยู่ใน#include <string>)
  3. ปลอดภัยและรวดเร็ว ต้องการFastFormatซึ่งจะต้องรวบรวม แพลตฟอร์มส่วนใหญ่ / ทั้งหมด
  4. ( เหมือนกัน )
  5. ปลอดภัยและรวดเร็ว ต้องการไลบรารี {fmt}ซึ่งสามารถรวบรวมหรือใช้ในโหมดส่วนหัวเท่านั้น แพลตฟอร์มส่วนใหญ่ / ทั้งหมด
  6. ปลอดภัยช้าและ verbose; ต้องการ#include <sstream>(จาก C ++ มาตรฐาน)
  7. เปราะ (คุณต้องระบุบัฟเฟอร์ที่ใหญ่พอ) รวดเร็วและ verbose itoa () เป็นส่วนเสริมที่ไม่ได้มาตรฐานและไม่รับประกันว่าจะมีในทุกแพลตฟอร์ม
  8. เปราะ (คุณต้องระบุบัฟเฟอร์ที่ใหญ่พอ) รวดเร็วและ verbose ไม่ต้องการอะไร (เป็นมาตรฐาน C ++); แพลตฟอร์มทั้งหมด
  9. เปราะ (คุณต้องระบุบัฟเฟอร์ที่ใหญ่พอ) อาจเป็นการแปลงที่เร็วที่สุดที่เป็นไปได้ verbose ต้องใช้STLSoft (ส่วนหัวเท่านั้น) แพลตฟอร์มส่วนใหญ่ / ทั้งหมด
  10. safe-ish (คุณไม่ได้ใช้มากกว่าหนึ่งสาย int_to_string ()ในคำสั่งเดียว) รวดเร็ว; ต้องใช้STLSoft (ส่วนหัวเท่านั้น) ใช้ Windows เท่านั้น
  11. ปลอดภัย แต่ช้า ต้องการPoco C ++ ; แพลตฟอร์มส่วนใหญ่ / ทั้งหมด

13
นอกเหนือจากลิงค์เดียวที่คุณให้มาคุณมีความคิดเห็นเกี่ยวกับประสิทธิภาพของอะไร
JamieH

2
เกือบจะเป็นชื่อเสียงทั้งหมดของคุณจากคำตอบเดียว !! คุณโชคดีที่ถั่ว;) ฉันคิดว่า 8 เป็นมาตรฐาน C (แน่นอนเช่นเดียวกันกับ C ++) แต่อาจมีความแตกต่างที่คุ้มค่า
noelicus

2. ช้าเนื่องจาก std :: to_string (อายุ) สร้างสตริงชั่วคราวที่ผนวกเข้ากับผลลัพธ์
Igor Bukanov

หากคุณใช้ Arduino คุณสามารถใช้งานString(number)ได้เช่นกัน
Machado


85

ถ้าคุณมี Boost boost::lexical_cast<std::string>(age)คุณสามารถแปลงจำนวนเต็มสตริงโดยใช้

อีกวิธีคือใช้ stringstreams:

std::stringstream ss;
ss << age;
std::cout << name << ss.str() << std::endl;

วิธีที่สามคือการใช้sprintfหรือsnprintfจากไลบรารี C

char buffer[128];
snprintf(buffer, sizeof(buffer), "%s%d", name.c_str(), age);
std::cout << buffer << std::endl;

โปสเตอร์อื่น ๆ itoaแนะนำให้ใช้ นี่ไม่ใช่ฟังก์ชั่นมาตรฐานดังนั้นรหัสของคุณจะไม่สามารถพกพาได้หากคุณใช้ มีคอมไพเลอร์ที่ไม่รองรับ


โปรดทราบว่า snprintf ไม่รับประกันว่าจะสิ้นสุดสตริง นี่คือวิธีหนึ่งในการตรวจสอบให้แน่ใจว่ามันใช้งานได้: <pre> char buffer [128] buffer [sizeof (buffer) -1] = '\ 0'; snprintf (บัฟเฟอร์, ขนาดของ (บัฟเฟอร์) -1, "% s% d", name.c_str (), อายุ); std :: cout << buffer << std :: endl; </pre>
Mr Fooz

แนวโน้มของฉันจะไม่ใช้ sprintf เนื่องจากอาจทำให้เกิดบัฟเฟอร์ล้น ตัวอย่างด้านบนเป็นตัวอย่างที่ดีที่การใช้ sprintf จะไม่ปลอดภัยหากชื่อยาวมาก
terson

โปรดทราบว่า snprintf นั้นไม่ได้มาตรฐานเท่ากัน c ++ (เช่น itoa ที่คุณพูดถึง) มันนำมาจาก c99
Johannes Schaub - litb

@terson: ผมเห็นการเกิดขึ้นของไม่มีในคำตอบเท่านั้นsprintf snprintf
David Foerster


52
#include <iostream>
#include <string>
#include <sstream>
using namespace std;
string itos(int i) // convert int to string
{
    stringstream s;
    s << i;
    return s.str();
}

ขโมยมาจากhttp://www.research.att.com/~bs/bs_faq2.htmlอย่างไร้ยางอาย


แต่s เป็นตัวแปรสแต็ค, ความทรงจำของจะได้รับฟรีหลังจากวิงวอนs ควรจัดสรรจากฮีปและหลังจากใช้ใช่ไหม itossfree
kgbook

1
ผลตอบแทนตามมูลค่าก็โอเคแม้ว่าวัตถุสตริงจะออกนอกขอบเขตstackoverflow.com/a/3977119/5393174
kgbook


23

หากคุณมี C ++ 11 std::to_stringคุณสามารถใช้

ตัวอย่าง:

std::string name = "John";
int age = 21;

name += std::to_string(age);

std::cout << name;

เอาท์พุท:

John21

2
มันจะอยู่name += std::to_string(static_cast<long long>(age));ใน VC ++ 2010 ตามที่คุณเห็นที่นี่
neonmate

@neonmate วิธีการเกี่ยวกับname += std::to_string(age + 0LL);แทน?
chux - Reinstate Monica

18

ฉันคิดว่าคำตอบที่ง่ายที่สุดคือการใช้sprintfฟังก์ชั่น:

sprintf(outString,"%s%d",name,age);

1
snprintf อาจเป็นเรื่องยุ่งยาก (ส่วนใหญ่เป็นเพราะมันอาจจะไม่รวมถึงตัวละครโมฆะในบางสถานการณ์) แต่ฉันชอบที่จะหลีกเลี่ยงบัฟเฟอร์ sprintf ล้นปัญหาที่อาจเกิดขึ้น
terson

3
sprintf (char *, const char *, ... ) จะล้มเหลวในคอมไพเลอร์บางเวอร์ชันเมื่อคุณส่งสตริง std :: ไปยัง% s ไม่ใช่ทุกอย่าง (มันเป็นพฤติกรรมที่ไม่ได้กำหนด) และอาจขึ้นอยู่กับความยาวของสตริง (SSO) โปรดใช้. c_str ()
MSalters

บวก sprintf อาจมีการบัฟเฟอร์มากเกินไปการฉีดโค้ดที่เป็นไปได้
Jean-François Fabre


11
#include <sstream>

template <class T>
inline std::string to_string (const T& t)
{
   std::stringstream ss;
   ss << t;
   return ss.str();
}

จากนั้นการใช้งานของคุณจะมีลักษณะเช่นนี้

   std::string szName = "John";
   int numAge = 23;
   szName += to_string<int>(numAge);
   cout << szName << endl;

Googled [และทดสอบ: p]


10

ปัญหานี้สามารถทำได้หลายวิธี ฉันจะแสดงสองวิธี:

  1. to_string(i)แปลงหมายเลขที่สตริงใช้

  2. การใช้สตรีมสตริง

    รหัส:

    #include <string>
    #include <sstream>
    #include <bits/stdc++.h>
    #include <iostream>
    using namespace std;
    
    int main() {
        string name = "John";
        int age = 21;
    
        string answer1 = "";
        // Method 1). string s1 = to_string(age).
    
        string s1=to_string(age); // Know the integer get converted into string
        // where as we know that concatenation can easily be done using '+' in C++
    
        answer1 = name + s1;
    
        cout << answer1 << endl;
    
        // Method 2). Using string streams
    
        ostringstream s2;
    
        s2 << age;
    
        string s3 = s2.str(); // The str() function will convert a number into a string
    
        string answer2 = "";  // For concatenation of strings.
    
        answer2 = name + s3;
    
        cout << answer2 << endl;
    
        return 0;
    }

อันไหนเร็วกว่ากัน?
GyuHyeon Choi

7

หากคุณต้องการใช้+สำหรับตัดแบ่งสิ่งที่มีตัวดำเนินการเอาท์พุทคุณสามารถระบุเทมเพลตเวอร์ชันของoperator+:

template <typename L, typename R> std::string operator+(L left, R right) {
  std::ostringstream os;
  os << left << right;
  return os.str();
}

จากนั้นคุณสามารถเขียนการต่อข้อมูลของคุณได้อย่างตรงไปตรงมา:

std::string foo("the answer is ");
int i = 42;
std::string bar(foo + i);    
std::cout << bar << std::endl;

เอาท์พุท:

the answer is 42

นี่ไม่ใช่วิธีที่มีประสิทธิภาพมากที่สุด แต่คุณไม่ต้องการวิธีที่มีประสิทธิภาพมากที่สุดเว้นแต่ว่าคุณกำลังทำการต่อหลายอย่างเข้าด้วยกัน


หากฉันพยายามเพิ่มจำนวนเต็มหรือจำนวนเต็มและสองเท่าฟังก์ชันนี้จะถูกเรียกใช้หรือไม่ ผมสงสัยว่าถ้าแก้ปัญหานี้จะแทนที่การเพิ่มปกติ ...
Hilder Vitor ลิมารา

ตัวดำเนินการส่งคืน a std::stringดังนั้นจะไม่เป็นตัวเลือกในนิพจน์ที่สตริงไม่สามารถแปลงเป็นประเภทที่ต้องการได้ เช่นนี้operator+ไม่ได้มีสิทธิ์ที่จะใช้สำหรับใน+ int x = 5 + 7;ทุกสิ่งที่พิจารณาแล้วฉันจะไม่นิยามผู้ประกอบการเช่นนี้โดยไม่มีเหตุผลที่น่าสนใจมากแต่เป้าหมายของฉันคือการเสนอคำตอบที่แตกต่างจากที่อื่น
uckelman

คุณถูก (ฉันเพิ่งทดสอบมัน ... ) และเมื่อฉันพยายามทำบางสิ่งเช่นสตริง s = 5 + 7ฉันได้รับข้อผิดพลาดในการแปลงที่ไม่ถูกต้องจาก 'int' เป็น 'const char ' *
Hilder Vitor Lima Pereira

5

ถ้าคุณใช้ MFC คุณสามารถใช้ CString

CString nameAge = "";
nameAge.Format("%s%d", "John", 21);

จัดการ c ++ นอกจากนี้ยังมี การจัดรูปแบบสตริง


4

std :: ostringstream เป็นวิธีการที่ดี แต่บางครั้งเคล็ดลับเพิ่มเติมนี้อาจทำให้การเปลี่ยนรูปแบบเป็นแบบหนึ่งซับ:

#include <sstream>
#define MAKE_STRING(tokens) /****************/ \
    static_cast<std::ostringstream&>(          \
        std::ostringstream().flush() << tokens \
    ).str()                                    \
    /**/

ตอนนี้คุณสามารถจัดรูปแบบสตริงเช่นนี้:

int main() {
    int i = 123;
    std::string message = MAKE_STRING("i = " << i);
    std::cout << message << std::endl; // prints: "i = 123"
}

4

เนื่องจากคำถามที่เกี่ยวข้องกับ Qt ถูกปิดลงในความโปรดปรานของคำถามนี้ต่อไปนี้เป็นวิธีการใช้ Qt:

QString string = QString("Some string %1 with an int somewhere").arg(someIntVariable);
string.append(someOtherIntVariable);

ตอนนี้ตัวแปรสตริงมีค่า someIntVariable แทน% 1 และค่า someOtherIntVariable ในตอนท้าย


QString ("Something") + QString :: number (someIntVariable) สามารถใช้ได้เช่นกัน
gremwell

3

มีตัวเลือกเพิ่มเติมที่เป็นไปได้ที่จะใช้เชื่อมต่อจำนวนเต็ม (หรือวัตถุตัวเลขอื่น ๆ ) กับสตริง มันเป็นBoost.Format

#include <boost/format.hpp>
#include <string>
int main()
{
    using boost::format;

    int age = 22;
    std::string str_age = str(format("age is %1%") % age);
}

และกรรมจากBoost.Spirit (v2)

#include <boost/spirit/include/karma.hpp>
#include <iterator>
#include <string>
int main()
{
    using namespace boost::spirit;

    int age = 22;
    std::string str_age("age is ");
    std::back_insert_iterator<std::string> sink(str_age);
    karma::generate(sink, int_, age);

    return 0;
}

Boost.Spirit Karma อ้างว่าเป็นหนึ่งในตัวเลือกที่เร็วที่สุดสำหรับการแปลงจำนวนเต็มเป็นสตริง



3

คุณสามารถเชื่อม int กับ string ได้โดยใช้กลวิธีง่ายๆด้านล่าง แต่โปรดทราบว่าวิธีนี้จะใช้ได้เมื่อจำนวนเต็มเป็นตัวเลขหลักเดียวเท่านั้น มิฉะนั้นให้เพิ่มจำนวนเต็มเป็นตัวเลขในสตริงนั้น

string name = "John";
int age = 5;
char temp = 5 + '0';
name = name + temp;
cout << name << endl;

Output:  John5

2

ต่อไปนี้เป็นการนำวิธีการผนวก int ไปยังสตริงโดยใช้การแยกวิเคราะห์และจัดรูปแบบ facets จากไลบรารี IOStreams

#include <iostream>
#include <locale>
#include <string>

template <class Facet>
struct erasable_facet : Facet
{
    erasable_facet() : Facet(1) { }
    ~erasable_facet() { }
};

void append_int(std::string& s, int n)
{
    erasable_facet<std::num_put<char,
                                std::back_insert_iterator<std::string>>> facet;
    std::ios str(nullptr);

    facet.put(std::back_inserter(s), str,
                                     str.fill(), static_cast<unsigned long>(n));
}

int main()
{
    std::string str = "ID: ";
    int id = 123;

    append_int(str, id);

    std::cout << str; // ID: 123
}

2
  • มาตรฐาน :: ostringstream
#include <sstream>

std::ostringstream s;
s << "John " << age;
std::string query(s.str());
  • std :: to_string (C ++ 11)
std::string query("John " + std::to_string(age));
  • เพิ่ม :: lexical_cast
#include <boost/lexical_cast.hpp>

std::string query("John " + boost::lexical_cast<std::string>(age));

อันไหนเร็วที่สุด?
GyuHyeon Choi

2

มีฟังก์ชั่นที่ฉันเขียนซึ่งใช้หมายเลข int เป็นพารามิเตอร์และแปลงเป็นสตริงตามตัวอักษร ฟังก์ชั่นนี้ขึ้นอยู่กับฟังก์ชั่นอื่นที่แปลงตัวเลขหนึ่งหลักเป็นค่าเทียบเท่า:

char intToChar(int num)
{
    if (num < 10 && num >= 0)
    {
        return num + 48;
        //48 is the number that we add to an integer number to have its character equivalent (see the unsigned ASCII table)
    }
    else
    {
        return '*';
    }
}

string intToString(int num)
{
    int digits = 0, process, single;
    string numString;
    process = num;

    // The following process the number of digits in num
    while (process != 0)
    {
        single  = process % 10; // 'single' now holds the rightmost portion of the int
        process = (process - single)/10;
        // Take out the rightmost number of the int (it's a zero in this portion of the int), then divide it by 10
        // The above combination eliminates the rightmost portion of the int
        digits ++;
    }

    process = num;

    // Fill the numString with '*' times digits
    for (int i = 0; i < digits; i++)
    {
        numString += '*';
    }


    for (int i = digits-1; i >= 0; i--)
    {
        single = process % 10;
        numString[i] = intToChar ( single);
        process = (process - single) / 10;
    }

    return numString;
}

1

ด้วยห้องสมุด {fmt} :

auto result = fmt::format("{}{}", name, age);

ส่วนย่อยของไลบรารีถูกเสนอเพื่อสร้างมาตรฐานเป็นP0645 การจัดรูปแบบข้อความและถ้ายอมรับจะเป็น:

auto result = std::format("{}{}", name, age);

คำเตือน : ฉันเป็นผู้เขียนของห้องสมุด {fmt}


0

ในฐานะที่เป็นหนึ่งซับ: name += std::to_string(age);


3
นั่นคือรหัสเดียวกันใน 0x499602D2 คำตอบ
BDL
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.