2019,江端さんの忘備録

(昨日の続きです)

(Continuation from yesterday)

思想家や哲学者の名前くらいしか知らない私にとっては、それを判断する材料がありません。

For me who only knows the names of thinkers and philosophers, there is no way how to judge true or false.

相手を論破するには、その思想家や哲学者の原書について、徹底的に学習しなければなりません。

In order to refute the opponent, I must thoroughly learn about the original book of the thinker and philosopher.

が、そもそも、私は、そういうものには興味がありませんでした。

But in the first place, I was not interested in such things.

それに、その当時から、

Besides, from that time,

―― 社会を良くしようと志向する者(哲学者)が、大衆に理解できない本を書いてどうする

と思っていましたし、

"How does a person (philosopher) who intends to improve society write a book that the public can not understand?"

「そういうスタンスの哲学者の著書など、私が苦労して理解してやる必要はない」とさえ思っていました。

I thought "I don't have to work hard to understand such a philosopher's book on such a stance,"

-----

そこに行くと、技術(特に工学)の世界は、明快です。

On the other hand, the world of technology (especially engineering) is clear.

―― 当初の予定通り動いたモノだけが正解であり、動かすことができた人間だけが正義

"The only thing that moved as originally planned is the correct answer, and only the person who was able to move is justice"

という絶対的な価値基準があります。

This is an absolute value standard.

ですから、どんなに偉そうに技術を語ろうとも、(少くとも私は)、モノを動かせない人間は、1mmも信用しません。

So, no matter how well they speak technology, at least I can not trust anyone who can not move things.

特にコンセプトだけ語って、パワーポイントを量産するだけの奴は、私の記憶の中には残りません ―― で、人工知能技術の分野で、こういう手合いが多い。

Especially, a person who speak a concept only and mass-produce the power point, is out of my scope. there are many such people in the field of artificial intelligence technology.

私は、100枚のパワーポイントを量産する人間より、自分の手で100行の動くプログラムを書く人間を、信用します。

I trust people who write 100-line executive programs with their own hands, rather than those who mass-produce 100 power points.

実際にモノを作る人の声は、有無を言わせぬ説得力があり、かつ、そのようにして作られたモノは、暴力的なまでに強い。

The voice of those who actually make things is very persuasive, and things that are made in this way are strong enough to be violent.

私は、その動くモノを見て、その中身を検証し、そのモノに込められた技術力に驚嘆するのです。

I look at the moving things, examine their contents, and marvel at the technological powers put into them.

-----

もっとも、私は「コンセプト」を作る人間をバカにしている訳ではありません。

However, I am not a fool of the person who makes the "concept".

「コンセプト」は、それが、実行が状態直前と言えるほど、具体的で現実性のあるならば、それは「モノ作り」と同じと思っています。

The "concept" is, I think that is the same as "making things", If it is so concrete and realistic that it is an immediate execution possible state.

その「コンセプト」を、現実に実行可能状態にして見せて貰えれば、私は、その人に心酔し、盲従すると思います。

If they show the concept in a practical state, I think that I will be obsessed and obsessed with them.

それとは逆に、単なるパワーポイント作成者に留まっている人間を、私は「コンセプト屋」と呼んで、蔑んでいます。

On the contrary, I don't like a person who is just a PowerPoint maker with calling "concept shop".

残念ながら、私の人生においては、「コンセプト屋」(その他、「標準化屋」とか「ビジネルモデル屋」とか)を越える人に、出会ったことがありません。

Unfortunately, in my life, I have never met people beyond "concept shop" (others, "standardizing shop", "business model shop", etc.).

今のところ、「モノ作り」以外のことで、この私を、心酔、盲従させた人間は、一人もいないのです。

For now, other than "making things", there is no one who has made me insane, blind and obedient.

2019,江端さんの忘備録

(昨日の続きです)

(Continuation from yesterday)

しかし、私が、その自治寮で行われる勉強会で、本当に困ったことは、彼らの「文章能力のおそまつさ」でした。

However, what I felt really troubled at the study sessions held at the dormitory was their "sorry writing ability".

私は理系の学生でしたので、コンセプトを図や数字で表わすことの必要性は、よく分かっていたのですが、寮生のプレゼンの酷さは、もう、絶望的なほどでした。

Since I was a science student, I knew well the necessity of representing the concept with figures and numbers. The badness of the dormitory's presentation was hopeless.

私の、物事を図面入りでロジカルに説明する、レポート形式の総括文は、

A summary statement in my report form that explains things in a complete and logical way, was just criticized as

「安直な図式主義」

"Easy Schemeism"

と批判されたくらいです。

もっとも今の私ならば、

Now I can say

『自分の主張を、人に説明できる程度に纏めて、ロジカルに説明できない、お前の頭が悪すぎるんだ』

"Your head is too bad to summarize your claim and explain it to others"

と断言できますが、当時の私は『そういうものかな』と思ったりもしました。

However at the time, I could not know which one is correct.

今の私ば、奴らの知性の低さに、1mmの疑義もありませんが。

Now there is no doubt of their ignorance.

-----

で、当時の自治寮で、この「頭の悪い文章」がまかり通ったのは、いわゆる哲学者とか思想家とか言われる者たちの、「訳の分からん原書」の影響があったのではないか、と、私は考察する訳です。

I will consider that at the time of the self-governing dormitory, the reason of this "innocent writing" passed through, was the influence of "the translation of the original book" by philosophers and thinkers.

「訳の分からん文章」を書くことが正しい ―― のような、アホな考え方の根底には、

"Unreasonable writing" is right --- at the bottom of this foolish thought, I think there are two possibilities.

(1)テクニカルライティング(ロジカルライティング)の能力が絶無

(1)There is no technical writing (logical writing) ability at all

(2)もし、上記(1)の手法で記載すると、自分の考えの底の浅さが簡単に露呈してしまう

(2)If it is described in the above method (1), the shallowness of the bottom of their idea is easily exposed

のいずれかがあっただろうと思います。

いずれもしても、そういうものに付き合わされるアホらしさに気がついて、とっとと退寮した(しかも寮長在任中に)私は、実に賢明だったと思うのです。

Anyway, I think I was really wise. Because I noticed that I was being stunned with such things, and I left the room soon. even I was a leader of the dormitory.

私が上記の自治寮での議論の中で、彼らが濫用する思想家や哲学者の引用を聞く度に、

In the above discussion in the dormitory, whenever I heard a quote from a thinker or philosopher they abuse, I thought

―― おまえ、本当にこの原書の内容を理解した上で、話をしているのか?

"Are you really talking after understanding the contents of this original book?"

と思っていました。

(続く)

(To be continued)

2019,江端さんの忘備録

(昨日の続きです)

(Continuation from yesterday)

私、学生の頃、いわゆる自治寮というところに住んでいました。

When I was a student, I lived in a so-called autonomous dormitory.

10年前の、学生運動の残滓の念が残っただけの、学生に対してほとんど存在感をアピールできない寮でした。

The dormitory had only the remnants of the student movement ten years ago remained, and could hardly appeal to students.

しかし、寮生の一部には、その自治寮の理念を誇りに思っている者もいました。

However, some of the dormitory students were proud of the principles of the dormitory.

そして、小さな町工場の息子であって、零細企業の悲哀を思う存分味わってきた私は、哲学者やら思想家やらの言葉を引用して、議論をする彼らとは ―― とことん、相容れませんでした。

And I was the son of a small town factory, and I had tasted the sorrows of small businesses. So I could not agree with them whom quoted the words of philosophers, thinkers and others.

■資本家による搾取だぁ? バカいってんじゃねーよ。日本の97%の企業は中小または零細企業で、その社長の殆どは、大企業のサラリーマンよりはるかに貧乏だぞ

