質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.46%
C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

Q&A

解決済

4回答

1260閲覧

C言語でSegmentation fault発生

pokemonta

総合スコア170

C

C言語は、1972年にAT&Tベル研究所の、デニス・リッチーが主体となって作成したプログラミング言語です。 B言語の後継言語として開発されたことからC言語と命名。そのため、表記法などはB言語やALGOLに近いとされています。 Cの拡張版であるC++言語とともに、現在世界中でもっとも普及されているプログラミング言語です。

0グッド

0クリップ

投稿2021/04/27 04:09

編集2021/04/27 04:28

8byteの文字列の初期化を試みています。
各バイトをFF(16進)で埋めるためにunsigned charにキャストしています。
(charでは、-128 ~ 127しか表現できないため)
その後、memsetで全バイトを0xffで埋めているのですがうまくいきません
何が原因でどのような対処方法がありございますでしょうか

リンク内容

c

1 2void main(void) 3{ 4 char str[] ="abcdefte"; 5 set_str(str); 6 return; 7} 8 9void set_str(char *str) 10{ 11 unsigned char uchr; 12 uchr = (unsigned char)str; 13 memset( uchr, 0xff, sizeof(str)); 14} 15

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

maisumakun

2021/04/27 04:55

> "abcdefte"は、文字列リテラルといって変更不可です。 ではないです。ここではchar[]の初期化子なので、単なる配列として書き換え可能です。
guest

回答4

0

ベストアンサー

いろいろおかしいです。

C

1#include <string.h> 2 3void set_str(char *str, size_t size) 4{ 5 unsigned char *uchr; 6 uchr = (unsigned char *)str; 7 memset( uchr, 0xff, size); 8 9 // キャストせずに下記でも問題なし。というか、普通はこう。 10 // memset( str, 0xff, size); 11 12} 13 14int main(void) 15{ 16 char str[] ="abcdefte"; 17 set_str(str, sizeof str); 18 return 0; 19}

memsetの第一引数はポインターなのにポインターを渡してない(セグメントフォールトの直接原因)
・関数に引数で渡された配列のサイズは関数側では不明なので、呼び出し側から渡す必要あり (sizeof(str)はポインターのサイズを表す)
mainの返り値はint

投稿2021/04/27 04:47

編集2021/04/27 04:53
otn

総合スコア84798

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

pokemonta

2021/04/27 04:51

私のPGM何がおかしいかったでしょうか?? 関数呼び出し側でsizeof strが必要なんでしょうか? 私は定義側でsizeof 使っています
otn

2021/04/27 04:54

おかしい点を補足しました。
pokemonta

2021/04/27 04:58

>関数に引数で渡された配列のサイズは関数側では不明なので、呼び出し側から渡す必要あり あ、そうなんですね。 文字列(abcdefte)をそのまま渡しているので 関数側は、それをカウントすればいけるのかな~と思ってました。 pythoんはいけるはず
otn

2021/04/27 05:02

> 文字列(abcdefte)をそのまま渡しているので いいえ。 渡しているのは、「文字列の先頭アドレス」だけです。 > pythoんはいけるはず C以外の多くの言語では、文字列自体が長さの属性を持っています。
pokemonta

2021/04/27 08:21

>渡しているのは、「文字列の先頭アドレス」だけです。 ええ、知らなかったっす。。。ええ
guest

0

"abcdefte"は、文字列リテラルと言って変更不可です。
参考:“文字列リテラルとはなんですか?”https://teratail.com/questions/20564
また、sizeof(str)では、ポインタのサイズ(64ビット環境なら8)しか求まりません。

投稿2021/04/27 04:56

cateye

総合スコア6851

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

otn

2021/04/27 04:59

char *str ="abcdefte"; じゃなくて、 char str[] ="abcdefte"; なので、書き換えているのはchar配列です。
fana

2021/04/27 04:59

本件はリテラルを書き換えているわけではないです.
cateye

2021/04/27 07:03

失礼、間違いました。m(_"_)m
guest

0

uchr = (unsigned char)str;

これは何をしたいのでしょう?

strは char*型であり,おそらくは32bitとか64bitとかの値です.
unsigned charが8bitだとしたら,これをunsigned charにしろと言われても当然入りきらないですから,下位8bit分しか残らない結果になるでしょう.

そうして著しく小さい値に切り詰められてしまったであろう値を memset にアドレスとして与えているとしたら,まぁ,Segmentation fault になり得るでしょう.


あと,それとは別に

memset( uchr, 0xff, sizeof(str));

これの最後の引数も記述として間違っているでしょう.(たまたま 「8byte」 という値として一致することになり得るかもしれないけれども)

投稿2021/04/27 04:53

編集2021/04/27 04:54
fana

総合スコア11708

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

C

1void set_str(char *str, size_t n) 2{ 3 unsigned char* uchr; 4 uchr = (unsigned char*)str; 5 memset( uchr, 0xff, n); 6}

uchrをポインタにしました。これがエラーと関係あります。
引数に文字列の長さを追加、これは今回のエラーとは関係ありません。ただ無いと別の不具合が出ます。

そして、そもそもですが

各バイトをFF(16進)で埋めるためにunsigned charにキャストしています。

キャストしなくてもmemsetできます。

投稿2021/04/27 04:50

ozwk

総合スコア13532

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.46%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問