前几天参加了一家公司的面试(技术测试),跟大家共享一下。
个人的简历
我是2001年从国内大学毕业之后就直接来的日本,一直从事IT行业,20多年了。来日本之后工作经历大概是这样的。
- 日系,中资系中小软件公司。主要是派遣和社内开发。
- 日本 埃森哲 。主要是大客户的社内系统的维护保守,提案。也负责跟大连开发中心联系。
- 创业公司 CTO。创业期加入。公司整体规模从初期的5人发展到100人左右。开发团队也从初期1人(我自己)发展到20人。主要是 B2B 的开发,公司内部开发团队的组建,管理。
- 创业公司 CTO。主要负责 越南 的开发团队和日本社内的开发团队的管理,当然自己也在做开发。
- 创业公司 联合创始人。创业公司B出售给上市公司之后,公司内的开发团队和一部分的营业团队,带着一部分项目,独立出来成立了这个公司。日本这边大概有20多个人,越南那里也有20多号人。
这次想换工作的背景
- 跟他人一块成立的创业公司进展不是太顺利,合伙人是4个,有点时候也感觉有劲使不到一块去。对公司前景有点灰心。
由于这个原因,并且岁数也有点大了(40多了),就打算找一个稳定点的公司,老老实实上班了。
工资情况
可能好多朋友也对在日本上班的IT人的工资情况有兴趣,就把我的情况简单分享一下。
- 刚来日本的时候是年收入300万日元,后来在派遣公司换来换去大概从第4年的时候,年收入达到600万日元
- 在埃森哲的时候,因为加班多少对收入会有很大的影响,我当时是分配到了运用维护的项目, 项目预算 控制得很严,不像新开发项目有那么大的加班空余,所以年收入当时是700万到800万日元上下浮动
- 创业公司,刚进去的时候是年收入800万日元,到后来公司做大,开发团队也壮大之后,基本上2年左右上调一次。800万到1000万到1200万日元。在这个公司待了5年。
- 创业公司,刚进去的时候年收入是1450万日元,一直到现在。
大概就是这个情况了,简单总结一下就是,20岁后半达到年收入600万日元,30岁中间之前达到年收入800万日元,40岁之前达到年收入1400万日元这样的过程。但是也不是所有的人都是这个情况,我是后期(最近10年)一直做公司的CTO,带团队开发。所以个人感觉应该比正常的IT行业的开发人员(比如说编码的,做上流设计或者项目管理的)的收入高一些。
日本这几年创业公司风起,行情比以前好很多很多,所以开发人员比较好找工作。比如说800万年收入上下的工作就比以前好找多了,但是要找很高工资的话还得是,开发组的负责人,技术负责人(Tech Lead),或者 CTO (VPoE)才行。
单纯搞开发的话,达到2000万日元年收入以上的就比较少了,个别的一些公司(尤其是外企)能提供,但是都会要求很强悍的技术(在一个行业要有一定的知名度)和管理背景,人脉关系等等。
对方公司的情况
这次面试的是一家日本的超大银行(传统的金融行业)和 互联网金融 公司合资开的一个创业公司。具体提供的服务就不讲了。对方公司现在正在设计服务和组建开发团队。
这次的职位能够提供的薪水范围是900万日元到2000万日元(这个范围比较大,有可能有猫腻),但是因为是对方的负责人直接发过来的邀请邮件,所以就打算去试一下看看。
面试经过
第一次面试通过之后,二次面试之前来了个编程考试,让在二次面试之前把代码完成。第二次面试的时候会以完成的代码为话题来考核,然后会在第二面的时候,再现场进行一次技术框架设计的考核。
技术考试(Coding Test)
日语内容是这样的