- Is it exploitation by a capitalist? Don't be stupid. 97% of companies in Japan are small companies, and most of their presidents are much poorer than business persons of large companies.

■大企業にしても、株主監視の元、アホな経営やっていたら忽ち解任されるし、生き残る為に、やむなく従業員を切りすてるという選択を行っているだけだ

- Even a large company, under the supervision of shareholders, under the supervision of shareholders, if management is doing bad management, they will be dismissed at the moment. In order to survive, they simply make the choice to lay off their employees.

とまあ、その当時にあって、不渡りの連鎖で、取引先の夫婦が首吊り自殺をして、父の会社も倒産直前まで至った、その社長の息子は ――

Well, at that time, the son of that president whose father's business partners were hanging suicide and father's company also reached just before bankruptcy in the chain of failure,

薄ら笑いを浮べながら、彼らのアホらしい空虚な議論を聞き流していました。

While making me laugh, he was listening to their ridiculously empty arguments.

(続く)

(To be continued)

2022/10,江端さんの忘備録

私は、数式を『数式のまま理解する能力』がありません。

I do not have the ability to 'understand formulas as they are'.

基本的には、その数式に数値を当てはめて、数字や図形にして、やっと『理解したような気になる』ことができます。

Basically, I can finally 'feel like I understand' by applying numerical values to those formulas and turning them into numbers and figures.

いや・・・正確に言うと、「図形」でもダメですね ―― 自分の身の回りに関する数値に落し込んで、ようやく分かったような気になります。

No... to be precise, not even "figures" -- I feel like I finally understand it when I drop it into figures related to my immediate surroundings.

上記のモデルの具体例を求めて、昨夜、YouTubeでインドの大学の講義を聞いて、具体的な数値をパクって、エクセルに落し込んでみました。

In search of a concrete example of the above model, I listened to a lecture at an Indian university on YouTube last night, took down the specific numbers, and dropped them into Excel.

グラフが出てきたの見て、ようやくホッとしました。

I was finally relieved when I saw the graphs.

-----

あるいは、

Or, about the following formula,

なども、訳が分からずにプログラムに組み込むと、後で酷い目になるので、とりあえずイメージを掴んでおきたいと思っていました。

I wanted to get an idea of what it would be like to incorporate mathematical formulas into a program without understanding them, because it would be terrible later on.

ただ、この式は入力変数が2つになるので、3次元表示をさせなければなりませんが、エクセルは3次元表示のバラエティが十分にありません。

However, since this formula has two input variables, it must be displayed in three dimensions, but Excel does not have enough variety in the three-dimensional display.

どうしようかなーーーー、と思っていたところ、"Gnuplot"のことを思い出しました。

I was wondering what to do and then I remembered "Gnuplot".

数年振りに動かしてみたところ、数式1行(事前設定ゼロ)で、この図が出てきました。

I ran it for the first time in several years and got this figure with one equation line (zero preconfiguration).

『やっぱり、Gunplotは凄いなぁ』と久々に感動してしまいました。

I was moved again with saying "Gunplot is great"

2022/10,江端さんの忘備録

『もはや高学歴、大企業就職の時代ではない』 ―― と、私が入社する頃、つまり30年前に言われていました。

When I joined the company, 30 years ago, I was told, "This is no longer the era of high education and big company employment".

ちょっと調べたら、70年の安保闘争の時代に、特に多く言われていました。

I did a little research and found that a lot of it was said, especially during the 70-year security struggle.

因みに、海軍の予備役で終戦を迎えた父から、『"この言葉"を信じて進学をしなかった』と聞いたことがあります。

Incidentally, my father, who was a reserve officer in the Navy when the war ended, once told me that he did not go on to higher education because he believed in "these words.

-----

という訳で、

Therefore, the phrase of

『もはや高学歴、大企業就職の時代ではない』

"this is no longer the era of high education and big company employment"

というセリフは、戦後80年間近く、ずっと言われ続けてきたことになります。

has been said for nearly 80 years.

-----

では、

If so,

―― 『高学歴、大企業就職の時代』がなくなるのは、一体、いつなのでしょう?

When will the "era of high education and big company employment" disappear?

少なくとも、「シグナリング理論」が完全否定されるまでは、続くのではないかなぁ、と、私は考えています。

I suspect that it will continue, at least until the "signaling theory" is completely debunked.

あるいは、『高学歴』『大企業』に取って代わる、新しいシグナルが登場するまでは、このセリフは言われ続けるんじゃないかな、と思っています。

Or, I think this phrase will continue to be said until a new signal is introduced to replace "highly educated" and "big company".

2022/10,江端さんの忘備録

昨日の続きですが、

As I continued yesterday,

メタバース、来ますかね?

メタバースについては、現時点では、私は分からないのですが ・・・まさか、『チャット』がくるとはなぁ、しかも、こんな形の、こんなタイミングで。

As for the Metaverse, at this point, I don't know... I didn't expect a "chat" to come, and in such a form and at such a time.

-----

私、今、2つの組織に属しており、いずれの組織からも、Microsoft社のTeamsのチャットで、連絡がやってきます。

I am currently part of two organizations, both of which contact me via Microsoft Teams chat.

ですので、2台のパソコンを立ち上げて、両面待ちで、常に待機している状態です。

So, I have two computers up and running, waiting for both sides, always on standby.

私、匿名性を濫用するSNSが嫌いで、家族用LINEを除いて、FacebookもTwitterも事実上停止していますが、まさか、「業務」と「学業」で、チャットに巻き込まれることになるとは思いませんでした。

I hate social networking service that abuse anonymity, and I have stopped both Facebook and Twitter expect for LINE for my family. I didn't expect to get involved in "business" and "academics"

このTeamsの『チャット』で分かったことなんですが、

I know the following with using this chat system,

- ガバナンスの効いた組織内(匿名性一切なし)でのチャットは、とても便利

- Chat within a well-governed organization (no anonymity at all) is very useful

- 会議と並行して、別の小会議も進められるので、マルチタスクワークに最適

- Ideal for multitasking, as you can have another small meeting going on in parallel with the meeting

- 資料の配布もリアルタイムでシェアできるので、会議を集中して進行可

- Materials can be distributed and shared in real time, allowing meetings to proceed in a focused manner.

なにより、

Moreover,

-『了解しました』の応答を、「いいね」ボタンで済ませられるコストパフォーマンスが『いいね!』

- Cost-effectiveness of "Good" button for "Good" response

です。

-----

コロナ禍 -> リモートワーク -> リモート会議 -> Teams等のチャットツールの隆盛(と同時に、メールの衰退)という流れで、こうなりました。

The Corona disaster -> remote work -> remote meetings -> the rise of chat tools such as Teams (and the decline of email at the same time) has led to this.

今後、リモートワークが少なくなり、F2Fミーティングに戻ったとしても、チャットツールは使われ続けると思います。

Even if remote work becomes less common in the future and we return to F2F meetings, chat tools will continue to be used.

-----

チャットツールなど、1990年の段階で、すでに存在していました(例えば"IP Messanger"など)が、それは「おもちゃ」の域を出ませんでした。

Chat tools already existed as early as 1990 (e.g., "IP Messanger"), but they were little more than "toys".

様々な経緯を経て、今、ここに、ビジネスツールとして完成しているチャットツールを見ると、

After all these different paths, now, here we are, looking at a chat tool that has been perfected as a business tool.

『私の未来予想が外れるのは当然だ』

"No wonder my future predictions are off"

と、胸を張って「開き直れる」のです。

I become defiant with saying that.

2022/10,江端さんの忘備録

メタバース、来ますかね?

Metaverse, will it come?

私、昔、『これは来る!』と公言して、思いっきり外した技術やトレンドが ―― 本当に『山ほど』ありますので、基本的に発言を控えております。

I have a lot of techs and trends that I used to say "It's coming!" but "It had not come", so basically I refrain from making statements.

私ができるのは、「分析」ぐらいですね。

I can only do so much "analysis".

例えば、ビットコインとか、

Bitcoin, for example.

あるいは、NFTとか、

or "NFT"

です。

-----

世の中で「当たるかどうか」と、技術の内容は、あまり関係ありません。

Whether or not t"popular" in the world has little to do with the content of technology.

もちろん、「当った」または「外れた」『後』で、それに理屈をつけること(分析)は、とっても簡単です。

Of course, it is very easy to reason (analyze) after a "hit" or "miss".

ただ、それでは勝てませんし、儲かりませんが。

However, I never win and make money.

-----

メタバースであれ、量子コンピュータであれ、現時点では、断定的なことは言わない方が得策です。

Even if it is metaverse or quantum computer, it is the best not to say something difinitively at the present.

それらの評価が落ち着いたところで、

After the trend have settled in, you will say,

『実は、俺、最初から分かっていたんだよね』

'Actually, I knew that all along'

と語るのが、一番、クールに見せることができます。

This way will make you cool.

もっとも、私は『仕事柄』そういうことを言う奴を、たくさん見てきました。

I've seen a lot of people who say like that in the course of my work.

私も、そういう奴の一人です。

I am also one of them.

ですから、同類を見抜くことなど、とても簡単です。

Therefore, it is easy for me to find such a person.

2022/09,江端さんの技術メモ

C言語によるファジィ(Fuzzy)推論コード

をGo言語に移植してみました。

で、できるだけ、

「GO言語でクラスっぽいことをする」をレビューする

に近づけてみました。

コメントのPrint分は、C++のを残しています(面倒だったので)。あとenum型がないので、テキトーに文字列を使うことにしました。

以前のコードには、豪快なバグがありましたので、ソックリ差し替えします。

設定条件は以下の通りです(テーブルの内容は、負荷量です)

package main

import (
	"fmt"
	"os"
)

func max_2(a, b float64) float64 {
	if a > b {
		return a
	} else {
		return b
	}
}

func min_2(a, b float64) float64 {
	if a > b {
		return b
	} else {
		return a
	}
}

type condition_MF3 struct { // condition_MF3の基底クラス
	center  float64
	width   float64
	express string
}

func new_condition_MF3(_center, _width float64, _express string) *condition_MF3 {
	c3 := new(condition_MF3)
	c3.center = _center
	c3.width = _width
	c3.express = _express
	return c3
}

// 前件部メンバーシップ関数(山3つ)クラス
func (c3 *condition_MF3) func_X(_x float64) float64 {
	// x,yは、メンバーシップ関数上の座標を示す
	x := _x
	y := 0.0 // yの値は、必ず0以上1以下になる

	if c3.express == "LESS" {
		if x <= c3.center-c3.width {
			y = 1.0
		} else if x <= c3.center {
			y = -1.0 / c3.width * (x - c3.center)
		} else {
			y = 0.0
		}
	} else if c3.express == "COMMON" {
		if x <= c3.center-c3.width {
			y = 0.0
		} else if x <= c3.center {
			y = 1.0/c3.width*(x-c3.center) + 1.0
		} else if x <= c3.center+c3.width {
			y = -1.0/c3.width*(x-c3.center) + 1.0
		} else {
			y = 0.0
		}
	} else if c3.express == "MORE" {
		if x <= c3.center {
			y = 0.0
		} else if x <= c3.center+c3.width {
			y = 1.0 / c3.width * (x - c3.center)
		} else {
			y = 1.0
		}
	} else {
		fmt.Println("MF3: wrong expression")
		os.Exit(1)
	}
	return y
}

type condition_MF5 struct { // condition_MF5の基底クラス
	center  float64
	width   float64
	express string
}

func new_condition_MF5(_center, _width float64, _express string) *condition_MF5 {
	c5 := new(condition_MF5)
	c5.center = _center
	c5.width = _width
	c5.express = _express
	return c5
}

func (c5 *condition_MF5) func_X(_x float64) float64 {
	// 前件部メンバーシップ関数(山5つ)クラス
	// x,yは、メンバーシップ関数上の座標を示す
	x := _x
	y := 0.0 // yの値は、必ず0以上1以下になる

	if c5.express == "LESSLESS" {
		if x <= c5.center-2.0*c5.width {
			y = 1.0
		} else if x <= c5.center-c5.width {
			y = -1.0/c5.width*(x-(c5.center-2.0*c5.width)) + 1.0
		} else {
			y = 0.0
		}
	} else if c5.express == "LESS" {
		if x <= c5.center-2.0*c5.width {
			y = 0.0
		} else if x <= c5.center-c5.width {
			y = 1.0/c5.width*(x-(c5.center-c5.width)) + 1.0
		} else if x <= c5.center {
			y = -1.0/c5.width*(x-(c5.center-c5.width)) + 1.0
		} else {
			y = 0.0
		}
	} else if c5.express == "COMMON" {
		if x <= c5.center-c5.width {
			y = 0.0
		} else if x <= c5.center {
			y = 1.0/c5.width*(x-c5.center) + 1.0
		} else if x <= c5.center+c5.width {
			y = -1.0/c5.width*(x-c5.center) + 1.0
		} else {
			y = 0.0
		}
	} else if c5.express == "MORE" {
		if x <= c5.center {
			y = 0.0
		} else if x <= c5.center+c5.width {
			y = 1.0/c5.width*(x-(c5.center+c5.width)) + 1.0
		} else if x <= c5.center+2.0*c5.width {
			y = -1.0/c5.width*(x-(c5.center+c5.width)) + 1.0
		} else {
			y = 0.0
		}
	} else if c5.express == "MOREMORE" {
		if x <= c5.center+c5.width {
			y = 0.0
		} else if x <= c5.center+2.0*c5.width {
			y = 1.0/c5.width*(x-(c5.center+2.0*c5.width)) + 1.0
		} else {
			y = 1.0
		}
	} else {
		fmt.Println("MF5 func_X(): wrong expression")
		os.Exit(1)
	}

	return y
}

/////////////////////////////

type action_MF5 struct { // condition_MF5の基底クラス
	center  float64
	width   float64
	express string
	x       float64
	y       float64
}

func new_action_MF5(_center, _width float64, _express string) *action_MF5 {
	a5 := new(action_MF5)
	a5.center = _center
	a5.width = _width
	a5.express = _express

	if a5.express == "LESSLESS" {
		a5.x = a5.center - 2.0*a5.width
	} else if a5.express == "LESS" {
		a5.x = a5.center - a5.width
	} else if a5.express == "COMMON" {
		a5.x = a5.center
	} else if a5.express == "MORE" {
		a5.x = a5.center + a5.width
	} else if a5.express == "MOREMORE" {
		a5.x = a5.center + 2.0*a5.width
	} else {
		fmt.Println("new_action_MF5: wrong scale expression")
		os.Exit(-1)
	}

	a5.y = 0.0

	return a5
}

// 後件部メンバーシップ関数(山3つ)クラス
func (a5 *action_MF5) func_Y() float64 {
	// x,yは、メンバーシップ関数上の座標を示す

	return a5.y
}

func (a5 *action_MF5) func_Max(b float64) {
	a5.y = max_2(b, a5.y)
}

func (a5 *action_MF5) func_X() float64 {
	return a5.x
}