技術課題テスト / 技术测试例题
大概意思就是
- 用户可以往系统里边登录银行交易的流水,但是累计金额不到超过1000,超过了1000的话就返回一个特定的代码(这个实际上讲的不是很清楚,实际上要返回什么代码,要去看他给的那个 Golang 开发的测试代码)
- 附送的代码里边有API测试用的 Go 代码,这个代码不允许修改。完成的API必须通过这个测试。
- 开发环境 使用 Docker -compose 搭建
- API开发不限开发语言
要开发的内容实际上很简单,平常也一直用docker-compose。倒也没什么问题。但是Go开发语言是第一次用,怎么执行测试文件也不知道。稍微焦虑了一些,额外花了一些时间。
先看一下对方给的代码,可以从 Github 上直接下载。
下载完之后的文件夹构成就是这样的
╰─ tree -L 1 ─╯
.
├── LICENSE
├── README.md
├── app
├── db
├── docker -compose.yml
├── go.mod
├── go.sum
└── main_test.go
2 directories, 6 files
1 | app | 放API代码的地方,开发语言不限。 |
2 | db | MySQL的创建表格和测试数据的脚本在里边 |
3 | docker-compose.yml | MySQL启动用的设定在里边 |
4 | go.mod / go.sum | Goalng测试代码的依存关系 |
5 | main_test.go | 测试API的代码,用Goalng编写的。 虽然不懂Golang,但是以前C, C+++, C#, Java 以前都用过,大概也能推测出来什么意思。
登录失败的时候:402: Payment Required 银行流水登录成功: 201: Created |
再看一下docker-compose.yml文件的内容
version: "3.8"
services:
db:
image: mysql:5.7
environment:
MYSQL_ALLOW_EMPTY_PASSWORD: 1
volumes:
- ./db:/docker-entrypoint-initdb.d
ports:
- 3306:3306
hostname: db
app:
build:
context: app
ports:
- 8888:8888
就是普普通通的一个docker-compose 设定文件,启动MySQL然后投入测试用的数据,然后再启动API服务。
我的任务就是开发这个API,然后放到那个app容器里边就好了。
说干就干,因为这7,8年都是用Ruby on Rails做开发,所以就用Rails来做这个API了。Rails里边也有一些专门的做API开发的Gem(比如说 grape ),但是这次的需求很简单,并没有提到那些东西,所以就打算用Rails的Controller+返回 JSON 数据来做了。
先把 docker- compose.yml文件按照设想的修改一下
version: "3.8"
services:
db:
image: mysql:5.7
environment:
MYSQL_ALLOW_EMPTY_PASSWORD: 1
volumes:
- ./db:/docker-entrypoint-initdb.d
ports:
- "3306:3306"
hostname: db
app:
build:
context: app
command: bash -c "rm -f tmp/pids/server.pid && bundle exec rails s -p 3000 -b '0.0.0.0'"
volumes:
- ./app:/app/
ports:
- "8888:3000"
depends_on:
- db
hostname: app
app_test:
build:
context: app_test
depends_on:
- app
volumes:
- ./app_test:/app_test/
tty: true
hostname: app_test
db就是那个MySQL了,这个是对方以前提前设定好的,不需要做任何改动。
app就是这次用Rails来开发API的那个容器。
app_test就是用来执行测试API的容器,因为对方给的是Golang的测试代码。我的本地机器上边并没有执行环境,也不打算污染自己的机器,所以就用docker-compose容器了。完事之后把Docker镜像都删掉就好了。
# 先把不要的东西都删掉
rm app/**
cd app
touch Gemfile
touch Gemfile.lock
touch entrypoint.sh
# 先这样就好,回头docker-compose up之后,通过docker-compose来操作
app容器的entrypoint.sh文件的内容,就是为了避免残留的文件影响Rails的启动。
#!/bin/bash
set -e
rm -f /app/tmp/pids/server.pid
exec "$@"
app容器的Dockerfile文件,这个也没什么特别。 ruby 的版本指定后,然后适当的安装需要的东西就好
FROM ruby:2.7.5
RUN apt-get update && apt-get install -y curl apt-transport-https wget && \
curl -sS | apt-key add - && \
echo "deb stable main" | tee /etc/apt/sources.list.d/yarn.list && \
apt-get update && apt-get install -y yarn
RUN apt-get update -qq && apt-get install -y nodejs yarn
RUN mkdir /app
WORKDIR /app
COPY Gemfile /app/Gemfile
COPY Gemfile.lock /app/Gemfile.lock
RUN bundle config set paht "vendor/bundle"
RUN bundle install
RUN yarn install --check-files
RUN bundle exec rails webpacker:compile
COPY entrypoint.sh /usr/bin/
RUN chmod +x /usr/bin/entrypoint.sh
ENTRYPOINT ["entrypoint.sh"]
EXPOSE 3000
CMD ["bundle", "exec", "rails", "server", "-b", "0.0.0.0"]
启动容器
docker-compose up -d
docker-compose ps
# 或者docker ps
# 会看到三个容器
rails安装
# 要注意的是不要用run,要用exec
docker-compose exec app bundle install
docker-compose exec app bundle exec rails new . --database=mysql --skip-test
docker-compose exec app bundle exec rails webpacker:install
docker-compose exec app bundle config set path 'vendor/bundle'
docker-compose exec app bundle install
# 重启容器
docker-compose stop app
docker-compose start app
# 有点时候不知道什么原因,一些设定或者改动的文件没有反应到容器的话,就把容器删掉重新启动就好
docker-compose down
docker-compose up -d
调整Gemfile文件,把这次要用的Gem都放进去
source '#39;
git_source(:github) { |repo| "#{repo}.git" }
ruby '2.7.5'
# Bundle edge Rails instead: gem 'rails', github: 'rails/rails', branch: 'main'
gem 'rails', '~> 6.1.5'
# Use mysql as the database for Active Record
gem 'mysql2', '~> 0.5'
# Use Puma as the app server
gem 'puma', '~> 5.0'
# Use SCSS for stylesheets
gem 'sass-rails', '>= 6'
# Transpile app-like JavaScript. Read more:
gem 'webpacker', '~> 5.0'
# Turbolinks makes navigating your web application faster. Read more:
gem 'turbolinks', '~> 5'
# Build JSON APIs with ease. Read more:
gem 'jbuilder', '~> 2.7'
# Use Redis adapter to run Action Cable in production
# gem 'redis', '~> 4.0'
# Use Active Model has_secure_password
# gem 'bcrypt', '~> 3.1.7'
# Use Active Storage variant
# gem 'image_processing', '~> 1.2'
# Reduces boot times through caching; required in config/boot.rb
gem 'bootsnap', '>= 1.4.4', require: false
gem 'devise'
group :development, :test do
# Call 'byebug' anywhere in the code to stop execution and get a debugger console
gem 'byebug', platforms: [:mri, :mingw, :x64_mingw]
gem "brakeman", require: false
gem "bundler-audit"
gem "rails_best_practices"
gem "rubocop", require: false
gem 'rubocop-airbnb', require: false
gem "rubocop-performance", require: false
gem "rubocop-rails", require: false
end
group :development do
gem "bullet"
# Access an interactive console on exception pages or by calling 'console' anywhere in the code.
gem 'web-console', '>= 4.1.0'
# Display performance information such as SQL time and flame graphs for each request in your browser.
# Can be configured to work on production as well see:
gem 'rack-mini-profiler', '~> 2.0'
gem 'listen', '~> 3.3'
# Spring speeds up development by keeping your application running in the background. Read more:
gem 'spring'
end
# Windows does not include zoneinfo files, so bundle the tzinfo-data gem
gem 'tzinfo-data', platforms: [:mingw, :mswin, :x64_mingw, :jruby]
主要加了这几个东西
- devise
- brakeman
- bundler-audit
- rails_best_practices
- rubocop
- rubocop-airbnb
- rubocop-performance
- rubocop-rails
因为跟容器是文件夹绑定的,所以在容器外边用别的编辑器直接修改就好。完事后再安装那些Gem。这次编写代码的时候用的是 vs code。但是我平常做公司项目的时候都是用 Rubymine 的
docker-compose exec app bundle install
环境都准备好了,现在就写代码就好了
主要的代码是下边的部分
- model
- controller
- views(json输出)
用户(user.rb)
# frozen_string_literal: true
class User < ApplicationRecord
has_many :transactions
end
流水(transaction.rb)
# frozen_string_literal: true
class Transaction < ApplicationRecord
TOTAL_AMOUNT_LIMIT = 1000
belongs_to :user
validates :amount,
presence: true,
numericality: {
only_integer: true,
greater_than: 0,
less_than_or_equal_to: Transaction::TOTAL_AMOUNT_LIMIT,
allow_nil: true,
}
validates :description, presence: true, length: { maximum: 256 }
validate :total_amount_must_less_than_or_equal_to_limit
private
def total_amount_must_less_than_or_equal_to_limit
return if errors[:amount].present?
permitted_total_amount = user.transactions.sum(:amount)
return if permitted_total_amount + amount <= Transaction::TOTAL_AMOUNT_LIMIT
errors.add :amount, :limit_exceeded, limit: Transaction::TOTAL_AMOUNT_LIMIT
end
end
transactions_controller.rb
这个部分的实现,需要先看一下那个测试代码,要看一下他是怎么把认证代码(api_key)传过来的。
从这里能看到,是用Header来传的。

go lang test code / api key
从这里能看到传过来的参数的名字。

go lang test code / json
# frozen_string_literal: true
class TransactionsController < ApplicationController
skip_before_action :verify_authenticity_token
before_action :load_user_from_api_key!
before_action :check_user_id_and_api_key!
layout false
def create
@transaction = @user.transactions.new(permitted_transaction_params.except(:user_id))
# 因为这次仅仅是个代码考试,并没有提到高并发性的时候的锁定性能问题。
# 所以没有用Redis setnx之类的,就只是用了Rails的记录锁定
@user.with_lock do
if @transaction.save
@balance = @user.transactions.where(id: -Float::INFINITY..@transaction.id).sum(:amount)
render status: :created
else
render status: :payment_required
end
end
end
private
def permitted_transaction_params
params.require(:transaction).permit(:user_id, :amount, :description)
end
# 从Request头里边取出api_key,然后去查找用户
def load_user_from_api_key!
api_key = request.headers['apikey'].presence
return render body: nil, status: :forbidden if api_key.nil?
@user = User.find_by(api_key: api_key)
render body: nil, status: :forbidden if @user.nil?
end
# 检查用api_key查找到的用户是否也参数中的用户匹配
def check_user_id_and_api_key!
# 好多朋友都知道,两个字符串的比较是不安全的,所以用Devise提供的这个函数来比较
return if Devise.secure_compare(@user.id.to_s, permitted_transaction_params[:user_id])
render body: nil, status: :forbidden
end
end
API的返回结果(transactions/create.json.jbuilder)
# frozen_string_literal: true
if @transaction.errors.empty?
json.transaction do
json.extract! @transaction, :amount
json.balance @balance
end
else
json.transaction do
json.errors do
json.array! @transaction.errors.full_messages
end
end
end
最后把路径设定一下(routes.rb)
Rails.application.routes.draw do
# For details on the DSL available within this file, see
resources :transactions, only: [:create]
end
databases.yml需要适当调整一下
- host: db
- database: 用你的这次的数据库的名字
这样的话,Rails程序就应该动起来了。
自己机器上并没有Golang环境,也不想安装,就用docker-compose了。因为第一次用Golang,怎么执行文件也不知道,查了半天。
最终的成品是这样的。除了Dockfile之外,剩下的都是他们提供的。main_test.go文件稍微改动了一些。DB和API的链接地址(因为是用Docker容器的)
把原先的127.0.0.1改成对应的Docker容器服务的名字

go lang test code
╰─ tree -L 1 ./app_test ─╯
./app_test
├── Dockerfile
├── go.mod
├── go.sum
└── main_test.go
Dockerfile的内容,很简单。Golang容器启动
FROM golang:1.16
RUN mkdir /app_test
WORKDIR /app_test
CMD [ "/bin/sh" ]
现在大功告成,进去Golang的容器执行测试就好了。
# 先检查一下代码里边有没有问题
docker-compose exec app bundle exec bundle-audit check --update
docker-compose exec app bundle exec brakeman
docker-compose exec app bundle exec rubocop
docker-compose exec app bundle exec rails_best_practices .
docker ps
# 找到app_test的容器ID
docker exec -it 容器ID bash
# 这样就进入到那个Golang的容器了
root@app_test:/app_test#
# 执行测试就好
root@app_test:/app_test# go version
go version go1.16.15 linux/amd64
root@app_test:/app_test# go test -run TestCreate
go: downloading github.com/go-sql-driver/mysql v1.6.0
PASS
ok github.com/........2.021s
root@app_test:/app_test# go test -v -run TestCreate
=== RUN TestCreate
main_test.go:69: {"transaction":{"amount":100,"balance":100}}
main_test.go:69: {"transaction":{"amount":100,"balance":100}}
main_test.go:69: {"transaction":{"amount":100,"balance":200}}
main_test.go:69: {"transaction":{"amount":100,"balance":200}}
main_test.go:69: {"transaction":{"amount":100,"balance":300}}
main_test.go:69: {"transaction":{"amount":100,"balance":300}}
main_test.go:69: {"transaction":{"amount":100,"balance":400}}
main_test.go:69: {"transaction":{"amount":100,"balance":400}}
main_test.go:69: {"transaction":{"amount":100,"balance":500}}
main_test.go:69: {"transaction":{"amount":100,"balance":500}}
main_test.go:69: {"transaction":{"amount":100,"balance":600}}
main_test.go:69: {"transaction":{"amount":100,"balance":600}}
main_test.go:69: {"transaction":{"amount":100,"balance":700}}
main_test.go:69: {"transaction":{"amount":100,"balance":700}}
main_test.go:69: {"transaction":{"amount":100,"balance":800}}
main_test.go:69: {"transaction":{"amount":100,"balance":800}}
main_test.go:69: {"transaction":{"amount":100,"balance":900}}
main_test.go:69: {"transaction":{"amount":100,"balance":1000}}
main_test.go:69: {"transaction":{"amount":100,"balance":900}}
main_test.go:69: {"transaction":{"amount":100,"balance":1000}}
main_test.go:69: {"transaction":{"errors":["Amount Transaction limit(1000) exceeded"]}}
main_test.go:69: {"transaction":{"errors":["Amount Transaction limit(1000) exceeded"]}}
main_test.go:69: {"transaction":{"errors":["Amount Transaction limit(1000) exceeded"]}}
main_test.go:69: {"transaction":{"errors":["Amount Transaction limit(1000) exceeded"]}}
main_test.go:87: 1 1000
main_test.go:87: 2 1000
--- PASS: TestCreate (1.29s)
PASS
ok github.com/........1.297s

go lang test container
大功告成!!!
再简单分享一下我用到的两个工具,vs code 和 iTerm2 的截图。

vs code / github plus theme

iTerm2 / powerlevel10k