func complain_reasoning(dis, age float64) float64 {

	// Walking(前件部)
	Walk_Less := new_condition_MF3(1.4, 0.4, "LESS")
	Walk_Common := new_condition_MF3(1.4, 0.4, "COMMON")
	Walk_More := new_condition_MF3(1.4, 0.4, "MORE")

	// Age(前件部)
	Age_LessLess := new_condition_MF5(42, 15, "LESSLESS")
	Age_Less := new_condition_MF5(42, 15, "LESS")
	Age_Common := new_condition_MF5(42, 15, "COMMON") // 中央が 42歳
	Age_More := new_condition_MF5(42, 15, "MORE")
	Age_MoreMore := new_condition_MF5(42, 15, "MOREMORE")

	// Complain(後件部)
	Complain_LessLess := new_action_MF5(0.5, 0.25, "LESSLESS") // 不満の中央値が0.5   0.0/0.25/0.50/0.75/1.00 の5段階

	Complain_Less := new_action_MF5(0.5, 0.25, "LESS")
	Complain_Common := new_action_MF5(0.5, 0.25, "COMMON")
	Complain_More := new_action_MF5(0.5, 0.25, "MORE")
	Complain_MoreMore := new_action_MF5(0.5, 0.25, "MOREMORE")

	// [ルール00] 歩行距離:少0 年令:子供0
	Rule00 := min_2(Walk_Less.func_X(dis), Age_LessLess.func_X(age))
	Complain_LessLess.func_Max(Rule00) // 後件部は上書きされていく
	//fmt.Println("Rule00", Rule00)

	// [ルール01] 歩行距離:少0 年令:若年1
	Rule01 := min_2(Walk_Less.func_X(dis), Age_Less.func_X(age))
	Complain_LessLess.func_Max(Rule01) // 後件部は上書きされていく
	//fmt.Println("Rule01", Rule01)

	// [ルール02] 歩行距離:少0 年令:壮年2
	Rule02 := min_2(Walk_Less.func_X(dis), Age_Common.func_X(age))
	Complain_Common.func_Max(Rule02) // 後件部は上書きされていく
	//fmt.Println("Rule02", Rule02)

	// [ルール03] 歩行距離:少0 年令:高齢3
	Rule03 := min_2(Walk_Less.func_X(dis), Age_More.func_X(age))
	Complain_Common.func_Max(Rule03) // 後件部は上書きされていく
	//fmt.Println("Rule03", Rule03)

	// [ルール04] 歩行距離:少0 年令:老齢4
	Rule04 := min_2(Walk_Less.func_X(dis), Age_MoreMore.func_X(age))
	Complain_More.func_Max(Rule04) // 後件部は上書きされていく
	//fmt.Println("Rule04", Rule04)

	// [ルール10] 歩行距離:普通1 年令:子供0
	Rule10 := min_2(Walk_Common.func_X(dis), Age_LessLess.func_X(age))
	Complain_LessLess.func_Max(Rule10) // 後件部は上書きされていく
	//fmt.Println("Rule10", Rule10)

	// [ルール11] 歩行距離:普通1 年令:若年1
	Rule11 := min_2(Walk_Common.func_X(dis), Age_Less.func_X(age))
	Complain_Less.func_Max(Rule11) // 後件部は上書きされていく
	//fmt.Println("Rule11", Rule11)

	// [ルール12] 歩行距離:普通1 年令:壮年2
	Rule12 := min_2(Walk_Common.func_X(dis), Age_Common.func_X(age))
	Complain_Common.func_Max(Rule12) // 後件部は上書きされていく
	//fmt.Println("Rule12", Rule12)

	// [ルール13] 歩行距離:普通1 年令:高齢3
	Rule13 := min_2(Walk_Common.func_X(dis), Age_More.func_X(age))
	Complain_More.func_Max(Rule13) // 後件部は上書きされていく
	//fmt.Println("Rule13", Rule13)

	// [ルール14] 歩行距離:普通1 年令:老齢4
	Rule14 := min_2(Walk_Common.func_X(dis), Age_MoreMore.func_X(age))
	Complain_MoreMore.func_Max(Rule14) // 後件部は上書きされていく
	//fmt.Println("Rule14", Rule14)

	// [ルール20] 歩行距離:多2 年令:こども0
	Rule20 := min_2(Walk_More.func_X(dis), Age_LessLess.func_X(age))
	Complain_Less.func_Max(Rule20) // 後件部は上書きされていく
	//fmt.Println("Rule20", Rule20)

	// [ルール21] 歩行距離:多2 年令:若年1
	Rule21 := min_2(Walk_More.func_X(dis), Age_Less.func_X(age))
	Complain_Common.func_Max(Rule21) // 後件部は上書きされていく
	//fmt.Println("Rule21", Rule21)

	// [ルール22] 歩行距離:多2 年令:壮年2
	Rule22 := min_2(Walk_More.func_X(dis), Age_Common.func_X(age))
	Complain_More.func_Max(Rule22) // 後件部は上書きされていく
	//fmt.Println("Rule22", Rule22)

	// [ルール23] 歩行距離:多2 年令:高齢3
	Rule23 := min_2(Walk_More.func_X(dis), Age_More.func_X(age))
	Complain_MoreMore.func_Max(Rule23) // 後件部は上書きされていく
	//fmt.Println("Rule23", Rule23)

	// [ルール24] 歩行距離:多2 年令:老齢4
	Rule24 := min_2(Walk_More.func_X(dis), Age_MoreMore.func_X(age))
	Complain_MoreMore.func_Max(Rule24) // 後件部は上書きされていく
	//fmt.Println("Rule24", Rule24)

	// 推論計算
	numerator :=
		Complain_LessLess.func_X()*Complain_LessLess.func_Y() +
			Complain_Less.func_X()*Complain_Less.func_Y() +
			Complain_Common.func_X()*Complain_Common.func_Y() +
			Complain_More.func_X()*Complain_More.func_Y() +
			Complain_MoreMore.func_X()*Complain_MoreMore.func_Y()

	denominator :=
		Complain_LessLess.func_Y() +
			Complain_Less.func_Y() +
			Complain_Common.func_Y() +
			Complain_More.func_Y() +
			Complain_MoreMore.func_Y()

	complain := numerator / denominator

	return complain

}

2020,江端さんの技術メモ

江端の環境でのローカルな場所 C:\Users\ebata\Desktop\bouncy2

C:\Users\ebata>cd C:\Users\ebata\Desktop\bouncy2
cd /c/Users/ebata/Desktop/bouncy2
python3 httpsrv.py
http://localhost:8080 で起動

 

  • 2020/03/14

     

  • 背景
    • ■これからは「C/C++ → Go」と「JavaScript → Webassembly」とシフトしながら色々やっていこうと思っている■正直、どっちも難しい。そういう時は、サンプルをパクって勉強するのが正解 ―― と思っている
  • 目的
  • 環境
    • ■Windows7 or 10
    • ■Go と python3 をインストールしてある
  • やってみたこと
    • ■サンプルプログラムをダウンロードして、コンパイルしてみた
      • ~ https://github.com/stdiopt/gowasm-experiments/tree/master/bouncy
      • コマンドプロンプトからこんな感じでコンパイルできた
        • $ set GOOS=js
        • $ set  GOARCH=wasm
        • $ go build -o main.wasm main.go
        • ちなみに、bashの環境があれば、build.shでコンパイルできる
      • 実際には、コンパイルしなくても、バイナリコード(main.wasm)も、ダウンロードの中に入っているので、コンパイルは不要だったが
    • ■ローカルサーバとしては、色々試してみた
      • python3 -m http.server 8080
      • Goでサーバも作ってみた
      • Perlでもやってみた
    • ■動画がどうしても出てこない
    • ■ここから丸2日間の格闘のスタート
      • もう、色々探しまくった
  • 確認していた問題点
    • ■Google Chromoから、→ 「その他のツール」 → 「ディベロッパーツール」 → 
      • Uncaught (in promise) TypeError: Failed to execute 'compile' on 'WebAssembly': Incorrect response MIME type. Expected 'application/wasm'.この「Expected 'application/wasm'.」がどうにも、不味いらしい ―― が、解決方法が、どうにも見つからない
    • ■kobore.netのサーバに上げても、改善が見られず
  • ローカルサーバを作ってみた
    • ■httpsrv.pyを作った
      #!/usr/bin/env python3
      import http.server
        import socketserver
        
        PORT =
      8080
        
        Handler = http.server.SimpleHTTPRequestHandler
        Handler.extensions_map.update({
        '.wasm': 'application/wasm',
        })
        
        socketserver.TCPServer.allow_reuse_address =
      True
      with socketserver.TCPServer(("", PORT), Handler) as httpd:
        httpd.allow_reuse_address =
      True
      print("serving at port", PORT)
        httpd.serve_forever()
      
  • ■httpsrv.pyを起動した
    • $ python3 httpsrv.py
  • http://localhost:8080 で起動
    • ■動いた
      ■動かなかったら、chromo → 「設定」 → 「閲覧履歴データの削除」でキャッシュをクリアてみること

以上

httpsrv.py

#!/usr/bin/env python3

import http.server
import socketserver

PORT = 8080

Handler = http.server.SimpleHTTPRequestHandler
Handler.extensions_map.update({
    '.wasm': 'application/wasm',
})

socketserver.TCPServer.allow_reuse_address = True
with socketserver.TCPServer(("", PORT), Handler) as httpd:
    httpd.allow_reuse_address = True
    print("serving at port", PORT)
    httpd.serve_forever()

main.go

//Wasming
// compile: GOOS=js GOARCH=wasm go build -o main.wasm ./main.go
package main

import (
	"fmt"
	"math"
	"math/rand"
	"strconv"
	"syscall/js"
)

var (
	width      float64
	height     float64
	mousePos   [2]float64
	ctx        js.Value    // "syscall/js"から引っ張られる
	lineDistSq float64 = 100 * 100
)

func main() {

	// Init Canvas stuff
	doc := js.Global().Get("document")
	canvasEl := doc.Call("getElementById", "mycanvas")
	width = doc.Get("body").Get("clientWidth").Float()
	height = doc.Get("body").Get("clientHeight").Float()
	canvasEl.Set("width", width)
	canvasEl.Set("height", height)
	ctx = canvasEl.Call("getContext", "2d")

	done := make(chan struct{}, 0)

	dt := DotThing{speed: 160, size: 6}

	mouseMoveEvt := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		e := args[0]
		mousePos[0] = e.Get("clientX").Float()
		mousePos[1] = e.Get("clientY").Float()
		return nil
	})
	defer mouseMoveEvt.Release()

	// Event handler for count range
	countChangeEvt := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		evt := args[0]
		intVal, err := strconv.Atoi(evt.Get("target").Get("value").String())
		if err != nil {
			println("Invalid value", err)
			return nil
		}
		dt.SetNDots(intVal)
		return nil
	})
	defer countChangeEvt.Release()

	// Event handler for speed range
	speedInputEvt := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		evt := args[0]
		fval, err := strconv.ParseFloat(evt.Get("target").Get("value").String(), 64)
		if err != nil {
			println("invalid value", err)
			return nil
		}
		dt.speed = fval
		return nil
	})
	defer speedInputEvt.Release()

	// Event handler for size
	sizeChangeEvt := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		evt := args[0]
		intVal, err := strconv.Atoi(evt.Get("target").Get("value").String())
		if err != nil {
			println("invalid value", err)
			return nil
		}
		dt.size = intVal
		return nil
	})
	defer sizeChangeEvt.Release()

	// Event handler for lines toggle
	lineChangeEvt := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		evt := args[0]
		dt.lines = evt.Get("target").Get("checked").Bool()
		return nil
	})
	defer lineChangeEvt.Release()

	// Event handler for dashed toggle
	dashedChangeEvt := js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		evt := args[0]
		dt.dashed = evt.Get("target").Get("checked").Bool()
		return nil
	})
	defer dashedChangeEvt.Release()

	doc.Call("addEventListener", "mousemove", mouseMoveEvt)
	doc.Call("getElementById", "count").Call("addEventListener", "change", countChangeEvt)
	doc.Call("getElementById", "speed").Call("addEventListener", "input", speedInputEvt)
	doc.Call("getElementById", "size").Call("addEventListener", "input", sizeChangeEvt)
	doc.Call("getElementById", "dashed").Call("addEventListener", "change", dashedChangeEvt)
	doc.Call("getElementById", "lines").Call("addEventListener", "change", lineChangeEvt)

	dt.SetNDots(100)
	dt.lines = false
	var renderFrame js.Func
	var tmark float64
	var markCount = 0
	var tdiffSum float64

	renderFrame = js.FuncOf(func(this js.Value, args []js.Value) interface{} {
		now := args[0].Float()
		tdiff := now - tmark
		tdiffSum += now - tmark
		markCount++
		if markCount > 10 {
			doc.Call("getElementById", "fps").Set("innerHTML", fmt.Sprintf("FPS: %.01f", 1000/(tdiffSum/float64(markCount))))
			tdiffSum, markCount = 0, 0
		}
		tmark = now

		// Pull window size to handle resize
		curBodyW := doc.Get("body").Get("clientWidth").Float()
		curBodyH := doc.Get("body").Get("clientHeight").Float()
		if curBodyW != width || curBodyH != height {
			width, height = curBodyW, curBodyH
			canvasEl.Set("width", width)
			canvasEl.Set("height", height)
		}
		dt.Update(tdiff / 1000)

		js.Global().Call("requestAnimationFrame", renderFrame)
		return nil
	})
	defer renderFrame.Release()

	// Start running
	js.Global().Call("requestAnimationFrame", renderFrame)

	<-done

}

// DotThing manager
type DotThing struct {
	dots   []*Dot
	dashed bool
	lines  bool
	speed  float64
	size   int
}

// Update updates the dot positions and draws
func (dt *DotThing) Update(dtTime float64) {
	if dt.dots == nil {
		return
	}
	ctx.Call("clearRect", 0, 0, width, height)

	// Update
	for i, dot := range dt.dots {
		dir := [2]float64{}
		// Bounce
		if dot.pos[0] < 0 {
			dot.pos[0] = 0
			dot.dir[0] *= -1
		}
		if dot.pos[0] > width {
			dot.pos[0] = width
			dot.dir[0] *= -1
		}

		if dot.pos[1] < 0 {
			dot.pos[1] = 0
			dot.dir[1] *= -1
		}

		if dot.pos[1] > height {
			dot.pos[1] = height
			dot.dir[1] *= -1
		}
		dir = dot.dir

		ctx.Set("globalAlpha", 0.5)
		ctx.Call("beginPath")
		ctx.Set("fillStyle", fmt.Sprintf("#%06x", dot.color))
		ctx.Set("strokeStyle", fmt.Sprintf("#%06x", dot.color))
		// Dashed array ref: https://github.com/golang/go/blob/release-branch.go1.11/src/syscall/js/js.go#L98
		ctx.Call("setLineDash", []interface{}{})
		if dt.dashed {
			ctx.Call("setLineDash", []interface{}{5, 10})
		}
		ctx.Set("lineWidth", dt.size)
		ctx.Call("arc", dot.pos[0], dot.pos[1], dt.size, 0, 2*math.Pi)
		ctx.Call("fill")

		mdx := mousePos[0] - dot.pos[0]
		mdy := mousePos[1] - dot.pos[1]
		d := math.Sqrt(mdx*mdx + mdy*mdy)
		if d < 200 {
			ctx.Set("globalAlpha", 1-d/200)
			ctx.Call("beginPath")
			ctx.Call("moveTo", dot.pos[0], dot.pos[1])
			ctx.Call("lineTo", mousePos[0], mousePos[1])
			ctx.Call("stroke")
			if d > 100 { // move towards mouse
				dir[0] = (mdx / d) * 2
				dir[1] = (mdy / d) * 2
			} else { // do not move
				dir[0] = 0
				dir[1] = 0
			}
		}

		if dt.lines {
			for _, dot2 := range dt.dots[i+1:] {
				mx := dot2.pos[0] - dot.pos[0]
				my := dot2.pos[1] - dot.pos[1]
				d := mx*mx + my*my
				if d < lineDistSq {
					ctx.Set("globalAlpha", 1-d/lineDistSq)
					ctx.Call("beginPath")
					ctx.Call("moveTo", dot.pos[0], dot.pos[1])
					ctx.Call("lineTo", dot2.pos[0], dot2.pos[1])
					ctx.Call("stroke")
				}
			}
		}

		dot.pos[0] += dir[0] * dt.speed * dtTime
		dot.pos[1] += dir[1] * dt.speed * dtTime
	}
}

// SetNDots reinitializes dots with n size
func (dt *DotThing) SetNDots(n int) {
	dt.dots = make([]*Dot, n)
	for i := 0; i < n; i++ {
		dt.dots[i] = &Dot{
			pos: [2]float64{
				rand.Float64() * width,
				rand.Float64() * height,
			},
			dir: [2]float64{
				rand.NormFloat64(),
				rand.NormFloat64(),
			},
			color: uint32(rand.Intn(0xFFFFFF)),
			size:  10,
		}
	}
}

// Dot represents a dot ...
type Dot struct {
	pos   [2]float64
	dir   [2]float64
	color uint32
	size  float64
}

build.sh

#!/bin/sh

GOOS=js GOARCH=wasm go build -o main.wasm ./main.go

wasm_exec.js

// Copyright 2018 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.

(() => {
	// Map multiple JavaScript environments to a single common API,
	// preferring web standards over Node.js API.
	//
	// Environments considered:
	// - Browsers
	// - Node.js
	// - Electron
	// - Parcel

	if (typeof global !== "undefined") {
		// global already exists
	} else if (typeof window !== "undefined") {
		window.global = window;
	} else if (typeof self !== "undefined") {
		self.global = self;
	} else {
		throw new Error("cannot export Go (neither global, window nor self is defined)");
	}

	if (!global.require && typeof require !== "undefined") {
		global.require = require;
	}

	if (!global.fs && global.require) {
		global.fs = require("fs");
	}

	const enosys = () => {
		const err = new Error("not implemented");
		err.code = "ENOSYS";
		return err;
	};

	if (!global.fs) {
		let outputBuf = "";
		global.fs = {
			constants: { O_WRONLY: -1, O_RDWR: -1, O_CREAT: -1, O_TRUNC: -1, O_APPEND: -1, O_EXCL: -1 }, // unused
			writeSync(fd, buf) {
				outputBuf += decoder.decode(buf);
				const nl = outputBuf.lastIndexOf("\n");
				if (nl != -1) {
					console.log(outputBuf.substr(0, nl));
					outputBuf = outputBuf.substr(nl + 1);
				}
				return buf.length;
			},
			write(fd, buf, offset, length, position, callback) {
				if (offset !== 0 || length !== buf.length || position !== null) {
					callback(enosys());
					return;
				}
				const n = this.writeSync(fd, buf);
				callback(null, n);
			},
			chmod(path, mode, callback) { callback(enosys()); },
			chown(path, uid, gid, callback) { callback(enosys()); },
			close(fd, callback) { callback(enosys()); },
			fchmod(fd, mode, callback) { callback(enosys()); },
			fchown(fd, uid, gid, callback) { callback(enosys()); },
			fstat(fd, callback) { callback(enosys()); },
			fsync(fd, callback) { callback(null); },
			ftruncate(fd, length, callback) { callback(enosys()); },
			lchown(path, uid, gid, callback) { callback(enosys()); },
			link(path, link, callback) { callback(enosys()); },
			lstat(path, callback) { callback(enosys()); },
			mkdir(path, perm, callback) { callback(enosys()); },
			open(path, flags, mode, callback) { callback(enosys()); },
			read(fd, buffer, offset, length, position, callback) { callback(enosys()); },
			readdir(path, callback) { callback(enosys()); },
			readlink(path, callback) { callback(enosys()); },
			rename(from, to, callback) { callback(enosys()); },
			rmdir(path, callback) { callback(enosys()); },
			stat(path, callback) { callback(enosys()); },
			symlink(path, link, callback) { callback(enosys()); },
			truncate(path, length, callback) { callback(enosys()); },
			unlink(path, callback) { callback(enosys()); },
			utimes(path, atime, mtime, callback) { callback(enosys()); },
		};
	}

	if (!global.process) {
		global.process = {
			getuid() { return -1; },
			getgid() { return -1; },
			geteuid() { return -1; },
			getegid() { return -1; },
			getgroups() { throw enosys(); },
			pid: -1,
			ppid: -1,
			umask() { throw enosys(); },
			cwd() { throw enosys(); },
			chdir() { throw enosys(); },
		}
	}

	if (!global.crypto) {
		const nodeCrypto = require("crypto");
		global.crypto = {
			getRandomValues(b) {
				nodeCrypto.randomFillSync(b);
			},
		};
	}

	if (!global.performance) {
		global.performance = {
			now() {
				const [sec, nsec] = process.hrtime();
				return sec * 1000 + nsec / 1000000;
			},
		};
	}

	if (!global.TextEncoder) {
		global.TextEncoder = require("util").TextEncoder;
	}

	if (!global.TextDecoder) {
		global.TextDecoder = require("util").TextDecoder;
	}

	// End of polyfills for common API.

	const encoder = new TextEncoder("utf-8");
	const decoder = new TextDecoder("utf-8");

	global.Go = class {
		constructor() {
			this.argv = ["js"];
			this.env = {};
			this.exit = (code) => {
				if (code !== 0) {
					console.warn("exit code:", code);
				}
			};
			this._exitPromise = new Promise((resolve) => {
				this._resolveExitPromise = resolve;
			});
			this._pendingEvent = null;
			this._scheduledTimeouts = new Map();
			this._nextCallbackTimeoutID = 1;

			const setInt64 = (addr, v) => {
				this.mem.setUint32(addr + 0, v, true);
				this.mem.setUint32(addr + 4, Math.floor(v / 4294967296), true);
			}

			const getInt64 = (addr) => {
				const low = this.mem.getUint32(addr + 0, true);
				const high = this.mem.getInt32(addr + 4, true);
				return low + high * 4294967296;
			}

			const loadValue = (addr) => {
				const f = this.mem.getFloat64(addr, true);
				if (f === 0) {
					return undefined;
				}
				if (!isNaN(f)) {
					return f;
				}

				const id = this.mem.getUint32(addr, true);
				return this._values[id];
			}

			const storeValue = (addr, v) => {
				const nanHead = 0x7FF80000;

				if (typeof v === "number") {
					if (isNaN(v)) {
						this.mem.setUint32(addr + 4, nanHead, true);
						this.mem.setUint32(addr, 0, true);
						return;
					}
					if (v === 0) {
						this.mem.setUint32(addr + 4, nanHead, true);
						this.mem.setUint32(addr, 1, true);
						return;
					}
					this.mem.setFloat64(addr, v, true);
					return;
				}

				switch (v) {
					case undefined:
						this.mem.setFloat64(addr, 0, true);
						return;
					case null:
						this.mem.setUint32(addr + 4, nanHead, true);
						this.mem.setUint32(addr, 2, true);
						return;
					case true:
						this.mem.setUint32(addr + 4, nanHead, true);
						this.mem.setUint32(addr, 3, true);
						return;
					case false:
						this.mem.setUint32(addr + 4, nanHead, true);
						this.mem.setUint32(addr, 4, true);
						return;
				}

				let id = this._ids.get(v);
				if (id === undefined) {
					id = this._idPool.pop();
					if (id === undefined) {
						id = this._values.length;
					}
					this._values[id] = v;
					this._goRefCounts[id] = 0;
					this._ids.set(v, id);
				}
				this._goRefCounts[id]++;
				let typeFlag = 1;
				switch (typeof v) {
					case "string":
						typeFlag = 2;
						break;
					case "symbol":
						typeFlag = 3;
						break;
					case "function":
						typeFlag = 4;
						break;
				}
				this.mem.setUint32(addr + 4, nanHead | typeFlag, true);
				this.mem.setUint32(addr, id, true);
			}

			const loadSlice = (addr) => {
				const array = getInt64(addr + 0);
				const len = getInt64(addr + 8);
				return new Uint8Array(this._inst.exports.mem.buffer, array, len);
			}

			const loadSliceOfValues = (addr) => {
				const array = getInt64(addr + 0);
				const len = getInt64(addr + 8);
				const a = new Array(len);
				for (let i = 0; i < len; i++) {
					a[i] = loadValue(array + i * 8);
				}
				return a;
			}

			const loadString = (addr) => {
				const saddr = getInt64(addr + 0);
				const len = getInt64(addr + 8);
				return decoder.decode(new DataView(this._inst.exports.mem.buffer, saddr, len));
			}

			const timeOrigin = Date.now() - performance.now();
			this.importObject = {
				go: {
					// Go's SP does not change as long as no Go code is running. Some operations (e.g. calls, getters and setters)
					// may synchronously trigger a Go event handler. This makes Go code get executed in the middle of the imported
					// function. A goroutine can switch to a new stack if the current stack is too small (see morestack function).
					// This changes the SP, thus we have to update the SP used by the imported function.

					// func wasmExit(code int32)
					"runtime.wasmExit": (sp) => {
						const code = this.mem.getInt32(sp + 8, true);
						this.exited = true;
						delete this._inst;
						delete this._values;
						delete this._goRefCounts;
						delete this._ids;
						delete this._idPool;
						this.exit(code);
					},

					// func wasmWrite(fd uintptr, p unsafe.Pointer, n int32)
					"runtime.wasmWrite": (sp) => {
						const fd = getInt64(sp + 8);
						const p = getInt64(sp + 16);
						const n = this.mem.getInt32(sp + 24, true);
						fs.writeSync(fd, new Uint8Array(this._inst.exports.mem.buffer, p, n));
					},

					// func resetMemoryDataView()
					"runtime.resetMemoryDataView": (sp) => {
						this.mem = new DataView(this._inst.exports.mem.buffer);
					},

					// func nanotime1() int64
					"runtime.nanotime1": (sp) => {
						setInt64(sp + 8, (timeOrigin + performance.now()) * 1000000);
					},

					// func walltime1() (sec int64, nsec int32)
					"runtime.walltime1": (sp) => {
						const msec = (new Date).getTime();
						setInt64(sp + 8, msec / 1000);
						this.mem.setInt32(sp + 16, (msec % 1000) * 1000000, true);
					},

					// func scheduleTimeoutEvent(delay int64) int32
					"runtime.scheduleTimeoutEvent": (sp) => {
						const id = this._nextCallbackTimeoutID;
						this._nextCallbackTimeoutID++;
						this._scheduledTimeouts.set(id, setTimeout(
							() => {
								this._resume();
								while (this._scheduledTimeouts.has(id)) {
									// for some reason Go failed to register the timeout event, log and try again
									// (temporary workaround for https://github.com/golang/go/issues/28975)
									console.warn("scheduleTimeoutEvent: missed timeout event");
									this._resume();
								}
							},
							getInt64(sp + 8) + 1, // setTimeout has been seen to fire up to 1 millisecond early
						));
						this.mem.setInt32(sp + 16, id, true);
					},

					// func clearTimeoutEvent(id int32)
					"runtime.clearTimeoutEvent": (sp) => {
						const id = this.mem.getInt32(sp + 8, true);
						clearTimeout(this._scheduledTimeouts.get(id));
						this._scheduledTimeouts.delete(id);
					},

					// func getRandomData(r []byte)
					"runtime.getRandomData": (sp) => {
						crypto.getRandomValues(loadSlice(sp + 8));
					},

					// func finalizeRef(v ref)
					"syscall/js.finalizeRef": (sp) => {
						const id = this.mem.getUint32(sp + 8, true);
						this._goRefCounts[id]--;
						if (this._goRefCounts[id] === 0) {
							const v = this._values[id];
							this._values[id] = null;
							this._ids.delete(v);
							this._idPool.push(id);
						}
					},

					// func stringVal(value string) ref
					"syscall/js.stringVal": (sp) => {
						storeValue(sp + 24, loadString(sp + 8));
					},

					// func valueGet(v ref, p string) ref
					"syscall/js.valueGet": (sp) => {
						const result = Reflect.get(loadValue(sp + 8), loadString(sp + 16));
						sp = this._inst.exports.getsp(); // see comment above
						storeValue(sp + 32, result);
					},

					// func valueSet(v ref, p string, x ref)
					"syscall/js.valueSet": (sp) => {
						Reflect.set(loadValue(sp + 8), loadString(sp + 16), loadValue(sp + 32));
					},

					// func valueDelete(v ref, p string)
					"syscall/js.valueDelete": (sp) => {
						Reflect.deleteProperty(loadValue(sp + 8), loadString(sp + 16));
					},

					// func valueIndex(v ref, i int) ref
					"syscall/js.valueIndex": (sp) => {
						storeValue(sp + 24, Reflect.get(loadValue(sp + 8), getInt64(sp + 16)));
					},

					// valueSetIndex(v ref, i int, x ref)
					"syscall/js.valueSetIndex": (sp) => {
						Reflect.set(loadValue(sp + 8), getInt64(sp + 16), loadValue(sp + 24));
					},

					// func valueCall(v ref, m string, args []ref) (ref, bool)
					"syscall/js.valueCall": (sp) => {
						try {
							const v = loadValue(sp + 8);
							const m = Reflect.get(v, loadString(sp + 16));
							const args = loadSliceOfValues(sp + 32);
							const result = Reflect.apply(m, v, args);
							sp = this._inst.exports.getsp(); // see comment above
							storeValue(sp + 56, result);
							this.mem.setUint8(sp + 64, 1);
						} catch (err) {
							storeValue(sp + 56, err);
							this.mem.setUint8(sp + 64, 0);
						}
					},

					// func valueInvoke(v ref, args []ref) (ref, bool)
					"syscall/js.valueInvoke": (sp) => {
						try {
							const v = loadValue(sp + 8);
							const args = loadSliceOfValues(sp + 16);
							const result = Reflect.apply(v, undefined, args);
							sp = this._inst.exports.getsp(); // see comment above
							storeValue(sp + 40, result);
							this.mem.setUint8(sp + 48, 1);
						} catch (err) {
							storeValue(sp + 40, err);
							this.mem.setUint8(sp + 48, 0);
						}
					},

					// func valueNew(v ref, args []ref) (ref, bool)
					"syscall/js.valueNew": (sp) => {
						try {
							const v = loadValue(sp + 8);
							const args = loadSliceOfValues(sp + 16);
							const result = Reflect.construct(v, args);
							sp = this._inst.exports.getsp(); // see comment above
							storeValue(sp + 40, result);
							this.mem.setUint8(sp + 48, 1);
						} catch (err) {
							storeValue(sp + 40, err);
							this.mem.setUint8(sp + 48, 0);
						}
					},

					// func valueLength(v ref) int
					"syscall/js.valueLength": (sp) => {
						setInt64(sp + 16, parseInt(loadValue(sp + 8).length));
					},

					// valuePrepareString(v ref) (ref, int)
					"syscall/js.valuePrepareString": (sp) => {
						const str = encoder.encode(String(loadValue(sp + 8)));
						storeValue(sp + 16, str);
						setInt64(sp + 24, str.length);
					},

					// valueLoadString(v ref, b []byte)
					"syscall/js.valueLoadString": (sp) => {
						const str = loadValue(sp + 8);
						loadSlice(sp + 16).set(str);
					},

					// func valueInstanceOf(v ref, t ref) bool
					"syscall/js.valueInstanceOf": (sp) => {
						this.mem.setUint8(sp + 24, loadValue(sp + 8) instanceof loadValue(sp + 16));
					},

					// func copyBytesToGo(dst []byte, src ref) (int, bool)
					"syscall/js.copyBytesToGo": (sp) => {
						const dst = loadSlice(sp + 8);
						const src = loadValue(sp + 32);
						if (!(src instanceof Uint8Array)) {
							this.mem.setUint8(sp + 48, 0);
							return;
						}
						const toCopy = src.subarray(0, dst.length);
						dst.set(toCopy);
						setInt64(sp + 40, toCopy.length);
						this.mem.setUint8(sp + 48, 1);
					},

					// func copyBytesToJS(dst ref, src []byte) (int, bool)
					"syscall/js.copyBytesToJS": (sp) => {
						const dst = loadValue(sp + 8);
						const src = loadSlice(sp + 16);
						if (!(dst instanceof Uint8Array)) {
							this.mem.setUint8(sp + 48, 0);
							return;
						}
						const toCopy = src.subarray(0, dst.length);
						dst.set(toCopy);
						setInt64(sp + 40, toCopy.length);
						this.mem.setUint8(sp + 48, 1);
					},

					"debug": (value) => {
						console.log(value);
					},
				}
			};
		}

		async run(instance) {
			this._inst = instance;
			this.mem = new DataView(this._inst.exports.mem.buffer);
			this._values = [ // JS values that Go currently has references to, indexed by reference id
				NaN,
				0,
				null,
				true,
				false,
				global,
				this,
			];
			this._goRefCounts = []; // number of references that Go has to a JS value, indexed by reference id
			this._ids = new Map();  // mapping from JS values to reference ids
			this._idPool = [];      // unused ids that have been garbage collected
			this.exited = false;    // whether the Go program has exited

			// Pass command line arguments and environment variables to WebAssembly by writing them to the linear memory.
			let offset = 4096;

			const strPtr = (str) => {
				const ptr = offset;
				const bytes = encoder.encode(str + "\0");
				new Uint8Array(this.mem.buffer, offset, bytes.length).set(bytes);
				offset += bytes.length;
				if (offset % 8 !== 0) {
					offset += 8 - (offset % 8);
				}
				return ptr;
			};

			const argc = this.argv.length;

			const argvPtrs = [];
			this.argv.forEach((arg) => {
				argvPtrs.push(strPtr(arg));
			});
			argvPtrs.push(0);

			const keys = Object.keys(this.env).sort();
			keys.forEach((key) => {
				argvPtrs.push(strPtr(`${key}=${this.env[key]}`));
			});
			argvPtrs.push(0);

			const argv = offset;
			argvPtrs.forEach((ptr) => {
				this.mem.setUint32(offset, ptr, true);
				this.mem.setUint32(offset + 4, 0, true);
				offset += 8;
			});

			this._inst.exports.run(argc, argv);
			if (this.exited) {
				this._resolveExitPromise();
			}
			await this._exitPromise;
		}

		_resume() {
			if (this.exited) {
				throw new Error("Go program has already exited");
			}
			this._inst.exports.resume();
			if (this.exited) {
				this._resolveExitPromise();
			}
		}

		_makeFuncWrapper(id) {
			const go = this;
			return function () {
				const event = { id: id, this: this, args: arguments };
				go._pendingEvent = event;
				go._resume();
				return event.result;
			};
		}
	}

	if (
		global.require &&
		global.require.main === module &&
		global.process &&
		global.process.versions &&
		!global.process.versions.electron
	) {
		if (process.argv.length < 3) {
			console.error("usage: go_js_wasm_exec [wasm binary] [arguments]");
			process.exit(1);
		}

		const go = new Go();
		go.argv = process.argv.slice(2);
		go.env = Object.assign({ TMPDIR: require("os").tmpdir() }, process.env);
		go.exit = process.exit;
		WebAssembly.instantiate(fs.readFileSync(process.argv[2]), go.importObject).then((result) => {
			process.on("exit", (code) => { // Node.js exits if no event handler is pending
				if (code === 0 && !go.exited) {
					// deadlock, make Go print error and stack traces
					go._pendingEvent = { id: 0 };
					go._resume();
				}
			});
			return go.run(result.instance);
		}).catch((err) => {
			console.error(err);
			process.exit(1);
		});
	}
})();

2022/10,江端さんの忘備録

私は、必要に迫られて、本(主に専門書)を、自腹で購入しなければならないことが多いです。

I often have to purchase books (mostly technical books), out of necessity, out of my own pocket.

―― 特に、コラムのネタが切れている時は、もう何振り構っていられない

"Especially when I am out of column material, I can't pretend anymore"

といった感じです。

It is like that.

-----

江端:「で、ソシャゲで課金する奴と、専門書を購入する奴の間に、それほどの差はないような気がするんだけど、どうかな」

Ebata: "So, I don't see much difference between those who pay for online games and those who buy technical books.

嫁さん:「そうかな?」

Wife: "Really?"

江端:「『好きなことをやっている』という点において、また、他人から見た場合『自分に価値のできないことをやっている』という点においても共通だと思う」

Ebata: "I think we share the same point of 'doing what I love' and also 'doing what others cannot value' in the eyes of others."

嫁さん:「ソシャゲで課金を続けられたら『キレる』と思うけど、本の場合は、別に腹が立たないなぁ」

Wife: "I would get 'mad' if you spend money for a online game, but with books, I think that I don't"

江端:「ちなみに、私もそう思う ―― ただ、両者の違いを論理的に説明ができないんだよなぁ」

Ebata: "I agree, however I just can't logically explain the difference between the two."

嫁さん:「投資に対して、報酬があるから?」

Wife: "Reward for investment?"

江端:「(論旨が)弱い。ソシャゲでも『自分が楽しい』という報酬がある」

Ebata: "(The argument is) weak. Even in online games, there is a reward of 'it makes me enjoy'."

嫁さん:「では、その『楽しい』が、夫婦で共有できないから?」

Wife: "Because couples can't share 'the enjoy'?"

江端:「確かに、本は共有できるけど ―― でも、まだ"弱い"なぁ」

Ebata: "Sure, we can share books - but it is still "weak".

-----

で、これは、仮説の域を出ないのですけど、

So, this is a bit of a hypothetical.

『私が本を買う時は、大抵、精神的に追い込まれていて、顔色が悪くて、全く楽しそうに見えない』

'When I buy books, I'm usually mentally driven, pale, and don't look happy at all.'

というのが、一因ではないかなぁ、と考えています。

I think that this may be one of the reasons.